mirror of
				https://github.com/TeamNewPipe/NewPipe
				synced 2025-10-25 04:17:39 +00:00 
			
		
		
		
	Merge branch 'dev' into pr8221
This commit is contained in:
		
							
								
								
									
										163
									
								
								app/build.gradle
									
									
									
									
									
								
							
							
						
						
									
										163
									
								
								app/build.gradle
									
									
									
									
									
								
							| @@ -1,28 +1,29 @@ | ||||
| import com.android.tools.profgen.ArtProfileKt | ||||
| import com.android.tools.profgen.ArtProfileSerializer | ||||
| import com.android.tools.profgen.DexFile | ||||
|  | ||||
| plugins { | ||||
|     id "com.android.application" | ||||
|     id "kotlin-android" | ||||
|     id "kotlin-kapt" | ||||
|     id "kotlin-parcelize" | ||||
|     id "checkstyle" | ||||
|     id "org.sonarqube" version "3.3" | ||||
|     id "org.sonarqube" version "4.0.0.2929" | ||||
| } | ||||
|  | ||||
| android { | ||||
|     compileSdk 31 | ||||
|     buildToolsVersion '31.0.0' | ||||
|     compileSdk 34 | ||||
|     namespace 'org.schabi.newpipe' | ||||
|  | ||||
|     defaultConfig { | ||||
|         applicationId "org.schabi.newpipe" | ||||
|         resValue "string", "app_name", "NewPipe" | ||||
|         minSdk 19 | ||||
|         targetSdk 29 | ||||
|         versionCode 984 | ||||
|         versionName "0.22.1" | ||||
|  | ||||
|         multiDexEnabled true | ||||
|         minSdk 21 | ||||
|         targetSdk 33 | ||||
|         versionCode 996 | ||||
|         versionName "0.26.1" | ||||
|  | ||||
|         testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner" | ||||
|         vectorDrawables.useSupportLibrary = true | ||||
|  | ||||
|         javaCompileOptions { | ||||
|             annotationProcessorOptions { | ||||
| @@ -49,9 +50,6 @@ android { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Keep the release build type at the end of the list to override 'archivesBaseName' of | ||||
|         // debug build. This seems to be a Gradle bug, therefore | ||||
|         // TODO: update Gradle version | ||||
|         release { | ||||
|             if (System.properties.containsKey('packageSuffix')) { | ||||
|                 applicationIdSuffix System.getProperty('packageSuffix') | ||||
| @@ -79,13 +77,13 @@ android { | ||||
|         // Flag to enable support for the new language APIs | ||||
|         coreLibraryDesugaringEnabled true | ||||
|  | ||||
|         sourceCompatibility JavaVersion.VERSION_11 | ||||
|         targetCompatibility JavaVersion.VERSION_11 | ||||
|         sourceCompatibility JavaVersion.VERSION_17 | ||||
|         targetCompatibility JavaVersion.VERSION_17 | ||||
|         encoding 'utf-8' | ||||
|     } | ||||
|  | ||||
|     kotlinOptions { | ||||
|         jvmTarget = JavaVersion.VERSION_11 | ||||
|         jvmTarget = JavaVersion.VERSION_17 | ||||
|     } | ||||
|  | ||||
|     sourceSets { | ||||
| @@ -95,25 +93,33 @@ android { | ||||
|     buildFeatures { | ||||
|         viewBinding true | ||||
|     } | ||||
|  | ||||
|     packagingOptions { | ||||
|         resources { | ||||
|             // remove two files which belong to jsoup | ||||
|             // no idea how they ended up in the META-INF dir... | ||||
|             excludes += ['META-INF/README.md', 'META-INF/CHANGES', | ||||
|                          // 'COPYRIGHT' belongs to RxJava... | ||||
|                          'META-INF/COPYRIGHT'] | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| ext { | ||||
|     checkstyleVersion = '10.0' | ||||
|     checkstyleVersion = '10.12.1' | ||||
|  | ||||
|     androidxLifecycleVersion = '2.3.1' | ||||
|     androidxRoomVersion = '2.4.2' | ||||
|     androidxWorkVersion = '2.7.1' | ||||
|     androidxLifecycleVersion = '2.6.2' | ||||
|     androidxRoomVersion = '2.6.1' | ||||
|     androidxWorkVersion = '2.8.1' | ||||
|  | ||||
|     icepickVersion = '3.2.0' | ||||
|     exoPlayerVersion = '2.17.1' | ||||
|     googleAutoServiceVersion = '1.0.1' | ||||
|     groupieVersion = '2.10.0' | ||||
|     exoPlayerVersion = '2.18.7' | ||||
|     googleAutoServiceVersion = '1.1.1' | ||||
|     groupieVersion = '2.10.1' | ||||
|     markwonVersion = '4.6.2' | ||||
|  | ||||
|     leakCanaryVersion = '2.5' | ||||
|     leakCanaryVersion = '2.12' | ||||
|     stethoVersion = '1.6.0' | ||||
|     mockitoVersion = '4.0.0' | ||||
|     assertJVersion = '3.22.0' | ||||
| } | ||||
|  | ||||
| configurations { | ||||
| @@ -128,7 +134,7 @@ checkstyle { | ||||
|     toolVersion = checkstyleVersion | ||||
| } | ||||
|  | ||||
| task runCheckstyle(type: Checkstyle) { | ||||
| tasks.register('runCheckstyle', Checkstyle) { | ||||
|     source 'src' | ||||
|     include '**/*.java' | ||||
|     exclude '**/gen/**' | ||||
| @@ -149,20 +155,22 @@ task runCheckstyle(type: Checkstyle) { | ||||
| def outputDir = "${project.buildDir}/reports/ktlint/" | ||||
| def inputFiles = project.fileTree(dir: "src", include: "**/*.kt") | ||||
|  | ||||
| task runKtlint(type: JavaExec) { | ||||
| tasks.register('runKtlint', JavaExec) { | ||||
|     inputs.files(inputFiles) | ||||
|     outputs.dir(outputDir) | ||||
|     getMainClass().set("com.pinterest.ktlint.Main") | ||||
|     classpath = configurations.ktlint | ||||
|     args "src/**/*.kt" | ||||
|     jvmArgs("--add-opens", "java.base/java.lang=ALL-UNNAMED") | ||||
| } | ||||
|  | ||||
| task formatKtlint(type: JavaExec) { | ||||
| tasks.register('formatKtlint', JavaExec) { | ||||
|     inputs.files(inputFiles) | ||||
|     outputs.dir(outputDir) | ||||
|     getMainClass().set("com.pinterest.ktlint.Main") | ||||
|     classpath = configurations.ktlint | ||||
|     args "-F", "src/**/*.kt" | ||||
|     jvmArgs("--add-opens", "java.base/java.lang=ALL-UNNAMED") | ||||
| } | ||||
|  | ||||
| afterEvaluate { | ||||
| @@ -172,7 +180,7 @@ afterEvaluate { | ||||
|     preDebugBuild.dependsOn runCheckstyle, runKtlint | ||||
| } | ||||
|  | ||||
| sonarqube { | ||||
| sonar { | ||||
|     properties { | ||||
|         property "sonar.projectKey", "TeamNewPipe_NewPipe" | ||||
|         property "sonar.organization", "teamnewpipe" | ||||
| @@ -182,7 +190,7 @@ sonarqube { | ||||
|  | ||||
| dependencies { | ||||
| /** Desugaring **/ | ||||
|     coreLibraryDesugaring 'com.android.tools:desugar_jdk_libs:1.1.5' | ||||
|     coreLibraryDesugaring 'com.android.tools:desugar_jdk_libs_nio:2.0.4' | ||||
|  | ||||
| /** NewPipe libraries **/ | ||||
|     // You can use a local version by uncommenting a few lines in settings.gradle | ||||
| @@ -190,41 +198,39 @@ dependencies { | ||||
|     // name and the commit hash with the commit hash of the (pushed) commit you want to test | ||||
|     // This works thanks to JitPack: https://jitpack.io/ | ||||
|     implementation 'com.github.TeamNewPipe:nanojson:1d9e1aea9049fc9f85e68b43ba39fe7be1c1f751' | ||||
|     implementation 'com.github.TeamNewPipe:NewPipeExtractor:5a1873084' | ||||
|     implementation 'com.github.Stypox:NewPipeExtractor:aaf3231fc75d7b4177549fec4aa7e672bfe84015' | ||||
|     implementation 'com.github.TeamNewPipe:NoNonsense-FilePicker:5.0.0' | ||||
|  | ||||
| /** Checkstyle **/ | ||||
|     checkstyle "com.puppycrawl.tools:checkstyle:${checkstyleVersion}" | ||||
|     ktlint 'com.pinterest:ktlint:0.44.0' | ||||
|     ktlint 'com.pinterest:ktlint:0.45.2' | ||||
|  | ||||
| /** Kotlin **/ | ||||
|     implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8:${kotlin_version}" | ||||
|     implementation "org.jetbrains.kotlin:kotlin-stdlib:${kotlin_version}" | ||||
|  | ||||
| /** AndroidX **/ | ||||
|     implementation 'androidx.appcompat:appcompat:1.3.1' | ||||
|     implementation 'androidx.appcompat:appcompat:1.6.1' | ||||
|     implementation 'androidx.cardview:cardview:1.0.0' | ||||
|     implementation 'androidx.constraintlayout:constraintlayout:2.1.3' | ||||
|     implementation 'androidx.core:core-ktx:1.6.0' | ||||
|     implementation 'androidx.constraintlayout:constraintlayout:2.1.4' | ||||
|     implementation 'androidx.core:core-ktx:1.12.0' | ||||
|     implementation 'androidx.documentfile:documentfile:1.0.1' | ||||
|     implementation 'androidx.fragment:fragment-ktx:1.3.6' | ||||
|     implementation "androidx.lifecycle:lifecycle-livedata:${androidxLifecycleVersion}" | ||||
|     implementation "androidx.lifecycle:lifecycle-viewmodel:${androidxLifecycleVersion}" | ||||
|     implementation 'androidx.fragment:fragment-ktx:1.6.2' | ||||
|     implementation "androidx.lifecycle:lifecycle-livedata-ktx:${androidxLifecycleVersion}" | ||||
|     implementation "androidx.lifecycle:lifecycle-viewmodel-ktx:${androidxLifecycleVersion}" | ||||
|     implementation 'androidx.localbroadcastmanager:localbroadcastmanager:1.1.0' | ||||
|     implementation 'androidx.media:media:1.5.0' | ||||
|     implementation 'androidx.multidex:multidex:2.0.1' | ||||
|     implementation 'androidx.preference:preference:1.2.0' | ||||
|     implementation 'androidx.recyclerview:recyclerview:1.2.1' | ||||
|     implementation 'androidx.media:media:1.7.0' | ||||
|     implementation 'androidx.preference:preference:1.2.1' | ||||
|     implementation 'androidx.recyclerview:recyclerview:1.3.2' | ||||
|     implementation "androidx.room:room-runtime:${androidxRoomVersion}" | ||||
|     implementation "androidx.room:room-rxjava3:${androidxRoomVersion}" | ||||
|     kapt "androidx.room:room-compiler:${androidxRoomVersion}" | ||||
|     implementation 'androidx.swiperefreshlayout:swiperefreshlayout:1.1.0' | ||||
|     // Newer version specified to prevent accessibility regressions with RecyclerView, see: | ||||
|     // https://developer.android.com/jetpack/androidx/releases/viewpager2#1.1.0-alpha01 | ||||
|     implementation 'androidx.viewpager2:viewpager2:1.1.0-beta01' | ||||
|     implementation 'androidx.webkit:webkit:1.4.0' | ||||
|     implementation 'com.google.android.material:material:1.5.0' | ||||
|     implementation "androidx.work:work-runtime:${androidxWorkVersion}" | ||||
|     implementation 'androidx.viewpager2:viewpager2:1.1.0-beta02' | ||||
|     implementation "androidx.work:work-runtime-ktx:${androidxWorkVersion}" | ||||
|     implementation "androidx.work:work-rxjava3:${androidxWorkVersion}" | ||||
|     implementation 'com.google.android.material:material:1.11.0' | ||||
|  | ||||
| /** Third-party libraries **/ | ||||
|     // Instance state boilerplate elimination | ||||
| @@ -232,14 +238,19 @@ dependencies { | ||||
|     kapt "frankiesardo:icepick-processor:${icepickVersion}" | ||||
|  | ||||
|     // HTML parser | ||||
|     implementation "org.jsoup:jsoup:1.14.3" | ||||
|     implementation "org.jsoup:jsoup:1.17.2" | ||||
|  | ||||
|     // HTTP client | ||||
|     //noinspection GradleDependency --> do not update okhttp to keep supporting Android 4.4 users | ||||
|     implementation "com.squareup.okhttp3:okhttp:3.12.13" | ||||
|     implementation "com.squareup.okhttp3:okhttp:4.12.0" | ||||
|  | ||||
|     // Media player | ||||
|     implementation "com.google.android.exoplayer:exoplayer:${exoPlayerVersion}" | ||||
|     implementation "com.google.android.exoplayer:exoplayer-core:${exoPlayerVersion}" | ||||
|     implementation "com.google.android.exoplayer:exoplayer-dash:${exoPlayerVersion}" | ||||
|     implementation "com.google.android.exoplayer:exoplayer-database:${exoPlayerVersion}" | ||||
|     implementation "com.google.android.exoplayer:exoplayer-datasource:${exoPlayerVersion}" | ||||
|     implementation "com.google.android.exoplayer:exoplayer-hls:${exoPlayerVersion}" | ||||
|     implementation "com.google.android.exoplayer:exoplayer-smoothstreaming:${exoPlayerVersion}" | ||||
|     implementation "com.google.android.exoplayer:exoplayer-ui:${exoPlayerVersion}" | ||||
|     implementation "com.google.android.exoplayer:extension-mediasession:${exoPlayerVersion}" | ||||
|  | ||||
|     // Metadata generator for service descriptors | ||||
| @@ -258,42 +269,38 @@ dependencies { | ||||
|     implementation "io.noties.markwon:core:${markwonVersion}" | ||||
|     implementation "io.noties.markwon:linkify:${markwonVersion}" | ||||
|  | ||||
|     // File picker | ||||
|     implementation "com.nononsenseapps:filepicker:4.2.1" | ||||
|  | ||||
|     // Crash reporting | ||||
|     implementation "ch.acra:acra-core:5.8.4" | ||||
|     implementation "ch.acra:acra-core:5.11.3" | ||||
|  | ||||
|     // Properly restarting | ||||
|     implementation 'com.jakewharton:process-phoenix:2.1.2' | ||||
|  | ||||
|     // Reactive extensions for Java VM | ||||
|     implementation "io.reactivex.rxjava3:rxjava:3.0.13" | ||||
|     implementation "io.reactivex.rxjava3:rxandroid:3.0.0" | ||||
|     implementation "io.reactivex.rxjava3:rxjava:3.1.8" | ||||
|     implementation "io.reactivex.rxjava3:rxandroid:3.0.2" | ||||
|     // RxJava binding APIs for Android UI widgets | ||||
|     implementation "com.jakewharton.rxbinding4:rxbinding:4.0.0" | ||||
|  | ||||
|     // Date and time formatting | ||||
|     implementation "org.ocpsoft.prettytime:prettytime:5.0.2.Final" | ||||
|     implementation "org.ocpsoft.prettytime:prettytime:5.0.7.Final" | ||||
|  | ||||
| /** Debugging **/ | ||||
|     // Memory leak detection | ||||
|     implementation "com.squareup.leakcanary:leakcanary-object-watcher-android:${leakCanaryVersion}" | ||||
|     implementation "com.squareup.leakcanary:plumber-android:${leakCanaryVersion}" | ||||
|     debugImplementation "com.squareup.leakcanary:leakcanary-android:${leakCanaryVersion}" | ||||
|     debugImplementation "com.squareup.leakcanary:leakcanary-object-watcher-android:${leakCanaryVersion}" | ||||
|     debugImplementation "com.squareup.leakcanary:plumber-android:${leakCanaryVersion}" | ||||
|     debugImplementation "com.squareup.leakcanary:leakcanary-android-core:${leakCanaryVersion}" | ||||
|     // Debug bridge for Android | ||||
|     debugImplementation "com.facebook.stetho:stetho:${stethoVersion}" | ||||
|     debugImplementation "com.facebook.stetho:stetho-okhttp3:${stethoVersion}" | ||||
|  | ||||
| /** Testing **/ | ||||
|     testImplementation 'junit:junit:4.13.2' | ||||
|     testImplementation "org.mockito:mockito-core:${mockitoVersion}" | ||||
|     testImplementation "org.mockito:mockito-inline:${mockitoVersion}" | ||||
|     testImplementation 'org.mockito:mockito-core:5.6.0' | ||||
|  | ||||
|     androidTestImplementation "androidx.test.ext:junit:1.1.3" | ||||
|     androidTestImplementation "androidx.test:runner:1.4.0" | ||||
|     androidTestImplementation "androidx.test.ext:junit:1.1.5" | ||||
|     androidTestImplementation "androidx.test:runner:1.5.2" | ||||
|     androidTestImplementation "androidx.room:room-testing:${androidxRoomVersion}" | ||||
|     androidTestImplementation "org.assertj:assertj-core:${assertJVersion}" | ||||
|     androidTestImplementation "org.assertj:assertj-core:3.24.2" | ||||
| } | ||||
|  | ||||
| static String getGitWorkingBranch() { | ||||
| @@ -311,3 +318,25 @@ static String getGitWorkingBranch() { | ||||
|         return "" | ||||
|     } | ||||
| } | ||||
|  | ||||
| // fix reproducible builds | ||||
| project.afterEvaluate { | ||||
|     tasks.compileReleaseArtProfile.doLast { | ||||
|         outputs.files.each { file -> | ||||
|             if (file.toString().endsWith(".profm")) { | ||||
|                 println("Sorting ${file} ...") | ||||
|                 def version = ArtProfileSerializer.valueOf("METADATA_0_0_2") | ||||
|                 def profile = ArtProfileKt.ArtProfile(file) | ||||
|                 def keys = new ArrayList(profile.profileData.keySet()) | ||||
|                 def sortedData = new LinkedHashMap() | ||||
|                 Collections.sort keys, new DexFile.Companion() | ||||
|                 keys.each { key -> sortedData[key] = profile.profileData[key] } | ||||
|                 new FileOutputStream(file).with { | ||||
|                     write(version.magicBytes$profgen) | ||||
|                     write(version.versionBytes$profgen) | ||||
|                     version.write$profgen(it, sortedData, "") | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										44
									
								
								app/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										44
									
								
								app/proguard-rules.pro
									
									
									
									
										vendored
									
									
								
							| @@ -1,36 +1,18 @@ | ||||
| # Add project specific ProGuard rules here. | ||||
| # By default, the flags in this file are appended to flags specified | ||||
| # in /home/the-scrabi/bin/Android/Sdk/tools/proguard/proguard-android.txt | ||||
| # You can edit the include path and order by changing the proguardFiles | ||||
| # directive in build.gradle. | ||||
| # | ||||
| # For more details, see | ||||
| #   http://developer.android.com/guide/developing/tools/proguard.html | ||||
|  | ||||
| # Add any project specific keep options here: | ||||
|  | ||||
| # If your project uses WebView with JS, uncomment the following | ||||
| # and specify the fully qualified class name to the JavaScript interface | ||||
| # class: | ||||
| #-keepclassmembers class fqcn.of.javascript.interface.for.webview { | ||||
| #   public *; | ||||
| #} | ||||
| # https://developer.android.com/build/shrink-code | ||||
|  | ||||
| ## Helps debug release versions | ||||
| -dontobfuscate | ||||
|  | ||||
| ## Rules for NewPipeExtractor | ||||
| -keep class org.schabi.newpipe.extractor.timeago.patterns.** { *; } | ||||
| -keep class org.ocpsoft.prettytime.i18n.** { *; } | ||||
|  | ||||
| -keep class org.mozilla.javascript.** { *; } | ||||
|  | ||||
| -keep class org.mozilla.classfile.ClassFileWriter | ||||
| -dontwarn org.mozilla.javascript.tools.** | ||||
|  | ||||
| ## Rules for ExoPlayer | ||||
| -keep class com.google.android.exoplayer2.** { *; } | ||||
|  | ||||
| -dontwarn org.mozilla.javascript.tools.** | ||||
| -dontwarn android.arch.util.paging.CountedDataSource | ||||
| -dontwarn android.arch.persistence.room.paging.LimitOffsetDataSource | ||||
|  | ||||
|  | ||||
| # Rules for icepick. Copy paste from https://github.com/frankiesardo/icepick | ||||
| ## Rules for Icepick. Copy pasted from https://github.com/frankiesardo/icepick | ||||
| -dontwarn icepick.** | ||||
| -keep class icepick.** { *; } | ||||
| -keep class **$$Icepick { *; } | ||||
| @@ -39,15 +21,17 @@ | ||||
| } | ||||
| -keepnames class * { @icepick.State *;} | ||||
|  | ||||
| # Rules for OkHttp. Copy paste from https://github.com/square/okhttp | ||||
| ## Rules for OkHttp. Copy pasted from https://github.com/square/okhttp | ||||
| -dontwarn okhttp3.** | ||||
| -dontwarn okio.** | ||||
| -dontwarn javax.annotation.** | ||||
| # A resource is loaded with a relative path so the package of this class must be preserved. | ||||
| -keepnames class okhttp3.internal.publicsuffix.PublicSuffixDatabase | ||||
|  | ||||
| ## See https://github.com/TeamNewPipe/NewPipe/pull/1441 | ||||
| -keepclassmembers class * implements java.io.Serializable { | ||||
|     static final long serialVersionUID; | ||||
|     !static !transient <fields>; | ||||
|     private void writeObject(java.io.ObjectOutputStream); | ||||
|     private void readObject(java.io.ObjectInputStream); | ||||
| } | ||||
|  | ||||
| ## For some reason NotificationModeConfigFragment wasn't kept (only referenced in a preference xml) | ||||
| -keep class org.schabi.newpipe.settings.notifications.** { *; } | ||||
|   | ||||
							
								
								
									
										19
									
								
								app/sampledata/channels.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								app/sampledata/channels.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| { | ||||
|   "data": [ | ||||
|     { | ||||
|       "name": "BBC", | ||||
|       "additional": "12K subscribers•233 videos", | ||||
|       "description": "The BBC is the world’s leading public service broadcaster. We’re impartial and independent, and every day we create distinctive, world-class programmes and content which inform, educate and entertain millions of people in the UK and around the world. SUBSCRIBE to our YouTube channel to get the best of BBC entertainment and comedy programmes, stories from science and nature documentaries, and much more! https://bit.ly/2IXqEIn Get ALL your fresh TV, and sofa-hugging box sets on iPlayer https://bbc.in/2J18jYJ" | ||||
|     }, | ||||
|     { | ||||
|       "name": "Linus Tech Tips", | ||||
|       "additional": "1M subscribers•233 videos", | ||||
|       "description": "Looking for a Tech YouTuber?\n\nLinus Tech Tips is a passionate team of \"professionally curious\" experts in consumer technology and video production which aims to inform and educate people of all ages through our entertaining videos. We create product reviews, step-by-step computer build guides, and a variety of other tech-focused content.\n\nSchedule:\nNew videos every Saturday to Thursday @ 10:00am Pacific\nLive WAN Show podcasts every Friday @ ~5:00pm Pacific" | ||||
|     }, | ||||
|     { | ||||
|       "name": "Marques Brownlee", | ||||
|       "additional": "13 subscribers•12K videos", | ||||
|       "description": "MKBHD: Quality Tech Videos | YouTuber | Geek | Consumer Electronics | Tech Head | Internet Personality!\n\nbusiness@MKBHD.com\n\nNYC" | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| @@ -2,7 +2,7 @@ | ||||
|   "formatVersion": 1, | ||||
|   "database": { | ||||
|     "version": 6, | ||||
|     "identityHash": "9ffc14521c566beed378d77430de3f0c", | ||||
|     "identityHash": "4084aa342aef315dc7b558770a7755a9", | ||||
|     "entities": [ | ||||
|       { | ||||
|         "tableName": "subscriptions", | ||||
| @@ -323,7 +323,7 @@ | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "playlists", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT, `thumbnail_url` TEXT, `display_index` INTEGER NOT NULL)", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT, `thumbnail_url` TEXT, `is_thumbnail_permanent` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
| @@ -344,8 +344,8 @@ | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "displayIndex", | ||||
|             "columnName": "display_index", | ||||
|             "fieldPath": "isThumbnailPermanent", | ||||
|             "columnName": "is_thumbnail_permanent", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
| @@ -447,7 +447,7 @@ | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "remote_playlists", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `name` TEXT, `url` TEXT, `thumbnail_url` TEXT, `uploader` TEXT, `display_index` INTEGER NOT NULL, `stream_count` INTEGER)", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `name` TEXT, `url` TEXT, `thumbnail_url` TEXT, `uploader` TEXT, `stream_count` INTEGER)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
| @@ -485,12 +485,6 @@ | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "displayIndex", | ||||
|             "columnName": "display_index", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamCount", | ||||
|             "columnName": "stream_count", | ||||
| @@ -737,7 +731,7 @@ | ||||
|     "views": [], | ||||
|     "setupQueries": [ | ||||
|       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||
|       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, '9ffc14521c566beed378d77430de3f0c')" | ||||
|       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, '4084aa342aef315dc7b558770a7755a9')" | ||||
|     ] | ||||
|   } | ||||
| } | ||||
							
								
								
									
										737
									
								
								app/schemas/org.schabi.newpipe.database.AppDatabase/7.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										737
									
								
								app/schemas/org.schabi.newpipe.database.AppDatabase/7.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,737 @@ | ||||
| { | ||||
|   "formatVersion": 1, | ||||
|   "database": { | ||||
|     "version": 7, | ||||
|     "identityHash": "012fc8e7ad3333f1597347f34e76a513", | ||||
|     "entities": [ | ||||
|       { | ||||
|         "tableName": "subscriptions", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `url` TEXT, `name` TEXT, `avatar_url` TEXT, `subscriber_count` INTEGER, `description` TEXT, `notification_mode` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "avatarUrl", | ||||
|             "columnName": "avatar_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriberCount", | ||||
|             "columnName": "subscriber_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "description", | ||||
|             "columnName": "description", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "notificationMode", | ||||
|             "columnName": "notification_mode", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ], | ||||
|           "autoGenerate": true | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_subscriptions_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_subscriptions_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "search_history", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`creation_date` INTEGER, `service_id` INTEGER NOT NULL, `search` TEXT, `id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "creationDate", | ||||
|             "columnName": "creation_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "search", | ||||
|             "columnName": "search", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "id", | ||||
|             "columnName": "id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "id" | ||||
|           ], | ||||
|           "autoGenerate": true | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_search_history_search", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "search" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_search_history_search` ON `${TABLE_NAME}` (`search`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "streams", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `url` TEXT NOT NULL, `title` TEXT NOT NULL, `stream_type` TEXT NOT NULL, `duration` INTEGER NOT NULL, `uploader` TEXT NOT NULL, `uploader_url` TEXT, `thumbnail_url` TEXT, `view_count` INTEGER, `textual_upload_date` TEXT, `upload_date` INTEGER, `is_upload_date_approximation` INTEGER)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "title", | ||||
|             "columnName": "title", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamType", | ||||
|             "columnName": "stream_type", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "duration", | ||||
|             "columnName": "duration", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploader", | ||||
|             "columnName": "uploader", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploaderUrl", | ||||
|             "columnName": "uploader_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailUrl", | ||||
|             "columnName": "thumbnail_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "viewCount", | ||||
|             "columnName": "view_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "textualUploadDate", | ||||
|             "columnName": "textual_upload_date", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploadDate", | ||||
|             "columnName": "upload_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "isUploadDateApproximation", | ||||
|             "columnName": "is_upload_date_approximation", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ], | ||||
|           "autoGenerate": true | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_streams_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_streams_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "stream_history", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `access_date` INTEGER NOT NULL, `repeat_count` INTEGER NOT NULL, PRIMARY KEY(`stream_id`, `access_date`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE )", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "accessDate", | ||||
|             "columnName": "access_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "repeatCount", | ||||
|             "columnName": "repeat_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "stream_id", | ||||
|             "access_date" | ||||
|           ], | ||||
|           "autoGenerate": false | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_stream_history_stream_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_stream_history_stream_id` ON `${TABLE_NAME}` (`stream_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "stream_state", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `progress_time` INTEGER NOT NULL, PRIMARY KEY(`stream_id`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE )", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "progressMillis", | ||||
|             "columnName": "progress_time", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "stream_id" | ||||
|           ], | ||||
|           "autoGenerate": false | ||||
|         }, | ||||
|         "indices": [], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "playlists", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT, `is_thumbnail_permanent` INTEGER NOT NULL, `thumbnail_stream_id` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "isThumbnailPermanent", | ||||
|             "columnName": "is_thumbnail_permanent", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailStreamId", | ||||
|             "columnName": "thumbnail_stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ], | ||||
|           "autoGenerate": true | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_playlists_name", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "name" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_playlists_name` ON `${TABLE_NAME}` (`name`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "playlist_stream_join", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`playlist_id` INTEGER NOT NULL, `stream_id` INTEGER NOT NULL, `join_index` INTEGER NOT NULL, PRIMARY KEY(`playlist_id`, `join_index`), FOREIGN KEY(`playlist_id`) REFERENCES `playlists`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "playlistUid", | ||||
|             "columnName": "playlist_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "index", | ||||
|             "columnName": "join_index", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "playlist_id", | ||||
|             "join_index" | ||||
|           ], | ||||
|           "autoGenerate": false | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_playlist_stream_join_playlist_id_join_index", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "playlist_id", | ||||
|               "join_index" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_playlist_stream_join_playlist_id_join_index` ON `${TABLE_NAME}` (`playlist_id`, `join_index`)" | ||||
|           }, | ||||
|           { | ||||
|             "name": "index_playlist_stream_join_stream_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_playlist_stream_join_stream_id` ON `${TABLE_NAME}` (`stream_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "playlists", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "playlist_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "remote_playlists", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `name` TEXT, `url` TEXT, `thumbnail_url` TEXT, `uploader` TEXT, `stream_count` INTEGER)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailUrl", | ||||
|             "columnName": "thumbnail_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploader", | ||||
|             "columnName": "uploader", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamCount", | ||||
|             "columnName": "stream_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ], | ||||
|           "autoGenerate": true | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_remote_playlists_name", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "name" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_remote_playlists_name` ON `${TABLE_NAME}` (`name`)" | ||||
|           }, | ||||
|           { | ||||
|             "name": "index_remote_playlists_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_remote_playlists_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `subscription_id` INTEGER NOT NULL, PRIMARY KEY(`stream_id`, `subscription_id`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamId", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "stream_id", | ||||
|             "subscription_id" | ||||
|           ], | ||||
|           "autoGenerate": false | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_subscription_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_subscription_id` ON `${TABLE_NAME}` (`subscription_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_group", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT NOT NULL, `icon_id` INTEGER NOT NULL, `sort_order` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "icon", | ||||
|             "columnName": "icon_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "sortOrder", | ||||
|             "columnName": "sort_order", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ], | ||||
|           "autoGenerate": true | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_group_sort_order", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "sort_order" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_group_sort_order` ON `${TABLE_NAME}` (`sort_order`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_group_subscription_join", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`group_id` INTEGER NOT NULL, `subscription_id` INTEGER NOT NULL, PRIMARY KEY(`group_id`, `subscription_id`), FOREIGN KEY(`group_id`) REFERENCES `feed_group`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "feedGroupId", | ||||
|             "columnName": "group_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "group_id", | ||||
|             "subscription_id" | ||||
|           ], | ||||
|           "autoGenerate": false | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_group_subscription_join_subscription_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_group_subscription_join_subscription_id` ON `${TABLE_NAME}` (`subscription_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "feed_group", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "group_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_last_updated", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`subscription_id` INTEGER NOT NULL, `last_updated` INTEGER, PRIMARY KEY(`subscription_id`), FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "lastUpdated", | ||||
|             "columnName": "last_updated", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "columnNames": [ | ||||
|             "subscription_id" | ||||
|           ], | ||||
|           "autoGenerate": false | ||||
|         }, | ||||
|         "indices": [], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       } | ||||
|     ], | ||||
|     "views": [], | ||||
|     "setupQueries": [ | ||||
|       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||
|       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, '012fc8e7ad3333f1597347f34e76a513')" | ||||
|     ] | ||||
|   } | ||||
| } | ||||
							
								
								
									
										737
									
								
								app/schemas/org.schabi.newpipe.database.AppDatabase/8.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										737
									
								
								app/schemas/org.schabi.newpipe.database.AppDatabase/8.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,737 @@ | ||||
| { | ||||
|   "formatVersion": 1, | ||||
|   "database": { | ||||
|     "version": 8, | ||||
|     "identityHash": "012fc8e7ad3333f1597347f34e76a513", | ||||
|     "entities": [ | ||||
|       { | ||||
|         "tableName": "subscriptions", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `url` TEXT, `name` TEXT, `avatar_url` TEXT, `subscriber_count` INTEGER, `description` TEXT, `notification_mode` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "avatarUrl", | ||||
|             "columnName": "avatar_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriberCount", | ||||
|             "columnName": "subscriber_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "description", | ||||
|             "columnName": "description", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "notificationMode", | ||||
|             "columnName": "notification_mode", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_subscriptions_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_subscriptions_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "search_history", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`creation_date` INTEGER, `service_id` INTEGER NOT NULL, `search` TEXT, `id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "creationDate", | ||||
|             "columnName": "creation_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "search", | ||||
|             "columnName": "search", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "id", | ||||
|             "columnName": "id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_search_history_search", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "search" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_search_history_search` ON `${TABLE_NAME}` (`search`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "streams", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `url` TEXT NOT NULL, `title` TEXT NOT NULL, `stream_type` TEXT NOT NULL, `duration` INTEGER NOT NULL, `uploader` TEXT NOT NULL, `uploader_url` TEXT, `thumbnail_url` TEXT, `view_count` INTEGER, `textual_upload_date` TEXT, `upload_date` INTEGER, `is_upload_date_approximation` INTEGER)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "title", | ||||
|             "columnName": "title", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamType", | ||||
|             "columnName": "stream_type", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "duration", | ||||
|             "columnName": "duration", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploader", | ||||
|             "columnName": "uploader", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploaderUrl", | ||||
|             "columnName": "uploader_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailUrl", | ||||
|             "columnName": "thumbnail_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "viewCount", | ||||
|             "columnName": "view_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "textualUploadDate", | ||||
|             "columnName": "textual_upload_date", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploadDate", | ||||
|             "columnName": "upload_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "isUploadDateApproximation", | ||||
|             "columnName": "is_upload_date_approximation", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_streams_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_streams_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "stream_history", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `access_date` INTEGER NOT NULL, `repeat_count` INTEGER NOT NULL, PRIMARY KEY(`stream_id`, `access_date`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE )", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "accessDate", | ||||
|             "columnName": "access_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "repeatCount", | ||||
|             "columnName": "repeat_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "stream_id", | ||||
|             "access_date" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_stream_history_stream_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_stream_history_stream_id` ON `${TABLE_NAME}` (`stream_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "stream_state", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `progress_time` INTEGER NOT NULL, PRIMARY KEY(`stream_id`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE )", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "progressMillis", | ||||
|             "columnName": "progress_time", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "stream_id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "playlists", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT, `is_thumbnail_permanent` INTEGER NOT NULL, `thumbnail_stream_id` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "isThumbnailPermanent", | ||||
|             "columnName": "is_thumbnail_permanent", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailStreamId", | ||||
|             "columnName": "thumbnail_stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_playlists_name", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "name" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_playlists_name` ON `${TABLE_NAME}` (`name`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "playlist_stream_join", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`playlist_id` INTEGER NOT NULL, `stream_id` INTEGER NOT NULL, `join_index` INTEGER NOT NULL, PRIMARY KEY(`playlist_id`, `join_index`), FOREIGN KEY(`playlist_id`) REFERENCES `playlists`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "playlistUid", | ||||
|             "columnName": "playlist_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "index", | ||||
|             "columnName": "join_index", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "playlist_id", | ||||
|             "join_index" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_playlist_stream_join_playlist_id_join_index", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "playlist_id", | ||||
|               "join_index" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_playlist_stream_join_playlist_id_join_index` ON `${TABLE_NAME}` (`playlist_id`, `join_index`)" | ||||
|           }, | ||||
|           { | ||||
|             "name": "index_playlist_stream_join_stream_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_playlist_stream_join_stream_id` ON `${TABLE_NAME}` (`stream_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "playlists", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "playlist_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "remote_playlists", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `name` TEXT, `url` TEXT, `thumbnail_url` TEXT, `uploader` TEXT, `stream_count` INTEGER)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailUrl", | ||||
|             "columnName": "thumbnail_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploader", | ||||
|             "columnName": "uploader", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamCount", | ||||
|             "columnName": "stream_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_remote_playlists_name", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "name" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_remote_playlists_name` ON `${TABLE_NAME}` (`name`)" | ||||
|           }, | ||||
|           { | ||||
|             "name": "index_remote_playlists_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_remote_playlists_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `subscription_id` INTEGER NOT NULL, PRIMARY KEY(`stream_id`, `subscription_id`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamId", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "stream_id", | ||||
|             "subscription_id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_subscription_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_subscription_id` ON `${TABLE_NAME}` (`subscription_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_group", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT NOT NULL, `icon_id` INTEGER NOT NULL, `sort_order` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "icon", | ||||
|             "columnName": "icon_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "sortOrder", | ||||
|             "columnName": "sort_order", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_group_sort_order", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "sort_order" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_group_sort_order` ON `${TABLE_NAME}` (`sort_order`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_group_subscription_join", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`group_id` INTEGER NOT NULL, `subscription_id` INTEGER NOT NULL, PRIMARY KEY(`group_id`, `subscription_id`), FOREIGN KEY(`group_id`) REFERENCES `feed_group`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "feedGroupId", | ||||
|             "columnName": "group_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "group_id", | ||||
|             "subscription_id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_group_subscription_join_subscription_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_group_subscription_join_subscription_id` ON `${TABLE_NAME}` (`subscription_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "feed_group", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "group_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_last_updated", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`subscription_id` INTEGER NOT NULL, `last_updated` INTEGER, PRIMARY KEY(`subscription_id`), FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "lastUpdated", | ||||
|             "columnName": "last_updated", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "subscription_id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       } | ||||
|     ], | ||||
|     "views": [], | ||||
|     "setupQueries": [ | ||||
|       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||
|       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, '012fc8e7ad3333f1597347f34e76a513')" | ||||
|     ] | ||||
|   } | ||||
| } | ||||
							
								
								
									
										749
									
								
								app/schemas/org.schabi.newpipe.database.AppDatabase/9.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										749
									
								
								app/schemas/org.schabi.newpipe.database.AppDatabase/9.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,749 @@ | ||||
| { | ||||
|   "formatVersion": 1, | ||||
|   "database": { | ||||
|     "version": 9, | ||||
|     "identityHash": "94596ea2227c63dd78b472ea4a83f1c4", | ||||
|     "entities": [ | ||||
|       { | ||||
|         "tableName": "subscriptions", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `url` TEXT, `name` TEXT, `avatar_url` TEXT, `subscriber_count` INTEGER, `description` TEXT, `notification_mode` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "avatarUrl", | ||||
|             "columnName": "avatar_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriberCount", | ||||
|             "columnName": "subscriber_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "description", | ||||
|             "columnName": "description", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "notificationMode", | ||||
|             "columnName": "notification_mode", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_subscriptions_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_subscriptions_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "search_history", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`creation_date` INTEGER, `service_id` INTEGER NOT NULL, `search` TEXT, `id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "creationDate", | ||||
|             "columnName": "creation_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "search", | ||||
|             "columnName": "search", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "id", | ||||
|             "columnName": "id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_search_history_search", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "search" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_search_history_search` ON `${TABLE_NAME}` (`search`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "streams", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `url` TEXT NOT NULL, `title` TEXT NOT NULL, `stream_type` TEXT NOT NULL, `duration` INTEGER NOT NULL, `uploader` TEXT NOT NULL, `uploader_url` TEXT, `thumbnail_url` TEXT, `view_count` INTEGER, `textual_upload_date` TEXT, `upload_date` INTEGER, `is_upload_date_approximation` INTEGER)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "title", | ||||
|             "columnName": "title", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamType", | ||||
|             "columnName": "stream_type", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "duration", | ||||
|             "columnName": "duration", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploader", | ||||
|             "columnName": "uploader", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploaderUrl", | ||||
|             "columnName": "uploader_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailUrl", | ||||
|             "columnName": "thumbnail_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "viewCount", | ||||
|             "columnName": "view_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "textualUploadDate", | ||||
|             "columnName": "textual_upload_date", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploadDate", | ||||
|             "columnName": "upload_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "isUploadDateApproximation", | ||||
|             "columnName": "is_upload_date_approximation", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_streams_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_streams_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "stream_history", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `access_date` INTEGER NOT NULL, `repeat_count` INTEGER NOT NULL, PRIMARY KEY(`stream_id`, `access_date`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE )", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "accessDate", | ||||
|             "columnName": "access_date", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "repeatCount", | ||||
|             "columnName": "repeat_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "stream_id", | ||||
|             "access_date" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_stream_history_stream_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_stream_history_stream_id` ON `${TABLE_NAME}` (`stream_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "stream_state", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `progress_time` INTEGER NOT NULL, PRIMARY KEY(`stream_id`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE )", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "progressMillis", | ||||
|             "columnName": "progress_time", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "stream_id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "playlists", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT, `is_thumbnail_permanent` INTEGER NOT NULL, `thumbnail_stream_id` INTEGER NOT NULL, `display_index` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "isThumbnailPermanent", | ||||
|             "columnName": "is_thumbnail_permanent", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailStreamId", | ||||
|             "columnName": "thumbnail_stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "displayIndex", | ||||
|             "columnName": "display_index", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_playlists_name", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "name" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_playlists_name` ON `${TABLE_NAME}` (`name`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "playlist_stream_join", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`playlist_id` INTEGER NOT NULL, `stream_id` INTEGER NOT NULL, `join_index` INTEGER NOT NULL, PRIMARY KEY(`playlist_id`, `join_index`), FOREIGN KEY(`playlist_id`) REFERENCES `playlists`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "playlistUid", | ||||
|             "columnName": "playlist_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamUid", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "index", | ||||
|             "columnName": "join_index", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "playlist_id", | ||||
|             "join_index" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_playlist_stream_join_playlist_id_join_index", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "playlist_id", | ||||
|               "join_index" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_playlist_stream_join_playlist_id_join_index` ON `${TABLE_NAME}` (`playlist_id`, `join_index`)" | ||||
|           }, | ||||
|           { | ||||
|             "name": "index_playlist_stream_join_stream_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_playlist_stream_join_stream_id` ON `${TABLE_NAME}` (`stream_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "playlists", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "playlist_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "remote_playlists", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `service_id` INTEGER NOT NULL, `name` TEXT, `url` TEXT, `thumbnail_url` TEXT, `uploader` TEXT, `display_index` INTEGER NOT NULL, `stream_count` INTEGER)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "serviceId", | ||||
|             "columnName": "service_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "url", | ||||
|             "columnName": "url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "thumbnailUrl", | ||||
|             "columnName": "thumbnail_url", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "uploader", | ||||
|             "columnName": "uploader", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": false | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "displayIndex", | ||||
|             "columnName": "display_index", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "streamCount", | ||||
|             "columnName": "stream_count", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_remote_playlists_name", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "name" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_remote_playlists_name` ON `${TABLE_NAME}` (`name`)" | ||||
|           }, | ||||
|           { | ||||
|             "name": "index_remote_playlists_service_id_url", | ||||
|             "unique": true, | ||||
|             "columnNames": [ | ||||
|               "service_id", | ||||
|               "url" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_remote_playlists_service_id_url` ON `${TABLE_NAME}` (`service_id`, `url`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`stream_id` INTEGER NOT NULL, `subscription_id` INTEGER NOT NULL, PRIMARY KEY(`stream_id`, `subscription_id`), FOREIGN KEY(`stream_id`) REFERENCES `streams`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "streamId", | ||||
|             "columnName": "stream_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "stream_id", | ||||
|             "subscription_id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_subscription_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_subscription_id` ON `${TABLE_NAME}` (`subscription_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "streams", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "stream_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_group", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT NOT NULL, `icon_id` INTEGER NOT NULL, `sort_order` INTEGER NOT NULL)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "uid", | ||||
|             "columnName": "uid", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "name", | ||||
|             "columnName": "name", | ||||
|             "affinity": "TEXT", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "icon", | ||||
|             "columnName": "icon_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "sortOrder", | ||||
|             "columnName": "sort_order", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": true, | ||||
|           "columnNames": [ | ||||
|             "uid" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_group_sort_order", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "sort_order" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_group_sort_order` ON `${TABLE_NAME}` (`sort_order`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_group_subscription_join", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`group_id` INTEGER NOT NULL, `subscription_id` INTEGER NOT NULL, PRIMARY KEY(`group_id`, `subscription_id`), FOREIGN KEY(`group_id`) REFERENCES `feed_group`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED, FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "feedGroupId", | ||||
|             "columnName": "group_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "group_id", | ||||
|             "subscription_id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [ | ||||
|           { | ||||
|             "name": "index_feed_group_subscription_join_subscription_id", | ||||
|             "unique": false, | ||||
|             "columnNames": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "orders": [], | ||||
|             "createSql": "CREATE INDEX IF NOT EXISTS `index_feed_group_subscription_join_subscription_id` ON `${TABLE_NAME}` (`subscription_id`)" | ||||
|           } | ||||
|         ], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "feed_group", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "group_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         "tableName": "feed_last_updated", | ||||
|         "createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`subscription_id` INTEGER NOT NULL, `last_updated` INTEGER, PRIMARY KEY(`subscription_id`), FOREIGN KEY(`subscription_id`) REFERENCES `subscriptions`(`uid`) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED)", | ||||
|         "fields": [ | ||||
|           { | ||||
|             "fieldPath": "subscriptionId", | ||||
|             "columnName": "subscription_id", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": true | ||||
|           }, | ||||
|           { | ||||
|             "fieldPath": "lastUpdated", | ||||
|             "columnName": "last_updated", | ||||
|             "affinity": "INTEGER", | ||||
|             "notNull": false | ||||
|           } | ||||
|         ], | ||||
|         "primaryKey": { | ||||
|           "autoGenerate": false, | ||||
|           "columnNames": [ | ||||
|             "subscription_id" | ||||
|           ] | ||||
|         }, | ||||
|         "indices": [], | ||||
|         "foreignKeys": [ | ||||
|           { | ||||
|             "table": "subscriptions", | ||||
|             "onDelete": "CASCADE", | ||||
|             "onUpdate": "CASCADE", | ||||
|             "columns": [ | ||||
|               "subscription_id" | ||||
|             ], | ||||
|             "referencedColumns": [ | ||||
|               "uid" | ||||
|             ] | ||||
|           } | ||||
|         ] | ||||
|       } | ||||
|     ], | ||||
|     "views": [], | ||||
|     "setupQueries": [ | ||||
|       "CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)", | ||||
|       "INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, '94596ea2227c63dd78b472ea4a83f1c4')" | ||||
|     ] | ||||
|   } | ||||
| } | ||||
| @@ -4,17 +4,18 @@ import android.content.ContentValues | ||||
| import android.database.sqlite.SQLiteDatabase | ||||
| import androidx.room.Room | ||||
| import androidx.room.testing.MigrationTestHelper | ||||
| import androidx.sqlite.db.framework.FrameworkSQLiteOpenHelperFactory | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.platform.app.InstrumentationRegistry | ||||
| import org.junit.Assert.assertEquals | ||||
| import org.junit.Assert.assertNotEquals | ||||
| import org.junit.Assert.assertNull | ||||
| import org.junit.Rule | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
| import org.schabi.newpipe.database.playlist.model.PlaylistEntity | ||||
| import org.schabi.newpipe.database.playlist.model.PlaylistRemoteEntity | ||||
| import org.schabi.newpipe.extractor.ServiceList | ||||
| import org.schabi.newpipe.extractor.stream.StreamType | ||||
|  | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @@ -39,7 +40,7 @@ class DatabaseMigrationTest { | ||||
|     @get:Rule | ||||
|     val testHelper = MigrationTestHelper( | ||||
|         InstrumentationRegistry.getInstrumentation(), | ||||
|         AppDatabase::class.java.canonicalName, FrameworkSQLiteOpenHelperFactory() | ||||
|         AppDatabase::class.java | ||||
|     ) | ||||
|  | ||||
|     @Test | ||||
| @@ -48,7 +49,8 @@ class DatabaseMigrationTest { | ||||
|  | ||||
|         databaseInV2.run { | ||||
|             insert( | ||||
|                 "streams", SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 "streams", | ||||
|                 SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 ContentValues().apply { | ||||
|                     put("service_id", DEFAULT_SERVICE_ID) | ||||
|                     put("url", DEFAULT_URL) | ||||
| @@ -60,14 +62,16 @@ class DatabaseMigrationTest { | ||||
|                 } | ||||
|             ) | ||||
|             insert( | ||||
|                 "streams", SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 "streams", | ||||
|                 SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 ContentValues().apply { | ||||
|                     put("service_id", DEFAULT_SECOND_SERVICE_ID) | ||||
|                     put("url", DEFAULT_SECOND_URL) | ||||
|                 } | ||||
|             ) | ||||
|             insert( | ||||
|                 "streams", SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 "streams", | ||||
|                 SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 ContentValues().apply { | ||||
|                     put("service_id", DEFAULT_SERVICE_ID) | ||||
|                 } | ||||
| @@ -76,18 +80,45 @@ class DatabaseMigrationTest { | ||||
|         } | ||||
|  | ||||
|         testHelper.runMigrationsAndValidate( | ||||
|             AppDatabase.DATABASE_NAME, Migrations.DB_VER_3, | ||||
|             true, Migrations.MIGRATION_2_3 | ||||
|             AppDatabase.DATABASE_NAME, | ||||
|             Migrations.DB_VER_3, | ||||
|             true, | ||||
|             Migrations.MIGRATION_2_3 | ||||
|         ) | ||||
|  | ||||
|         testHelper.runMigrationsAndValidate( | ||||
|             AppDatabase.DATABASE_NAME, Migrations.DB_VER_4, | ||||
|             true, Migrations.MIGRATION_3_4 | ||||
|             AppDatabase.DATABASE_NAME, | ||||
|             Migrations.DB_VER_4, | ||||
|             true, | ||||
|             Migrations.MIGRATION_3_4 | ||||
|         ) | ||||
|  | ||||
|         testHelper.runMigrationsAndValidate( | ||||
|             AppDatabase.DATABASE_NAME, Migrations.DB_VER_5, | ||||
|             true, Migrations.MIGRATION_4_5 | ||||
|             AppDatabase.DATABASE_NAME, | ||||
|             Migrations.DB_VER_5, | ||||
|             true, | ||||
|             Migrations.MIGRATION_4_5 | ||||
|         ) | ||||
|  | ||||
|         testHelper.runMigrationsAndValidate( | ||||
|             AppDatabase.DATABASE_NAME, | ||||
|             Migrations.DB_VER_6, | ||||
|             true, | ||||
|             Migrations.MIGRATION_5_6 | ||||
|         ) | ||||
|  | ||||
|         testHelper.runMigrationsAndValidate( | ||||
|             AppDatabase.DATABASE_NAME, | ||||
|             Migrations.DB_VER_7, | ||||
|             true, | ||||
|             Migrations.MIGRATION_6_7 | ||||
|         ) | ||||
|  | ||||
|         testHelper.runMigrationsAndValidate( | ||||
|             AppDatabase.DATABASE_NAME, | ||||
|             Migrations.DB_VER_8, | ||||
|             true, | ||||
|             Migrations.MIGRATION_7_8 | ||||
|         ) | ||||
|  | ||||
|         testHelper.runMigrationsAndValidate( | ||||
| @@ -130,7 +161,65 @@ class DatabaseMigrationTest { | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun migrateDatabaseFrom5to6() { | ||||
|     fun migrateDatabaseFrom7to8() { | ||||
|         val databaseInV7 = testHelper.createDatabase(AppDatabase.DATABASE_NAME, Migrations.DB_VER_7) | ||||
|  | ||||
|         val defaultSearch1 = " abc " | ||||
|         val defaultSearch2 = " abc" | ||||
|  | ||||
|         val serviceId = DEFAULT_SERVICE_ID // YouTube | ||||
|         // Use id different to YouTube because two searches with the same query | ||||
|         // but different service are considered not equal. | ||||
|         val otherServiceId = ServiceList.SoundCloud.serviceId | ||||
|  | ||||
|         databaseInV7.run { | ||||
|             insert( | ||||
|                 "search_history", SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 ContentValues().apply { | ||||
|                     put("service_id", serviceId) | ||||
|                     put("search", defaultSearch1) | ||||
|                 } | ||||
|             ) | ||||
|             insert( | ||||
|                 "search_history", SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 ContentValues().apply { | ||||
|                     put("service_id", serviceId) | ||||
|                     put("search", defaultSearch2) | ||||
|                 } | ||||
|             ) | ||||
|             insert( | ||||
|                 "search_history", SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 ContentValues().apply { | ||||
|                     put("service_id", otherServiceId) | ||||
|                     put("search", defaultSearch1) | ||||
|                 } | ||||
|             ) | ||||
|             insert( | ||||
|                 "search_history", SQLiteDatabase.CONFLICT_FAIL, | ||||
|                 ContentValues().apply { | ||||
|                     put("service_id", otherServiceId) | ||||
|                     put("search", defaultSearch2) | ||||
|                 } | ||||
|             ) | ||||
|             close() | ||||
|         } | ||||
|  | ||||
|         testHelper.runMigrationsAndValidate( | ||||
|             AppDatabase.DATABASE_NAME, Migrations.DB_VER_8, | ||||
|             true, Migrations.MIGRATION_7_8 | ||||
|         ) | ||||
|  | ||||
|         val migratedDatabaseV8 = getMigratedDatabase() | ||||
|         val listFromDB = migratedDatabaseV8.searchHistoryDAO().all.blockingFirst() | ||||
|  | ||||
|         assertEquals(2, listFromDB.size) | ||||
|         assertEquals("abc", listFromDB[0].search) | ||||
|         assertEquals("abc", listFromDB[1].search) | ||||
|         assertNotEquals(listFromDB[0].serviceId, listFromDB[1].serviceId) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun migrateDatabaseFrom8to9() { | ||||
|         val databaseInV5 = testHelper.createDatabase(AppDatabase.DATABASE_NAME, Migrations.DB_VER_5) | ||||
|  | ||||
|         val localUid1: Long | ||||
| @@ -216,7 +305,8 @@ class DatabaseMigrationTest { | ||||
|     private fun getMigratedDatabase(): AppDatabase { | ||||
|         val database: AppDatabase = Room.databaseBuilder( | ||||
|             ApplicationProvider.getApplicationContext(), | ||||
|             AppDatabase::class.java, AppDatabase.DATABASE_NAME | ||||
|             AppDatabase::class.java, | ||||
|             AppDatabase.DATABASE_NAME | ||||
|         ) | ||||
|             .build() | ||||
|         testHelper.closeWhenFinished(database) | ||||
|   | ||||
| @@ -0,0 +1,130 @@ | ||||
| package org.schabi.newpipe.database | ||||
|  | ||||
| import android.content.Context | ||||
| import androidx.room.Room | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import io.reactivex.rxjava3.core.Single | ||||
| import org.junit.After | ||||
| import org.junit.Assert.assertEquals | ||||
| import org.junit.Assert.assertNotNull | ||||
| import org.junit.Before | ||||
| import org.junit.Test | ||||
| import org.schabi.newpipe.database.feed.dao.FeedDAO | ||||
| import org.schabi.newpipe.database.feed.model.FeedEntity | ||||
| import org.schabi.newpipe.database.feed.model.FeedGroupEntity | ||||
| import org.schabi.newpipe.database.stream.StreamWithState | ||||
| import org.schabi.newpipe.database.stream.dao.StreamDAO | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity | ||||
| import org.schabi.newpipe.database.subscription.SubscriptionDAO | ||||
| import org.schabi.newpipe.database.subscription.SubscriptionEntity | ||||
| import org.schabi.newpipe.extractor.ServiceList | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfo | ||||
| import org.schabi.newpipe.extractor.stream.StreamType | ||||
| import java.io.IOException | ||||
| import java.time.OffsetDateTime | ||||
| import kotlin.streams.toList | ||||
|  | ||||
| class FeedDAOTest { | ||||
|     private lateinit var db: AppDatabase | ||||
|     private lateinit var feedDAO: FeedDAO | ||||
|     private lateinit var streamDAO: StreamDAO | ||||
|     private lateinit var subscriptionDAO: SubscriptionDAO | ||||
|  | ||||
|     private val serviceId = ServiceList.YouTube.serviceId | ||||
|  | ||||
|     private val stream1 = StreamEntity(1, serviceId, "https://youtube.com/watch?v=1", "stream 1", StreamType.VIDEO_STREAM, 1000, "channel-1", "https://youtube.com/channel/1", "https://i.ytimg.com/vi/1/hqdefault.jpg", 100, "2023-01-01", OffsetDateTime.parse("2023-01-01T00:00:00Z")) | ||||
|     private val stream2 = StreamEntity(2, serviceId, "https://youtube.com/watch?v=2", "stream 2", StreamType.VIDEO_STREAM, 1000, "channel-1", "https://youtube.com/channel/1", "https://i.ytimg.com/vi/1/hqdefault.jpg", 100, "2023-01-02", OffsetDateTime.parse("2023-01-02T00:00:00Z")) | ||||
|     private val stream3 = StreamEntity(3, serviceId, "https://youtube.com/watch?v=3", "stream 3", StreamType.LIVE_STREAM, 1000, "channel-1", "https://youtube.com/channel/1", "https://i.ytimg.com/vi/1/hqdefault.jpg", 100, "2023-01-03", OffsetDateTime.parse("2023-01-03T00:00:00Z")) | ||||
|     private val stream4 = StreamEntity(4, serviceId, "https://youtube.com/watch?v=4", "stream 4", StreamType.VIDEO_STREAM, 1000, "channel-2", "https://youtube.com/channel/2", "https://i.ytimg.com/vi/1/hqdefault.jpg", 100, "2023-08-10", OffsetDateTime.parse("2023-08-10T00:00:00Z")) | ||||
|     private val stream5 = StreamEntity(5, serviceId, "https://youtube.com/watch?v=5", "stream 5", StreamType.VIDEO_STREAM, 1000, "channel-2", "https://youtube.com/channel/2", "https://i.ytimg.com/vi/1/hqdefault.jpg", 100, "2023-08-20", OffsetDateTime.parse("2023-08-20T00:00:00Z")) | ||||
|     private val stream6 = StreamEntity(6, serviceId, "https://youtube.com/watch?v=6", "stream 6", StreamType.VIDEO_STREAM, 1000, "channel-3", "https://youtube.com/channel/3", "https://i.ytimg.com/vi/1/hqdefault.jpg", 100, "2023-09-01", OffsetDateTime.parse("2023-09-01T00:00:00Z")) | ||||
|     private val stream7 = StreamEntity(7, serviceId, "https://youtube.com/watch?v=7", "stream 7", StreamType.VIDEO_STREAM, 1000, "channel-4", "https://youtube.com/channel/4", "https://i.ytimg.com/vi/1/hqdefault.jpg", 100, "2023-08-10", OffsetDateTime.parse("2023-08-10T00:00:00Z")) | ||||
|  | ||||
|     private val allStreams = listOf( | ||||
|         stream1, stream2, stream3, stream4, stream5, stream6, stream7 | ||||
|     ) | ||||
|  | ||||
|     @Before | ||||
|     fun createDb() { | ||||
|         val context = ApplicationProvider.getApplicationContext<Context>() | ||||
|         db = Room.inMemoryDatabaseBuilder( | ||||
|             context, AppDatabase::class.java | ||||
|         ).build() | ||||
|         feedDAO = db.feedDAO() | ||||
|         streamDAO = db.streamDAO() | ||||
|         subscriptionDAO = db.subscriptionDAO() | ||||
|     } | ||||
|  | ||||
|     @After | ||||
|     @Throws(IOException::class) | ||||
|     fun closeDb() { | ||||
|         db.close() | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testUnlinkStreamsOlderThan_KeepOne() { | ||||
|         setupUnlinkDelete("2023-08-15T00:00:00Z") | ||||
|         val streams = feedDAO.getStreams( | ||||
|             FeedGroupEntity.GROUP_ALL_ID, includePlayed = true, includePartiallyPlayed = true, null | ||||
|         ) | ||||
|             .blockingGet() | ||||
|         val allowedStreams = listOf(stream3, stream5, stream6, stream7) | ||||
|         assertEqual(streams, allowedStreams) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun testUnlinkStreamsOlderThan_KeepMultiple() { | ||||
|         setupUnlinkDelete("2023-08-01T00:00:00Z") | ||||
|         val streams = feedDAO.getStreams( | ||||
|             FeedGroupEntity.GROUP_ALL_ID, includePlayed = true, includePartiallyPlayed = true, null | ||||
|         ) | ||||
|             .blockingGet() | ||||
|         val allowedStreams = listOf(stream3, stream4, stream5, stream6, stream7) | ||||
|         assertEqual(streams, allowedStreams) | ||||
|     } | ||||
|  | ||||
|     private fun assertEqual(streams: List<StreamWithState>?, allowedStreams: List<StreamEntity>) { | ||||
|         assertNotNull(streams) | ||||
|         assertEquals( | ||||
|             allowedStreams, | ||||
|             streams!! | ||||
|                 .map { it.stream } | ||||
|                 .sortedBy { it.uid } | ||||
|                 .toList() | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     private fun setupUnlinkDelete(time: String) { | ||||
|         clearAndFillTables() | ||||
|         Single.fromCallable { | ||||
|             feedDAO.unlinkStreamsOlderThan(OffsetDateTime.parse(time)) | ||||
|         }.blockingSubscribe() | ||||
|         Single.fromCallable { | ||||
|             streamDAO.deleteOrphans() | ||||
|         }.blockingSubscribe() | ||||
|     } | ||||
|  | ||||
|     private fun clearAndFillTables() { | ||||
|         db.clearAllTables() | ||||
|         streamDAO.insertAll(allStreams) | ||||
|         subscriptionDAO.insertAll( | ||||
|             listOf( | ||||
|                 SubscriptionEntity.from(ChannelInfo(serviceId, "1", "https://youtube.com/channel/1", "https://youtube.com/channel/1", "channel-1")), | ||||
|                 SubscriptionEntity.from(ChannelInfo(serviceId, "2", "https://youtube.com/channel/2", "https://youtube.com/channel/2", "channel-2")), | ||||
|                 SubscriptionEntity.from(ChannelInfo(serviceId, "3", "https://youtube.com/channel/3", "https://youtube.com/channel/3", "channel-3")), | ||||
|                 SubscriptionEntity.from(ChannelInfo(serviceId, "4", "https://youtube.com/channel/4", "https://youtube.com/channel/4", "channel-4")), | ||||
|             ) | ||||
|         ) | ||||
|         feedDAO.insertAll( | ||||
|             listOf( | ||||
|                 FeedEntity(1, 1), | ||||
|                 FeedEntity(2, 1), | ||||
|                 FeedEntity(3, 1), | ||||
|                 FeedEntity(4, 2), | ||||
|                 FeedEntity(5, 2), | ||||
|                 FeedEntity(6, 3), | ||||
|                 FeedEntity(7, 4), | ||||
|             ) | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,82 @@ | ||||
| package org.schabi.newpipe.local.subscription; | ||||
|  | ||||
| import static org.junit.Assert.assertEquals; | ||||
|  | ||||
| import androidx.test.core.app.ApplicationProvider; | ||||
|  | ||||
| import org.junit.After; | ||||
| import org.junit.Before; | ||||
| import org.junit.Rule; | ||||
| import org.junit.Test; | ||||
| import org.schabi.newpipe.database.AppDatabase; | ||||
| import org.schabi.newpipe.database.feed.model.FeedGroupEntity; | ||||
| import org.schabi.newpipe.database.subscription.SubscriptionEntity; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfo; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.testUtil.TestDatabase; | ||||
| import org.schabi.newpipe.testUtil.TrampolineSchedulerRule; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.List; | ||||
|  | ||||
| public class SubscriptionManagerTest { | ||||
|     private AppDatabase database; | ||||
|     private SubscriptionManager manager; | ||||
|  | ||||
|     @Rule | ||||
|     public TrampolineSchedulerRule trampolineScheduler = new TrampolineSchedulerRule(); | ||||
|  | ||||
|  | ||||
|     private SubscriptionEntity getAssertOneSubscriptionEntity() { | ||||
|         final List<SubscriptionEntity> entities = manager | ||||
|                 .getSubscriptions(FeedGroupEntity.GROUP_ALL_ID, "", false) | ||||
|                 .blockingFirst(); | ||||
|         assertEquals(1, entities.size()); | ||||
|         return entities.get(0); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Before | ||||
|     public void setup() { | ||||
|         database = TestDatabase.Companion.createReplacingNewPipeDatabase(); | ||||
|         manager = new SubscriptionManager(ApplicationProvider.getApplicationContext()); | ||||
|     } | ||||
|  | ||||
|     @After | ||||
|     public void cleanUp() { | ||||
|         database.close(); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testInsert() throws ExtractionException, IOException { | ||||
|         final ChannelInfo info = ChannelInfo.getInfo("https://www.youtube.com/c/3blue1brown"); | ||||
|         final SubscriptionEntity subscription = SubscriptionEntity.from(info); | ||||
|  | ||||
|         manager.insertSubscription(subscription); | ||||
|         final SubscriptionEntity readSubscription = getAssertOneSubscriptionEntity(); | ||||
|  | ||||
|         // the uid has changed, since the uid is chosen upon inserting, but the rest should match | ||||
|         assertEquals(subscription.getServiceId(), readSubscription.getServiceId()); | ||||
|         assertEquals(subscription.getUrl(), readSubscription.getUrl()); | ||||
|         assertEquals(subscription.getName(), readSubscription.getName()); | ||||
|         assertEquals(subscription.getAvatarUrl(), readSubscription.getAvatarUrl()); | ||||
|         assertEquals(subscription.getSubscriberCount(), readSubscription.getSubscriberCount()); | ||||
|         assertEquals(subscription.getDescription(), readSubscription.getDescription()); | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     public void testUpdateNotificationMode() throws ExtractionException, IOException { | ||||
|         final ChannelInfo info = ChannelInfo.getInfo("https://www.youtube.com/c/veritasium"); | ||||
|         final SubscriptionEntity subscription = SubscriptionEntity.from(info); | ||||
|         subscription.setNotificationMode(0); | ||||
|  | ||||
|         manager.insertSubscription(subscription); | ||||
|         manager.updateNotificationMode(subscription.getServiceId(), subscription.getUrl(), 1) | ||||
|                 .blockingAwait(); | ||||
|         final SubscriptionEntity anotherSubscription = getAssertOneSubscriptionEntity(); | ||||
|  | ||||
|         assertEquals(0, subscription.getNotificationMode()); | ||||
|         assertEquals(subscription.getUrl(), anotherSubscription.getUrl()); | ||||
|         assertEquals(1, anotherSubscription.getNotificationMode()); | ||||
|     } | ||||
| } | ||||
| @@ -1,26 +1,32 @@ | ||||
| package org.schabi.newpipe.util | ||||
|  | ||||
| import android.content.Context | ||||
| import android.util.SparseArray | ||||
| import android.view.View | ||||
| import android.view.View.GONE | ||||
| import android.view.View.INVISIBLE | ||||
| import android.view.View.VISIBLE | ||||
| import android.widget.Spinner | ||||
| import androidx.collection.SparseArrayCompat | ||||
| import androidx.test.core.app.ApplicationProvider | ||||
| import androidx.test.ext.junit.runners.AndroidJUnit4 | ||||
| import androidx.test.filters.MediumTest | ||||
| import androidx.test.internal.runner.junit4.statement.UiThreadStatement | ||||
| import org.junit.Assert | ||||
| import org.junit.Assert.assertEquals | ||||
| import org.junit.Assert.assertFalse | ||||
| import org.junit.Assert.assertNull | ||||
| import org.junit.Assert.assertTrue | ||||
| import org.junit.Before | ||||
| import org.junit.Test | ||||
| import org.junit.runner.RunWith | ||||
| import org.schabi.newpipe.R | ||||
| import org.schabi.newpipe.extractor.MediaFormat | ||||
| import org.schabi.newpipe.extractor.downloader.Response | ||||
| import org.schabi.newpipe.extractor.stream.AudioStream | ||||
| import org.schabi.newpipe.extractor.stream.Stream | ||||
| import org.schabi.newpipe.extractor.stream.SubtitlesStream | ||||
| import org.schabi.newpipe.extractor.stream.VideoStream | ||||
| import org.schabi.newpipe.util.StreamItemAdapter.StreamInfoWrapper | ||||
|  | ||||
| @MediumTest | ||||
| @RunWith(AndroidJUnit4::class) | ||||
| @@ -39,9 +45,7 @@ class StreamItemAdapterTest { | ||||
|     @Test | ||||
|     fun videoStreams_noSecondaryStream() { | ||||
|         val adapter = StreamItemAdapter<VideoStream, AudioStream>( | ||||
|             context, | ||||
|             getVideoStreams(true, true, true, true), | ||||
|             null | ||||
|             getVideoStreams(true, true, true, true) | ||||
|         ) | ||||
|  | ||||
|         spinner.adapter = adapter | ||||
| @@ -54,7 +58,6 @@ class StreamItemAdapterTest { | ||||
|     @Test | ||||
|     fun videoStreams_hasSecondaryStream() { | ||||
|         val adapter = StreamItemAdapter( | ||||
|             context, | ||||
|             getVideoStreams(false, true, false, true), | ||||
|             getAudioStreams(false, true, false, true) | ||||
|         ) | ||||
| @@ -69,7 +72,6 @@ class StreamItemAdapterTest { | ||||
|     @Test | ||||
|     fun videoStreams_Mixed() { | ||||
|         val adapter = StreamItemAdapter( | ||||
|             context, | ||||
|             getVideoStreams(true, true, true, true, true, false, true, true), | ||||
|             getAudioStreams(false, true, false, false, false, true, true, true) | ||||
|         ) | ||||
| @@ -88,14 +90,17 @@ class StreamItemAdapterTest { | ||||
|     @Test | ||||
|     fun subtitleStreams_noIcon() { | ||||
|         val adapter = StreamItemAdapter<SubtitlesStream, Stream>( | ||||
|             context, | ||||
|             StreamItemAdapter.StreamSizeWrapper( | ||||
|             StreamItemAdapter.StreamInfoWrapper( | ||||
|                 (0 until 5).map { | ||||
|                     SubtitlesStream(MediaFormat.SRT, "pt-BR", "https://example.com", false) | ||||
|                     SubtitlesStream.Builder() | ||||
|                         .setContent("https://example.com", true) | ||||
|                         .setMediaFormat(MediaFormat.SRT) | ||||
|                         .setLanguageCode("pt-BR") | ||||
|                         .setAutoGenerated(false) | ||||
|                         .build() | ||||
|                 }, | ||||
|                 context | ||||
|             ), | ||||
|             null | ||||
|             ) | ||||
|         ) | ||||
|         spinner.adapter = adapter | ||||
|         for (i in 0 until spinner.count) { | ||||
| @@ -106,12 +111,17 @@ class StreamItemAdapterTest { | ||||
|     @Test | ||||
|     fun audioStreams_noIcon() { | ||||
|         val adapter = StreamItemAdapter<AudioStream, Stream>( | ||||
|             context, | ||||
|             StreamItemAdapter.StreamSizeWrapper( | ||||
|                 (0 until 5).map { AudioStream("https://example.com/$it", MediaFormat.OPUS, 192) }, | ||||
|             StreamItemAdapter.StreamInfoWrapper( | ||||
|                 (0 until 5).map { | ||||
|                     AudioStream.Builder() | ||||
|                         .setId(Stream.ID_UNKNOWN) | ||||
|                         .setContent("https://example.com/$it", true) | ||||
|                         .setMediaFormat(MediaFormat.OPUS) | ||||
|                         .setAverageBitrate(192) | ||||
|                         .build() | ||||
|                 }, | ||||
|                 context | ||||
|             ), | ||||
|             null | ||||
|             ) | ||||
|         ) | ||||
|         spinner.adapter = adapter | ||||
|         for (i in 0 until spinner.count) { | ||||
| @@ -119,14 +129,117 @@ class StreamItemAdapterTest { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun retrieveMediaFormatFromFileTypeHeaders() { | ||||
|         val streams = getIncompleteAudioStreams(5) | ||||
|         val wrapper = StreamInfoWrapper(streams, context) | ||||
|         val retrieveMediaFormat = { stream: AudioStream, response: Response -> | ||||
|             StreamInfoWrapper.retrieveMediaFormatFromFileTypeHeaders(stream, wrapper, response) | ||||
|         } | ||||
|         val helper = AssertionHelper(streams, wrapper, retrieveMediaFormat) | ||||
|  | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("content-length", "mp3"))), 0) | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("file-type", "mp0"))), 1) | ||||
|  | ||||
|         helper.assertValidResponse(getResponse(mapOf(Pair("x-amz-meta-file-type", "aiff"))), 2, MediaFormat.AIFF) | ||||
|         helper.assertValidResponse(getResponse(mapOf(Pair("file-type", "mp3"))), 3, MediaFormat.MP3) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun retrieveMediaFormatFromContentDispositionHeader() { | ||||
|         val streams = getIncompleteAudioStreams(11) | ||||
|         val wrapper = StreamInfoWrapper(streams, context) | ||||
|         val retrieveMediaFormat = { stream: AudioStream, response: Response -> | ||||
|             StreamInfoWrapper.retrieveMediaFormatFromContentDispositionHeader(stream, wrapper, response) | ||||
|         } | ||||
|         val helper = AssertionHelper(streams, wrapper, retrieveMediaFormat) | ||||
|  | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("content-length", "mp3"))), 0) | ||||
|         helper.assertInvalidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "filename=\"train.png\""))), 1 | ||||
|         ) | ||||
|         helper.assertInvalidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "form-data; name=\"data.csv\""))), 2 | ||||
|         ) | ||||
|         helper.assertInvalidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "form-data; filename=\"data.csv\""))), 3 | ||||
|         ) | ||||
|         helper.assertInvalidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "form-data; name=\"fieldName\"; filename*=\"filename.jpg\""))), 4 | ||||
|         ) | ||||
|  | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "filename=\"train.ogg\""))), | ||||
|             5, MediaFormat.OGG | ||||
|         ) | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "some-form-data; filename=\"audio.flac\""))), | ||||
|             6, MediaFormat.FLAC | ||||
|         ) | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "form-data; name=\"audio.aiff\"; filename=\"audio.aiff\""))), | ||||
|             7, MediaFormat.AIFF | ||||
|         ) | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "form-data; name=\"alien?\"; filename*=UTF-8''%CE%B1%CE%BB%CE%B9%CF%B5%CE%BD.m4a"))), | ||||
|             8, MediaFormat.M4A | ||||
|         ) | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "form-data; name=\"audio.mp3\"; filename=\"audio.opus\"; filename*=UTF-8''alien.opus"))), | ||||
|             9, MediaFormat.OPUS | ||||
|         ) | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Disposition", "form-data; name=\"audio.mp3\"; filename=\"audio.opus\"; filename*=\"UTF-8''alien.opus\""))), | ||||
|             10, MediaFormat.OPUS | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     @Test | ||||
|     fun retrieveMediaFormatFromContentTypeHeader() { | ||||
|         val streams = getIncompleteAudioStreams(12) | ||||
|         val wrapper = StreamInfoWrapper(streams, context) | ||||
|         val retrieveMediaFormat = { stream: AudioStream, response: Response -> | ||||
|             StreamInfoWrapper.retrieveMediaFormatFromContentTypeHeader(stream, wrapper, response) | ||||
|         } | ||||
|         val helper = AssertionHelper(streams, wrapper, retrieveMediaFormat) | ||||
|  | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("content-length", "984501"))), 0) | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("Content-Type", "audio/xyz"))), 1) | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("Content-Type", "mp3"))), 2) | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("Content-Type", "mp3"))), 3) | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("Content-Type", "audio/mpeg"))), 4) | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("Content-Type", "audio/aif"))), 5) | ||||
|         helper.assertInvalidResponse(getResponse(mapOf(Pair("Content-Type", "whatever"))), 6) | ||||
|         helper.assertInvalidResponse(getResponse(mapOf()), 7) | ||||
|  | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Type", "audio/flac"))), 8, MediaFormat.FLAC | ||||
|         ) | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Type", "audio/wav"))), 9, MediaFormat.WAV | ||||
|         ) | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Type", "audio/opus"))), 10, MediaFormat.OPUS | ||||
|         ) | ||||
|         helper.assertValidResponse( | ||||
|             getResponse(mapOf(Pair("Content-Type", "audio/aiff"))), 11, MediaFormat.AIFF | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return a list of video streams, in which their video only property mirrors the provided | ||||
|      * [videoOnly] vararg. | ||||
|      */ | ||||
|     private fun getVideoStreams(vararg videoOnly: Boolean) = | ||||
|         StreamItemAdapter.StreamSizeWrapper( | ||||
|         StreamItemAdapter.StreamInfoWrapper( | ||||
|             videoOnly.map { | ||||
|                 VideoStream("https://example.com", MediaFormat.MPEG_4, "720p", it) | ||||
|                 VideoStream.Builder() | ||||
|                     .setId(Stream.ID_UNKNOWN) | ||||
|                     .setContent("https://example.com", true) | ||||
|                     .setMediaFormat(MediaFormat.MPEG_4) | ||||
|                     .setResolution("720p") | ||||
|                     .setIsVideoOnly(it) | ||||
|                     .build() | ||||
|             }, | ||||
|             context | ||||
|         ) | ||||
| @@ -138,11 +251,32 @@ class StreamItemAdapterTest { | ||||
|     private fun getAudioStreams(vararg shouldBeValid: Boolean) = | ||||
|         getSecondaryStreamsFromList( | ||||
|             shouldBeValid.map { | ||||
|                 if (it) AudioStream("https://example.com", MediaFormat.OPUS, 192) | ||||
|                 else null | ||||
|                 if (it) { | ||||
|                     AudioStream.Builder() | ||||
|                         .setId(Stream.ID_UNKNOWN) | ||||
|                         .setContent("https://example.com", true) | ||||
|                         .setMediaFormat(MediaFormat.OPUS) | ||||
|                         .setAverageBitrate(192) | ||||
|                         .build() | ||||
|                 } else { | ||||
|                     null | ||||
|                 } | ||||
|             } | ||||
|         ) | ||||
|  | ||||
|     private fun getIncompleteAudioStreams(size: Int): List<AudioStream> { | ||||
|         val list = ArrayList<AudioStream>(size) | ||||
|         for (i in 1..size) { | ||||
|             list.add( | ||||
|                 AudioStream.Builder() | ||||
|                     .setId(Stream.ID_UNKNOWN) | ||||
|                     .setContent("https://example.com/$i", true) | ||||
|                     .build() | ||||
|             ) | ||||
|         } | ||||
|         return list | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Checks whether the item at [position] in the [spinner] has the correct icon visibility when | ||||
|      * it is shown in normal mode (selected) and in dropdown mode (user is choosing one of a list). | ||||
| @@ -174,15 +308,60 @@ class StreamItemAdapterTest { | ||||
|      * Helper function that builds a secondary stream list. | ||||
|      */ | ||||
|     private fun <T : Stream> getSecondaryStreamsFromList(streams: List<T?>) = | ||||
|         SparseArray<SecondaryStreamHelper<T>?>(streams.size).apply { | ||||
|         SparseArrayCompat<SecondaryStreamHelper<T>?>(streams.size).apply { | ||||
|             streams.forEachIndexed { index, stream -> | ||||
|                 val secondaryStreamHelper: SecondaryStreamHelper<T>? = stream?.let { | ||||
|                     SecondaryStreamHelper( | ||||
|                         StreamItemAdapter.StreamSizeWrapper(streams, context), | ||||
|                         StreamItemAdapter.StreamInfoWrapper(streams, context), | ||||
|                         it | ||||
|                     ) | ||||
|                 } | ||||
|                 put(index, secondaryStreamHelper) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|     private fun getResponse(headers: Map<String, String>): Response { | ||||
|         val listHeaders = HashMap<String, List<String>>() | ||||
|         headers.forEach { entry -> | ||||
|             listHeaders[entry.key] = listOf(entry.value) | ||||
|         } | ||||
|         return Response(200, null, listHeaders, "", "") | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Helper class for assertion related to extractions of [MediaFormat]s. | ||||
|      */ | ||||
|     class AssertionHelper<T : Stream>( | ||||
|         private val streams: List<T>, | ||||
|         private val wrapper: StreamInfoWrapper<T>, | ||||
|         private val retrieveMediaFormat: (stream: T, response: Response) -> Boolean | ||||
|     ) { | ||||
|  | ||||
|         /** | ||||
|          * Assert that an invalid response does not result in wrongly extracted [MediaFormat]. | ||||
|          */ | ||||
|         fun assertInvalidResponse( | ||||
|             response: Response, | ||||
|             index: Int | ||||
|         ) { | ||||
|             assertFalse( | ||||
|                 "invalid header returns valid value", retrieveMediaFormat(streams[index], response) | ||||
|             ) | ||||
|             assertNull("Media format extracted although stated otherwise", wrapper.getFormat(index)) | ||||
|         } | ||||
|  | ||||
|         /** | ||||
|          * Assert that a valid response results in correctly extracted and handled [MediaFormat]. | ||||
|          */ | ||||
|         fun assertValidResponse( | ||||
|             response: Response, | ||||
|             index: Int, | ||||
|             format: MediaFormat | ||||
|         ) { | ||||
|             assertTrue( | ||||
|                 "header was not recognized", retrieveMediaFormat(streams[index], response) | ||||
|             ) | ||||
|             assertEquals("Wrong media format extracted", format, wrapper.getFormat(index)) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,8 +1,6 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <manifest | ||||
|     xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     package="org.schabi.newpipe"> | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:tools="http://schemas.android.com/tools"> | ||||
|  | ||||
|     <application | ||||
|         android:name=".DebugApp" | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package org.schabi.newpipe | ||||
| import androidx.preference.PreferenceManager | ||||
| import com.facebook.stetho.Stetho | ||||
| import com.facebook.stetho.okhttp3.StethoInterceptor | ||||
| import leakcanary.AppWatcher | ||||
| import leakcanary.LeakCanary | ||||
| import okhttp3.OkHttpClient | ||||
| import org.schabi.newpipe.extractor.downloader.Downloader | ||||
| @@ -13,8 +12,6 @@ class DebugApp : App() { | ||||
|         super.onCreate() | ||||
|         initStetho() | ||||
|  | ||||
|         // Give each object 10 seconds to be GC'ed, before LeakCanary gets nosy on it | ||||
|         AppWatcher.config = AppWatcher.config.copy(watchDurationMillis = 10000) | ||||
|         LeakCanary.config = LeakCanary.config.copy( | ||||
|             dumpHeap = PreferenceManager | ||||
|                 .getDefaultSharedPreferences(this).getBoolean( | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||||
|     xmlns:tools="http://schemas.android.com/tools" | ||||
|     package="org.schabi.newpipe" | ||||
|     android:installLocation="auto"> | ||||
|  | ||||
|     <uses-permission android:name="android.permission.INTERNET" /> | ||||
| @@ -10,10 +9,22 @@ | ||||
|     <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> | ||||
|     <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" /> | ||||
|     <uses-permission android:name="android.permission.FOREGROUND_SERVICE" /> | ||||
|     <uses-permission android:name="android.permission.POST_NOTIFICATIONS"/> | ||||
|  | ||||
|     <!-- We need to be able to open links in the browser on API 30+ --> | ||||
|     <queries> | ||||
|         <intent> | ||||
|             <action android:name="android.intent.action.VIEW" /> | ||||
|             <data android:scheme="http" /> | ||||
|         </intent> | ||||
|     </queries> | ||||
|  | ||||
|     <uses-feature | ||||
|         android:name="android.hardware.touchscreen" | ||||
|         android:required="false" /> | ||||
|     <uses-feature | ||||
|         android:name="android.software.leanback" | ||||
|         android:required="false" /> | ||||
|  | ||||
|     <application | ||||
|         android:name=".App" | ||||
| @@ -22,11 +33,12 @@ | ||||
|         android:icon="@mipmap/ic_launcher" | ||||
|         android:label="@string/app_name" | ||||
|         android:logo="@mipmap/ic_launcher" | ||||
|         android:theme="@style/OpeningTheme" | ||||
|         android:resizeableActivity="true" | ||||
|         android:theme="@style/OpeningTheme" | ||||
|         tools:ignore="AllowBackup"> | ||||
|         <activity | ||||
|             android:name=".MainActivity" | ||||
|             android:exported="true" | ||||
|             android:label="@string/app_name" | ||||
|             android:launchMode="singleTask"> | ||||
|             <intent-filter> | ||||
| @@ -37,15 +49,17 @@ | ||||
|             </intent-filter> | ||||
|         </activity> | ||||
|  | ||||
|         <receiver android:name="androidx.media.session.MediaButtonReceiver"> | ||||
|         <receiver | ||||
|             android:name="androidx.media.session.MediaButtonReceiver" | ||||
|             android:exported="true"> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.MEDIA_BUTTON" /> | ||||
|             </intent-filter> | ||||
|         </receiver> | ||||
|  | ||||
|         <service | ||||
|             android:name=".player.MainPlayer" | ||||
|             android:exported="false" | ||||
|             android:name=".player.PlayerService" | ||||
|             android:exported="true" | ||||
|             android:foregroundServiceType="mediaPlayback"> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.MEDIA_BUTTON" /> | ||||
| @@ -54,15 +68,18 @@ | ||||
|  | ||||
|         <activity | ||||
|             android:name=".player.PlayQueueActivity" | ||||
|             android:exported="false" | ||||
|             android:label="@string/title_activity_play_queue" | ||||
|             android:launchMode="singleTask" /> | ||||
|  | ||||
|         <activity | ||||
|             android:name=".settings.SettingsActivity" | ||||
|             android:exported="false" | ||||
|             android:label="@string/settings" /> | ||||
|  | ||||
|         <activity | ||||
|             android:name=".about.AboutActivity" | ||||
|             android:exported="false" | ||||
|             android:label="@string/title_activity_about" /> | ||||
|  | ||||
|         <service android:name=".local.subscription.services.SubscriptionsImportService" /> | ||||
| @@ -71,6 +88,7 @@ | ||||
|  | ||||
|         <activity | ||||
|             android:name=".PanicResponderActivity" | ||||
|             android:exported="true" | ||||
|             android:launchMode="singleInstance" | ||||
|             android:noHistory="true" | ||||
|             android:theme="@android:style/Theme.NoDisplay"> | ||||
| @@ -83,13 +101,18 @@ | ||||
|  | ||||
|         <activity | ||||
|             android:name=".ExitActivity" | ||||
|             android:exported="false" | ||||
|             android:label="@string/general_error" | ||||
|             android:theme="@android:style/Theme.NoDisplay" /> | ||||
|         <activity android:name=".error.ErrorActivity" /> | ||||
|  | ||||
|         <activity | ||||
|             android:name=".error.ErrorActivity" | ||||
|             android:exported="false" /> | ||||
|  | ||||
|         <!-- giga get related --> | ||||
|         <activity | ||||
|             android:name=".download.DownloadActivity" | ||||
|             android:exported="false" | ||||
|             android:label="@string/app_name" | ||||
|             android:launchMode="singleTask" /> | ||||
|  | ||||
| @@ -97,6 +120,7 @@ | ||||
|  | ||||
|         <activity | ||||
|             android:name=".util.FilePickerActivityHelper" | ||||
|             android:exported="true" | ||||
|             android:label="@string/app_name" | ||||
|             android:theme="@style/FilePickerThemeDark"> | ||||
|             <intent-filter> | ||||
| @@ -107,6 +131,7 @@ | ||||
|  | ||||
|         <activity | ||||
|             android:name=".error.ReCaptchaActivity" | ||||
|             android:exported="false" | ||||
|             android:label="@string/recaptcha" /> | ||||
|  | ||||
|         <provider | ||||
| @@ -122,6 +147,7 @@ | ||||
|         <activity | ||||
|             android:name=".RouterActivity" | ||||
|             android:excludeFromRecents="true" | ||||
|             android:exported="true" | ||||
|             android:label="@string/preferred_open_action_share_menu_title" | ||||
|             android:taskAffinity="" | ||||
|             android:theme="@style/RouterActivityThemeDark"> | ||||
| @@ -147,10 +173,12 @@ | ||||
|                 <data android:pathPrefix="/watch" /> | ||||
|                 <data android:pathPrefix="/attribution_link" /> | ||||
|                 <data android:pathPrefix="/shorts/" /> | ||||
|                 <data android:pathPrefix="/live/" /> | ||||
|                 <!-- channel prefix --> | ||||
|                 <data android:pathPrefix="/channel/" /> | ||||
|                 <data android:pathPrefix="/user/" /> | ||||
|                 <data android:pathPrefix="/c/" /> | ||||
|                 <data android:pathPrefix="/@" /> | ||||
|                 <!-- playlist prefix --> | ||||
|                 <data android:pathPrefix="/playlist" /> | ||||
|             </intent-filter> | ||||
| @@ -329,16 +357,16 @@ | ||||
|                 <data android:host="eduvid.org" /> | ||||
|                 <data android:host="framatube.org" /> | ||||
|                 <data android:host="media.assassinate-you.net" /> | ||||
|                 <data android:host="media.fsfe.org" /> | ||||
|                 <data android:host="peertube.co.uk" /> | ||||
|                 <data android:host="peertube.cpy.re" /> | ||||
|                 <data android:host="peertube.mastodon.host" /> | ||||
|                 <data android:host="peertube.fr" /> | ||||
|                 <data android:host="tilvids.com" /> | ||||
|                 <data android:host="tube.privacytools.io" /> | ||||
|                 <data android:host="video.ploud.fr" /> | ||||
|                 <data android:host="video.lqdn.fr" /> | ||||
|                 <data android:host="peertube.mastodon.host" /> | ||||
|                 <data android:host="peertube.stream" /> | ||||
|                 <data android:host="skeptikon.fr" /> | ||||
|                 <data android:host="media.fsfe.org" /> | ||||
|                 <data android:host="tilvids.com" /> | ||||
|                 <data android:host="video.lqdn.fr" /> | ||||
|                 <data android:host="video.ploud.fr" /> | ||||
|  | ||||
|                 <data android:pathPrefix="/videos/" /> <!-- it contains playlists --> | ||||
|                 <data android:pathPrefix="/w/" /> <!-- short video URLs --> | ||||
| @@ -351,30 +379,30 @@ | ||||
|  | ||||
|             <!-- Bandcamp filter for tracks, albums and playlists --> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.VIEW"/> | ||||
|                 <action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/> | ||||
|                 <action android:name="android.nfc.action.NDEF_DISCOVERED"/> | ||||
|                 <action android:name="android.intent.action.VIEW" /> | ||||
|                 <action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH" /> | ||||
|                 <action android:name="android.nfc.action.NDEF_DISCOVERED" /> | ||||
|  | ||||
|                 <category android:name="android.intent.category.DEFAULT"/> | ||||
|                 <category android:name="android.intent.category.BROWSABLE"/> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <category android:name="android.intent.category.BROWSABLE" /> | ||||
|  | ||||
|                 <data android:scheme="http"/> | ||||
|                 <data android:scheme="https"/> | ||||
|                 <data android:host="*.bandcamp.com"/> | ||||
|                 <data android:scheme="http" /> | ||||
|                 <data android:scheme="https" /> | ||||
|                 <data android:host="*.bandcamp.com" /> | ||||
|             </intent-filter> | ||||
|  | ||||
|             <!-- Bandcamp filter for radio --> | ||||
|             <intent-filter> | ||||
|                 <action android:name="android.intent.action.VIEW"/> | ||||
|                 <action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/> | ||||
|                 <action android:name="android.nfc.action.NDEF_DISCOVERED"/> | ||||
|                 <action android:name="android.intent.action.VIEW" /> | ||||
|                 <action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH" /> | ||||
|                 <action android:name="android.nfc.action.NDEF_DISCOVERED" /> | ||||
|  | ||||
|                 <category android:name="android.intent.category.DEFAULT"/> | ||||
|                 <category android:name="android.intent.category.BROWSABLE"/> | ||||
|                 <category android:name="android.intent.category.DEFAULT" /> | ||||
|                 <category android:name="android.intent.category.BROWSABLE" /> | ||||
|  | ||||
|                 <data android:scheme="http"/> | ||||
|                 <data android:scheme="https"/> | ||||
|                 <data android:sspPattern="bandcamp.com/?show=*"/> | ||||
|                 <data android:scheme="http" /> | ||||
|                 <data android:scheme="https" /> | ||||
|                 <data android:sspPattern="bandcamp.com/?show=*" /> | ||||
|             </intent-filter> | ||||
|  | ||||
|         </activity> | ||||
| @@ -383,11 +411,17 @@ | ||||
|             android:exported="false" /> | ||||
|  | ||||
|         <!-- opting out of sending metrics to Google in Android System WebView --> | ||||
|         <meta-data android:name="android.webkit.WebView.MetricsOptOut" android:value="true" /> | ||||
|         <meta-data | ||||
|             android:name="android.webkit.WebView.MetricsOptOut" | ||||
|             android:value="true" /> | ||||
|         <!-- see https://github.com/TeamNewPipe/NewPipe/issues/3947 --> | ||||
|         <!-- Version < 3.0. DeX Mode and Screen Mirroring support --> | ||||
|         <meta-data android:name="com.samsung.android.keepalive.density" android:value="true"/> | ||||
|         <meta-data | ||||
|             android:name="com.samsung.android.keepalive.density" | ||||
|             android:value="true" /> | ||||
|         <!-- Version >= 3.0. DeX Dual Mode support --> | ||||
|         <meta-data android:name="com.samsung.android.multidisplay.keep_process_alive" android:value="true"/> | ||||
|         <meta-data | ||||
|             android:name="com.samsung.android.multidisplay.keep_process_alive" | ||||
|             android:value="true" /> | ||||
|     </application> | ||||
| </manifest> | ||||
|   | ||||
| @@ -25,6 +25,7 @@ import android.view.ViewGroup; | ||||
| import androidx.annotation.IntDef; | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.core.os.BundleCompat; | ||||
| import androidx.lifecycle.Lifecycle; | ||||
| import androidx.viewpager.widget.PagerAdapter; | ||||
|  | ||||
| @@ -282,11 +283,9 @@ public abstract class FragmentStatePagerAdapterMenuWorkaround extends PagerAdapt | ||||
|     @Nullable | ||||
|     public Parcelable saveState() { | ||||
|         Bundle state = null; | ||||
|         if (mSavedState.size() > 0) { | ||||
|         if (!mSavedState.isEmpty()) { | ||||
|             state = new Bundle(); | ||||
|             final Fragment.SavedState[] fss = new Fragment.SavedState[mSavedState.size()]; | ||||
|             mSavedState.toArray(fss); | ||||
|             state.putParcelableArray("states", fss); | ||||
|             state.putParcelableArrayList("states", mSavedState); | ||||
|         } | ||||
|         for (int i = 0; i < mFragments.size(); i++) { | ||||
|             final Fragment f = mFragments.get(i); | ||||
| @@ -313,13 +312,12 @@ public abstract class FragmentStatePagerAdapterMenuWorkaround extends PagerAdapt | ||||
|         if (state != null) { | ||||
|             final Bundle bundle = (Bundle) state; | ||||
|             bundle.setClassLoader(loader); | ||||
|             final Parcelable[] fss = bundle.getParcelableArray("states"); | ||||
|             final var states = BundleCompat.getParcelableArrayList(bundle, "states", | ||||
|                     Fragment.SavedState.class); | ||||
|             mSavedState.clear(); | ||||
|             mFragments.clear(); | ||||
|             if (fss != null) { | ||||
|                 for (final Parcelable parcelable : fss) { | ||||
|                     mSavedState.add((Fragment.SavedState) parcelable); | ||||
|                 } | ||||
|             if (states != null) { | ||||
|                 mSavedState.addAll(states); | ||||
|             } | ||||
|             final Iterable<String> keys = bundle.keySet(); | ||||
|             for (final String key : keys) { | ||||
|   | ||||
| @@ -14,7 +14,6 @@ import androidx.coordinatorlayout.widget.CoordinatorLayout; | ||||
| import org.schabi.newpipe.R; | ||||
|  | ||||
| import java.lang.reflect.Field; | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
|  | ||||
| // See https://stackoverflow.com/questions/56849221#57997489 | ||||
| @@ -27,7 +26,7 @@ public final class FlingBehavior extends AppBarLayout.Behavior { | ||||
|  | ||||
|     private boolean allowScroll = true; | ||||
|     private final Rect globalRect = new Rect(); | ||||
|     private final List<Integer> skipInterceptionOfElements = Arrays.asList( | ||||
|     private final List<Integer> skipInterceptionOfElements = List.of( | ||||
|             R.id.itemsListPanel, R.id.playbackSeekBar, | ||||
|             R.id.playPauseButton, R.id.playPreviousButton, R.id.playNextButton); | ||||
|  | ||||
| @@ -67,7 +66,7 @@ public final class FlingBehavior extends AppBarLayout.Behavior { | ||||
|     public boolean onInterceptTouchEvent(@NonNull final CoordinatorLayout parent, | ||||
|                                          @NonNull final AppBarLayout child, | ||||
|                                          @NonNull final MotionEvent ev) { | ||||
|         for (final Integer element : skipInterceptionOfElements) { | ||||
|         for (final int element : skipInterceptionOfElements) { | ||||
|             final View view = child.findViewById(element); | ||||
|             if (view != null) { | ||||
|                 final boolean visible = view.getGlobalVisibleRect(globalRect); | ||||
| @@ -132,8 +131,8 @@ public final class FlingBehavior extends AppBarLayout.Behavior { | ||||
|         try { | ||||
|             final Class<?> headerBehaviorType = this.getClass().getSuperclass().getSuperclass(); | ||||
|             if (headerBehaviorType != null) { | ||||
|                 final Field field | ||||
|                         = headerBehaviorType.getDeclaredField("lastNestedScrollingChildRef"); | ||||
|                 final Field field = | ||||
|                         headerBehaviorType.getDeclaredField("lastNestedScrollingChildRef"); | ||||
|                 field.setAccessible(true); | ||||
|                 return field; | ||||
|             } | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| package org.schabi.newpipe; | ||||
|  | ||||
| import android.app.Application; | ||||
| import android.content.Context; | ||||
| import android.content.SharedPreferences; | ||||
| import android.util.Log; | ||||
| @@ -7,7 +8,6 @@ import android.util.Log; | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.core.app.NotificationChannelCompat; | ||||
| import androidx.core.app.NotificationManagerCompat; | ||||
| import androidx.multidex.MultiDexApplication; | ||||
| import androidx.preference.PreferenceManager; | ||||
|  | ||||
| import com.jakewharton.processphoenix.ProcessPhoenix; | ||||
| @@ -20,16 +20,17 @@ import org.schabi.newpipe.extractor.downloader.Downloader; | ||||
| import org.schabi.newpipe.ktx.ExceptionUtils; | ||||
| import org.schabi.newpipe.settings.NewPipeSettings; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.PicassoHelper; | ||||
| import org.schabi.newpipe.util.image.ImageStrategy; | ||||
| import org.schabi.newpipe.util.image.PicassoHelper; | ||||
| import org.schabi.newpipe.util.ServiceHelper; | ||||
| import org.schabi.newpipe.util.StateSaver; | ||||
| import org.schabi.newpipe.util.image.PreferredImageQuality; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.io.InterruptedIOException; | ||||
| import java.net.SocketException; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
|  | ||||
| import io.reactivex.rxjava3.exceptions.CompositeException; | ||||
| import io.reactivex.rxjava3.exceptions.MissingBackpressureException; | ||||
| @@ -56,9 +57,11 @@ import io.reactivex.rxjava3.plugins.RxJavaPlugins; | ||||
|  * along with NewPipe.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
|  | ||||
| public class App extends MultiDexApplication { | ||||
| public class App extends Application { | ||||
|     public static final String PACKAGE_NAME = BuildConfig.APPLICATION_ID; | ||||
|     private static final String TAG = App.class.toString(); | ||||
|  | ||||
|     private boolean isFirstRun = false; | ||||
|     private static App app; | ||||
|  | ||||
|     @NonNull | ||||
| @@ -84,7 +87,13 @@ public class App extends MultiDexApplication { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Initialize settings first because others inits can use its values | ||||
|         // check if the last used preference version is set | ||||
|         // to determine whether this is the first app run | ||||
|         final int lastUsedPrefVersion = PreferenceManager.getDefaultSharedPreferences(this) | ||||
|                 .getInt(getString(R.string.last_used_preferences_version), -1); | ||||
|         isFirstRun = lastUsedPrefVersion == -1; | ||||
|  | ||||
|         // Initialize settings first because other initializations can use its values | ||||
|         NewPipeSettings.initSettings(this); | ||||
|  | ||||
|         NewPipe.init(getDownloader(), | ||||
| @@ -100,8 +109,9 @@ public class App extends MultiDexApplication { | ||||
|         // Initialize image loader | ||||
|         final SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(this); | ||||
|         PicassoHelper.init(this); | ||||
|         PicassoHelper.setShouldLoadImages( | ||||
|                 prefs.getBoolean(getString(R.string.download_thumbnail_key), true)); | ||||
|         ImageStrategy.setPreferredImageQuality(PreferredImageQuality.fromPreferenceKey(this, | ||||
|                 prefs.getString(getString(R.string.image_quality_key), | ||||
|                         getString(R.string.image_quality_default)))); | ||||
|         PicassoHelper.setIndicatorsEnabled(MainActivity.DEBUG | ||||
|                 && prefs.getBoolean(getString(R.string.show_image_indicators_key), false)); | ||||
|  | ||||
| @@ -140,7 +150,7 @@ public class App extends MultiDexApplication { | ||||
|                 if (throwable instanceof UndeliverableException) { | ||||
|                     // As UndeliverableException is a wrapper, | ||||
|                     // get the cause of it to get the "real" exception | ||||
|                     actualThrowable = throwable.getCause(); | ||||
|                     actualThrowable = Objects.requireNonNull(throwable.getCause()); | ||||
|                 } else { | ||||
|                     actualThrowable = throwable; | ||||
|                 } | ||||
| @@ -149,7 +159,7 @@ public class App extends MultiDexApplication { | ||||
|                 if (actualThrowable instanceof CompositeException) { | ||||
|                     errors = ((CompositeException) actualThrowable).getExceptions(); | ||||
|                 } else { | ||||
|                     errors = Collections.singletonList(actualThrowable); | ||||
|                     errors = List.of(actualThrowable); | ||||
|                 } | ||||
|  | ||||
|                 for (final Throwable error : errors) { | ||||
| @@ -205,7 +215,7 @@ public class App extends MultiDexApplication { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         final CoreConfigurationBuilder acraConfig = new CoreConfigurationBuilder(this) | ||||
|         final CoreConfigurationBuilder acraConfig = new CoreConfigurationBuilder() | ||||
|                 .withBuildConfigClass(BuildConfig.class); | ||||
|         ACRA.init(this, acraConfig); | ||||
|     } | ||||
| @@ -213,41 +223,37 @@ public class App extends MultiDexApplication { | ||||
|     private void initNotificationChannels() { | ||||
|         // Keep the importance below DEFAULT to avoid making noise on every notification update for | ||||
|         // the main and update channels | ||||
|         final List<NotificationChannelCompat> notificationChannelCompats = new ArrayList<>(); | ||||
|         notificationChannelCompats.add(new NotificationChannelCompat | ||||
|                 .Builder(getString(R.string.notification_channel_id), | ||||
|         final List<NotificationChannelCompat> notificationChannelCompats = List.of( | ||||
|                 new NotificationChannelCompat.Builder(getString(R.string.notification_channel_id), | ||||
|                         NotificationManagerCompat.IMPORTANCE_LOW) | ||||
|                 .setName(getString(R.string.notification_channel_name)) | ||||
|                 .setDescription(getString(R.string.notification_channel_description)) | ||||
|                 .build()); | ||||
|  | ||||
|         notificationChannelCompats.add(new NotificationChannelCompat | ||||
|                 .Builder(getString(R.string.app_update_notification_channel_id), | ||||
|                         .setName(getString(R.string.notification_channel_name)) | ||||
|                         .setDescription(getString(R.string.notification_channel_description)) | ||||
|                         .build(), | ||||
|                 new NotificationChannelCompat | ||||
|                         .Builder(getString(R.string.app_update_notification_channel_id), | ||||
|                         NotificationManagerCompat.IMPORTANCE_LOW) | ||||
|                 .setName(getString(R.string.app_update_notification_channel_name)) | ||||
|                 .setDescription(getString(R.string.app_update_notification_channel_description)) | ||||
|                 .build()); | ||||
|  | ||||
|         notificationChannelCompats.add(new NotificationChannelCompat | ||||
|                 .Builder(getString(R.string.hash_channel_id), | ||||
|                         .setName(getString(R.string.app_update_notification_channel_name)) | ||||
|                         .setDescription( | ||||
|                                 getString(R.string.app_update_notification_channel_description)) | ||||
|                         .build(), | ||||
|                 new NotificationChannelCompat.Builder(getString(R.string.hash_channel_id), | ||||
|                         NotificationManagerCompat.IMPORTANCE_HIGH) | ||||
|                 .setName(getString(R.string.hash_channel_name)) | ||||
|                 .setDescription(getString(R.string.hash_channel_description)) | ||||
|                 .build()); | ||||
|  | ||||
|         notificationChannelCompats.add(new NotificationChannelCompat | ||||
|                 .Builder(getString(R.string.error_report_channel_id), | ||||
|                         .setName(getString(R.string.hash_channel_name)) | ||||
|                         .setDescription(getString(R.string.hash_channel_description)) | ||||
|                         .build(), | ||||
|                 new NotificationChannelCompat.Builder(getString(R.string.error_report_channel_id), | ||||
|                         NotificationManagerCompat.IMPORTANCE_LOW) | ||||
|                 .setName(getString(R.string.error_report_channel_name)) | ||||
|                 .setDescription(getString(R.string.error_report_channel_description)) | ||||
|                 .build()); | ||||
|  | ||||
|         notificationChannelCompats.add(new NotificationChannelCompat | ||||
|                 .Builder(getString(R.string.streams_notification_channel_id), | ||||
|                     NotificationManagerCompat.IMPORTANCE_DEFAULT) | ||||
|                 .setName(getString(R.string.streams_notification_channel_name)) | ||||
|                 .setDescription(getString(R.string.streams_notification_channel_description)) | ||||
|                 .build()); | ||||
|                         .setName(getString(R.string.error_report_channel_name)) | ||||
|                         .setDescription(getString(R.string.error_report_channel_description)) | ||||
|                         .build(), | ||||
|                 new NotificationChannelCompat | ||||
|                         .Builder(getString(R.string.streams_notification_channel_id), | ||||
|                         NotificationManagerCompat.IMPORTANCE_DEFAULT) | ||||
|                         .setName(getString(R.string.streams_notification_channel_name)) | ||||
|                         .setDescription( | ||||
|                                 getString(R.string.streams_notification_channel_description)) | ||||
|                         .build() | ||||
|         ); | ||||
|  | ||||
|         final NotificationManagerCompat notificationManager = NotificationManagerCompat.from(this); | ||||
|         notificationManager.createNotificationChannelsCompat(notificationChannelCompats); | ||||
| @@ -257,4 +263,7 @@ public class App extends MultiDexApplication { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     public boolean isFirstRun() { | ||||
|         return isFirstRun; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -12,7 +12,6 @@ import androidx.fragment.app.FragmentManager; | ||||
|  | ||||
| import icepick.Icepick; | ||||
| import icepick.State; | ||||
| import leakcanary.AppWatcher; | ||||
|  | ||||
| public abstract class BaseFragment extends Fragment { | ||||
|     protected final String TAG = getClass().getSimpleName() + "@" + Integer.toHexString(hashCode()); | ||||
| @@ -77,20 +76,33 @@ public abstract class BaseFragment extends Fragment { | ||||
|     protected void onRestoreInstanceState(@NonNull final Bundle savedInstanceState) { | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroy() { | ||||
|         super.onDestroy(); | ||||
|  | ||||
|         AppWatcher.INSTANCE.getObjectWatcher().watch(this); | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Init | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     /** | ||||
|      * This method is called in {@link #onViewCreated(View, Bundle)} to initialize the views. | ||||
|      * | ||||
|      * <p> | ||||
|      * {@link #initListeners()} is called after this method to initialize the corresponding | ||||
|      * listeners. | ||||
|      * </p> | ||||
|      * @param rootView The inflated view for this fragment | ||||
|      *                 (provided by {@link #onViewCreated(View, Bundle)}) | ||||
|      * @param savedInstanceState The saved state of this fragment | ||||
|  *                               (provided by {@link #onViewCreated(View, Bundle)}) | ||||
|      */ | ||||
|     protected void initViews(final View rootView, final Bundle savedInstanceState) { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Initialize the listeners for this fragment. | ||||
|      * | ||||
|      * <p> | ||||
|      * This method is called after {@link #initViews(View, Bundle)} | ||||
|      * in {@link #onViewCreated(View, Bundle)}. | ||||
|      * </p> | ||||
|      */ | ||||
|     protected void initListeners() { | ||||
|     } | ||||
|  | ||||
| @@ -108,9 +120,20 @@ public abstract class BaseFragment extends Fragment { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Finds the root fragment by looping through all of the parent fragments. The root fragment | ||||
|      * is supposed to be {@link org.schabi.newpipe.fragments.MainFragment}, and is the fragment that | ||||
|      * handles keeping the backstack of opened fragments in NewPipe, and also the player bottom | ||||
|      * sheet. This function therefore returns the fragment manager of said fragment. | ||||
|      * | ||||
|      * @return the fragment manager of the root fragment, i.e. | ||||
|      *         {@link org.schabi.newpipe.fragments.MainFragment} | ||||
|      */ | ||||
|     protected FragmentManager getFM() { | ||||
|         return getParentFragment() == null | ||||
|                 ? getFragmentManager() | ||||
|                 : getParentFragment().getFragmentManager(); | ||||
|         Fragment current = this; | ||||
|         while (current.getParentFragment() != null) { | ||||
|             current = current.getParentFragment(); | ||||
|         } | ||||
|         return current.getFragmentManager(); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| package org.schabi.newpipe; | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.os.Build; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| @@ -12,40 +11,27 @@ import org.schabi.newpipe.extractor.downloader.Downloader; | ||||
| import org.schabi.newpipe.extractor.downloader.Request; | ||||
| import org.schabi.newpipe.extractor.downloader.Response; | ||||
| import org.schabi.newpipe.extractor.exceptions.ReCaptchaException; | ||||
| import org.schabi.newpipe.util.CookieUtils; | ||||
| import org.schabi.newpipe.util.InfoCache; | ||||
| import org.schabi.newpipe.util.TLSSocketFactoryCompat; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.security.KeyManagementException; | ||||
| import java.security.KeyStore; | ||||
| import java.security.KeyStoreException; | ||||
| import java.security.NoSuchAlgorithmException; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Objects; | ||||
| import java.util.concurrent.TimeUnit; | ||||
| import java.util.stream.Collectors; | ||||
| import java.util.stream.Stream; | ||||
|  | ||||
| import javax.net.ssl.SSLSocketFactory; | ||||
| import javax.net.ssl.TrustManager; | ||||
| import javax.net.ssl.TrustManagerFactory; | ||||
| import javax.net.ssl.X509TrustManager; | ||||
|  | ||||
| import okhttp3.CipherSuite; | ||||
| import okhttp3.ConnectionSpec; | ||||
| import okhttp3.OkHttpClient; | ||||
| import okhttp3.RequestBody; | ||||
| import okhttp3.ResponseBody; | ||||
|  | ||||
| import static org.schabi.newpipe.MainActivity.DEBUG; | ||||
|  | ||||
| public final class DownloaderImpl extends Downloader { | ||||
|     public static final String USER_AGENT | ||||
|             = "Mozilla/5.0 (Windows NT 10.0; rv:78.0) Gecko/20100101 Firefox/78.0"; | ||||
|     public static final String YOUTUBE_RESTRICTED_MODE_COOKIE_KEY | ||||
|             = "youtube_restricted_mode_key"; | ||||
|     public static final String USER_AGENT = | ||||
|             "Mozilla/5.0 (Windows NT 10.0; rv:91.0) Gecko/20100101 Firefox/91.0"; | ||||
|     public static final String YOUTUBE_RESTRICTED_MODE_COOKIE_KEY = | ||||
|             "youtube_restricted_mode_key"; | ||||
|     public static final String YOUTUBE_RESTRICTED_MODE_COOKIE = "PREF=f2=8000000"; | ||||
|     public static final String YOUTUBE_DOMAIN = "youtube.com"; | ||||
|  | ||||
| @@ -54,9 +40,6 @@ public final class DownloaderImpl extends Downloader { | ||||
|     private final OkHttpClient client; | ||||
|  | ||||
|     private DownloaderImpl(final OkHttpClient.Builder builder) { | ||||
|         if (Build.VERSION.SDK_INT == Build.VERSION_CODES.KITKAT) { | ||||
|             enableModernTLS(builder); | ||||
|         } | ||||
|         this.client = builder | ||||
|                 .readTimeout(30, TimeUnit.SECONDS) | ||||
| //                .cache(new Cache(new File(context.getExternalCacheDir(), "okhttp"), | ||||
| @@ -81,69 +64,16 @@ public final class DownloaderImpl extends Downloader { | ||||
|         return instance; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Enable TLS 1.2 and 1.1 on Android Kitkat. This function is mostly taken | ||||
|      * from the documentation of OkHttpClient.Builder.sslSocketFactory(_,_). | ||||
|      * <p> | ||||
|      * If there is an error, the function will safely fall back to doing nothing | ||||
|      * and printing the error to the console. | ||||
|      * </p> | ||||
|      * | ||||
|      * @param builder The HTTPClient Builder on which TLS is enabled on (will be modified in-place) | ||||
|      */ | ||||
|     private static void enableModernTLS(final OkHttpClient.Builder builder) { | ||||
|         try { | ||||
|             // get the default TrustManager | ||||
|             final TrustManagerFactory trustManagerFactory = TrustManagerFactory.getInstance( | ||||
|                     TrustManagerFactory.getDefaultAlgorithm()); | ||||
|             trustManagerFactory.init((KeyStore) null); | ||||
|             final TrustManager[] trustManagers = trustManagerFactory.getTrustManagers(); | ||||
|             if (trustManagers.length != 1 || !(trustManagers[0] instanceof X509TrustManager)) { | ||||
|                 throw new IllegalStateException("Unexpected default trust managers:" | ||||
|                         + Arrays.toString(trustManagers)); | ||||
|             } | ||||
|             final X509TrustManager trustManager = (X509TrustManager) trustManagers[0]; | ||||
|  | ||||
|             // insert our own TLSSocketFactory | ||||
|             final SSLSocketFactory sslSocketFactory = TLSSocketFactoryCompat.getInstance(); | ||||
|  | ||||
|             builder.sslSocketFactory(sslSocketFactory, trustManager); | ||||
|  | ||||
|             // This will try to enable all modern CipherSuites(+2 more) | ||||
|             // that are supported on the device. | ||||
|             // Necessary because some servers (e.g. Framatube.org) | ||||
|             // don't support the old cipher suites. | ||||
|             // https://github.com/square/okhttp/issues/4053#issuecomment-402579554 | ||||
|             final List<CipherSuite> cipherSuites = | ||||
|                     new ArrayList<>(ConnectionSpec.MODERN_TLS.cipherSuites()); | ||||
|             cipherSuites.add(CipherSuite.TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA); | ||||
|             cipherSuites.add(CipherSuite.TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA); | ||||
|             final ConnectionSpec legacyTLS = new ConnectionSpec.Builder(ConnectionSpec.MODERN_TLS) | ||||
|                     .cipherSuites(cipherSuites.toArray(new CipherSuite[0])) | ||||
|                     .build(); | ||||
|  | ||||
|             builder.connectionSpecs(Arrays.asList(legacyTLS, ConnectionSpec.CLEARTEXT)); | ||||
|         } catch (final KeyManagementException | NoSuchAlgorithmException | KeyStoreException e) { | ||||
|             if (DEBUG) { | ||||
|                 e.printStackTrace(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public String getCookies(final String url) { | ||||
|         final List<String> resultCookies = new ArrayList<>(); | ||||
|         if (url.contains(YOUTUBE_DOMAIN)) { | ||||
|             final String youtubeCookie = getCookie(YOUTUBE_RESTRICTED_MODE_COOKIE_KEY); | ||||
|             if (youtubeCookie != null) { | ||||
|                 resultCookies.add(youtubeCookie); | ||||
|             } | ||||
|         } | ||||
|         final String youtubeCookie = url.contains(YOUTUBE_DOMAIN) | ||||
|                 ? getCookie(YOUTUBE_RESTRICTED_MODE_COOKIE_KEY) : null; | ||||
|  | ||||
|         // Recaptcha cookie is always added TODO: not sure if this is necessary | ||||
|         final String recaptchaCookie = getCookie(ReCaptchaActivity.RECAPTCHA_COOKIES_KEY); | ||||
|         if (recaptchaCookie != null) { | ||||
|             resultCookies.add(recaptchaCookie); | ||||
|         } | ||||
|         return CookieUtils.concatCookies(resultCookies); | ||||
|         return Stream.of(youtubeCookie, getCookie(ReCaptchaActivity.RECAPTCHA_COOKIES_KEY)) | ||||
|                 .filter(Objects::nonNull) | ||||
|                 .flatMap(cookies -> Arrays.stream(cookies.split("; *"))) | ||||
|                 .distinct() | ||||
|                 .collect(Collectors.joining("; ")); | ||||
|     } | ||||
|  | ||||
|     public String getCookie(final String key) { | ||||
| @@ -203,7 +133,7 @@ public final class DownloaderImpl extends Downloader { | ||||
|  | ||||
|         RequestBody requestBody = null; | ||||
|         if (dataToSend != null) { | ||||
|             requestBody = RequestBody.create(null, dataToSend); | ||||
|             requestBody = RequestBody.create(dataToSend); | ||||
|         } | ||||
|  | ||||
|         final okhttp3.Request.Builder requestBuilder = new okhttp3.Request.Builder() | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package org.schabi.newpipe; | ||||
| import android.annotation.SuppressLint; | ||||
| import android.app.Activity; | ||||
| import android.content.Intent; | ||||
| import android.os.Build; | ||||
| import android.os.Bundle; | ||||
|  | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| @@ -44,11 +43,7 @@ public class ExitActivity extends Activity { | ||||
|     protected void onCreate(final Bundle savedInstanceState) { | ||||
|         super.onCreate(savedInstanceState); | ||||
|  | ||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { | ||||
|             finishAndRemoveTask(); | ||||
|         } else { | ||||
|             finish(); | ||||
|         } | ||||
|         finishAndRemoveTask(); | ||||
|  | ||||
|         NavigationHelper.restartApp(this); | ||||
|     } | ||||
|   | ||||
| @@ -28,7 +28,6 @@ import android.content.Intent; | ||||
| import android.content.IntentFilter; | ||||
| import android.content.SharedPreferences; | ||||
| import android.content.pm.PackageManager; | ||||
| import android.os.Build; | ||||
| import android.os.Bundle; | ||||
| import android.os.Handler; | ||||
| import android.os.Looper; | ||||
| @@ -45,6 +44,7 @@ import android.widget.FrameLayout; | ||||
| import android.widget.Spinner; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.appcompat.app.ActionBar; | ||||
| import androidx.appcompat.app.ActionBarDrawerToggle; | ||||
| import androidx.appcompat.app.AppCompatActivity; | ||||
| @@ -52,6 +52,7 @@ import androidx.core.app.ActivityCompat; | ||||
| import androidx.core.view.GravityCompat; | ||||
| import androidx.drawerlayout.widget.DrawerLayout; | ||||
| import androidx.fragment.app.Fragment; | ||||
| import androidx.fragment.app.FragmentContainerView; | ||||
| import androidx.fragment.app.FragmentManager; | ||||
| import androidx.preference.PreferenceManager; | ||||
|  | ||||
| @@ -65,17 +66,20 @@ import org.schabi.newpipe.databinding.ToolbarLayoutBinding; | ||||
| import org.schabi.newpipe.error.ErrorUtil; | ||||
| import org.schabi.newpipe.extractor.NewPipe; | ||||
| import org.schabi.newpipe.extractor.StreamingService; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.extractor.services.peertube.PeertubeInstance; | ||||
| import org.schabi.newpipe.fragments.BackPressable; | ||||
| import org.schabi.newpipe.fragments.MainFragment; | ||||
| import org.schabi.newpipe.fragments.detail.VideoDetailFragment; | ||||
| import org.schabi.newpipe.fragments.list.comments.CommentRepliesFragment; | ||||
| import org.schabi.newpipe.fragments.list.search.SearchFragment; | ||||
| import org.schabi.newpipe.local.feed.notifications.NotificationWorker; | ||||
| import org.schabi.newpipe.player.Player; | ||||
| import org.schabi.newpipe.player.event.OnKeyDownListener; | ||||
| import org.schabi.newpipe.player.helper.PlayerHolder; | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueue; | ||||
| import org.schabi.newpipe.settings.UpdateSettingsFragment; | ||||
| import org.schabi.newpipe.util.Constants; | ||||
| import org.schabi.newpipe.util.DeviceUtils; | ||||
| import org.schabi.newpipe.util.KioskTranslator; | ||||
| @@ -83,10 +87,10 @@ import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.PeertubeHelper; | ||||
| import org.schabi.newpipe.util.PermissionHelper; | ||||
| import org.schabi.newpipe.util.ReleaseVersionUtil; | ||||
| import org.schabi.newpipe.util.SerializedCache; | ||||
| import org.schabi.newpipe.util.ServiceHelper; | ||||
| import org.schabi.newpipe.util.StateSaver; | ||||
| import org.schabi.newpipe.util.TLSSocketFactoryCompat; | ||||
| import org.schabi.newpipe.util.ThemeHelper; | ||||
| import org.schabi.newpipe.views.FocusOverlayView; | ||||
|  | ||||
| @@ -131,11 +135,6 @@ public class MainActivity extends AppCompatActivity { | ||||
|                     + "savedInstanceState = [" + savedInstanceState + "]"); | ||||
|         } | ||||
|  | ||||
|         // enable TLS1.1/1.2 for kitkat devices, to fix download and play for media.ccc.de sources | ||||
|         if (Build.VERSION.SDK_INT == Build.VERSION_CODES.KITKAT) { | ||||
|             TLSSocketFactoryCompat.setAsDefault(); | ||||
|         } | ||||
|  | ||||
|         ThemeHelper.setDayNightMode(this); | ||||
|         ThemeHelper.setTheme(this, ServiceHelper.getSelectedServiceId(this)); | ||||
|  | ||||
| @@ -164,9 +163,17 @@ public class MainActivity extends AppCompatActivity { | ||||
|         } | ||||
|         openMiniPlayerUponPlayerStarted(); | ||||
|  | ||||
|         // Schedule worker for checking for new streams and creating corresponding notifications | ||||
|         // if this is enabled by the user. | ||||
|         NotificationWorker.initialize(this); | ||||
|         if (PermissionHelper.checkPostNotificationsPermission(this, | ||||
|                 PermissionHelper.POST_NOTIFICATIONS_REQUEST_CODE)) { | ||||
|             // Schedule worker for checking for new streams and creating corresponding notifications | ||||
|             // if this is enabled by the user. | ||||
|             NotificationWorker.initialize(this); | ||||
|         } | ||||
|         if (!UpdateSettingsFragment.wasUserAskedForConsent(this) | ||||
|                 && !App.getApp().isFirstRun() | ||||
|                 && ReleaseVersionUtil.INSTANCE.isReleaseApk()) { | ||||
|             UpdateSettingsFragment.askForConsentToUpdateChecks(this); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -176,10 +183,11 @@ public class MainActivity extends AppCompatActivity { | ||||
|         final App app = App.getApp(); | ||||
|         final SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(app); | ||||
|  | ||||
|         if (prefs.getBoolean(app.getString(R.string.update_app_key), true)) { | ||||
|         if (prefs.getBoolean(app.getString(R.string.update_app_key), false) | ||||
|                 && prefs.getBoolean(app.getString(R.string.update_check_consent_key), false)) { | ||||
|             // Start the worker which is checking all conditions | ||||
|             // and eventually searching for a new version. | ||||
|             NewVersionWorker.enqueueNewVersionCheckingWork(app); | ||||
|             NewVersionWorker.enqueueNewVersionCheckingWork(app, false); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -223,14 +231,14 @@ public class MainActivity extends AppCompatActivity { | ||||
|         final int currentServiceId = ServiceHelper.getSelectedServiceId(this); | ||||
|         final StreamingService service = NewPipe.getService(currentServiceId); | ||||
|  | ||||
|         int kioskId = 0; | ||||
|         int kioskMenuItemId = 0; | ||||
|  | ||||
|         for (final String ks : service.getKioskList().getAvailableKiosks()) { | ||||
|             drawerLayoutBinding.navigation.getMenu() | ||||
|                     .add(R.id.menu_tabs_group, kioskId, 0, KioskTranslator | ||||
|                     .add(R.id.menu_tabs_group, kioskMenuItemId, 0, KioskTranslator | ||||
|                             .getTranslatedKioskName(ks, this)) | ||||
|                     .setIcon(KioskTranslator.getKioskIcon(ks)); | ||||
|             kioskId++; | ||||
|             kioskMenuItemId++; | ||||
|         } | ||||
|  | ||||
|         drawerLayoutBinding.navigation.getMenu() | ||||
| @@ -239,7 +247,7 @@ public class MainActivity extends AppCompatActivity { | ||||
|                 .setIcon(R.drawable.ic_tv); | ||||
|         drawerLayoutBinding.navigation.getMenu() | ||||
|                 .add(R.id.menu_tabs_group, ITEM_ID_FEED, ORDER, R.string.fragment_feed_title) | ||||
|                 .setIcon(R.drawable.ic_rss_feed); | ||||
|                 .setIcon(R.drawable.ic_subscriptions); | ||||
|         drawerLayoutBinding.navigation.getMenu() | ||||
|                 .add(R.id.menu_tabs_group, ITEM_ID_BOOKMARKS, ORDER, R.string.tab_bookmarks) | ||||
|                 .setIcon(R.drawable.ic_bookmark); | ||||
| @@ -310,20 +318,16 @@ public class MainActivity extends AppCompatActivity { | ||||
|                 NavigationHelper.openStatisticFragment(getSupportFragmentManager()); | ||||
|                 break; | ||||
|             default: | ||||
|                 final int currentServiceId = ServiceHelper.getSelectedServiceId(this); | ||||
|                 final StreamingService service = NewPipe.getService(currentServiceId); | ||||
|                 String serviceName = ""; | ||||
|  | ||||
|                 int kioskId = 0; | ||||
|                 for (final String ks : service.getKioskList().getAvailableKiosks()) { | ||||
|                     if (kioskId == item.getItemId()) { | ||||
|                         serviceName = ks; | ||||
|                 final StreamingService currentService = ServiceHelper.getSelectedService(this); | ||||
|                 int kioskMenuItemId = 0; | ||||
|                 for (final String kioskId : currentService.getKioskList().getAvailableKiosks()) { | ||||
|                     if (kioskMenuItemId == item.getItemId()) { | ||||
|                         NavigationHelper.openKioskFragment(getSupportFragmentManager(), | ||||
|                                 currentService.getServiceId(), kioskId); | ||||
|                         break; | ||||
|                     } | ||||
|                     kioskId++; | ||||
|                     kioskMenuItemId++; | ||||
|                 } | ||||
|  | ||||
|                 NavigationHelper.openKioskFragment(getSupportFragmentManager(), currentServiceId, | ||||
|                         serviceName); | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
| @@ -381,8 +385,7 @@ public class MainActivity extends AppCompatActivity { | ||||
|  | ||||
|     private void showServices() { | ||||
|         for (final StreamingService s : NewPipe.getServices()) { | ||||
|             final String title = s.getServiceInfo().getName() | ||||
|                     + (ServiceHelper.isBeta(s) ? " (beta)" : ""); | ||||
|             final String title = s.getServiceInfo().getName(); | ||||
|  | ||||
|             final MenuItem menuItem = drawerLayoutBinding.navigation.getMenu() | ||||
|                     .add(R.id.menu_services_group, s.getServiceId(), ORDER, title) | ||||
| @@ -390,7 +393,7 @@ public class MainActivity extends AppCompatActivity { | ||||
|  | ||||
|             // peertube specifics | ||||
|             if (s.getServiceId() == 3) { | ||||
|                 enhancePeertubeMenu(s, menuItem); | ||||
|                 enhancePeertubeMenu(menuItem); | ||||
|             } | ||||
|         } | ||||
|         drawerLayoutBinding.navigation.getMenu() | ||||
| @@ -398,9 +401,9 @@ public class MainActivity extends AppCompatActivity { | ||||
|                 .setChecked(true); | ||||
|     } | ||||
|  | ||||
|     private void enhancePeertubeMenu(final StreamingService s, final MenuItem menuItem) { | ||||
|     private void enhancePeertubeMenu(final MenuItem menuItem) { | ||||
|         final PeertubeInstance currentInstance = PeertubeHelper.getCurrentInstance(); | ||||
|         menuItem.setTitle(currentInstance.getName() + (ServiceHelper.isBeta(s) ? " (beta)" : "")); | ||||
|         menuItem.setTitle(currentInstance.getName()); | ||||
|         final Spinner spinner = InstanceSpinnerLayoutBinding.inflate(LayoutInflater.from(this)) | ||||
|                 .getRoot(); | ||||
|         final List<PeertubeInstance> instances = PeertubeHelper.getInstanceList(this); | ||||
| @@ -480,8 +483,8 @@ public class MainActivity extends AppCompatActivity { | ||||
|             ErrorUtil.showUiErrorSnackbar(this, "Setting up service toggle", e); | ||||
|         } | ||||
|  | ||||
|         final SharedPreferences sharedPreferences | ||||
|                 = PreferenceManager.getDefaultSharedPreferences(this); | ||||
|         final SharedPreferences sharedPreferences = | ||||
|                 PreferenceManager.getDefaultSharedPreferences(this); | ||||
|         if (sharedPreferences.getBoolean(Constants.KEY_THEME_CHANGE, false)) { | ||||
|             if (DEBUG) { | ||||
|                 Log.d(TAG, "Theme has changed, recreating activity..."); | ||||
| @@ -555,14 +558,21 @@ public class MainActivity extends AppCompatActivity { | ||||
|         // interacts with a fragment inside fragment_holder so all back presses should be | ||||
|         // handled by it | ||||
|         if (bottomSheetHiddenOrCollapsed()) { | ||||
|             final Fragment fragment = getSupportFragmentManager() | ||||
|                     .findFragmentById(R.id.fragment_holder); | ||||
|             final FragmentManager fm = getSupportFragmentManager(); | ||||
|             final Fragment fragment = fm.findFragmentById(R.id.fragment_holder); | ||||
|             // If current fragment implements BackPressable (i.e. can/wanna handle back press) | ||||
|             // delegate the back press to it | ||||
|             if (fragment instanceof BackPressable) { | ||||
|                 if (((BackPressable) fragment).onBackPressed()) { | ||||
|                     return; | ||||
|                 } | ||||
|             } else if (fragment instanceof CommentRepliesFragment) { | ||||
|                 // expand DetailsFragment if CommentRepliesFragment was opened | ||||
|                 // to show the top level comments again | ||||
|                 // Expand DetailsFragment if CommentRepliesFragment was opened | ||||
|                 // and no other CommentRepliesFragments are on top of the back stack | ||||
|                 // to show the top level comments again. | ||||
|                 openDetailFragmentFromCommentReplies(fm, false); | ||||
|             } | ||||
|  | ||||
|         } else { | ||||
| @@ -607,6 +617,9 @@ public class MainActivity extends AppCompatActivity { | ||||
|                     ((VideoDetailFragment) fragment).openDownloadDialog(); | ||||
|                 } | ||||
|                 break; | ||||
|             case PermissionHelper.POST_NOTIFICATIONS_REQUEST_CODE: | ||||
|                 NotificationWorker.initialize(this); | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -635,10 +648,17 @@ public class MainActivity extends AppCompatActivity { | ||||
|      * </pre> | ||||
|      */ | ||||
|     private void onHomeButtonPressed() { | ||||
|         // If search fragment wasn't found in the backstack... | ||||
|         if (!NavigationHelper.tryGotoSearchFragment(getSupportFragmentManager())) { | ||||
|             // ...go to the main fragment | ||||
|             NavigationHelper.gotoMainFragment(getSupportFragmentManager()); | ||||
|         final FragmentManager fm = getSupportFragmentManager(); | ||||
|         final Fragment fragment = fm.findFragmentById(R.id.fragment_holder); | ||||
|  | ||||
|         if (fragment instanceof CommentRepliesFragment) { | ||||
|             // Expand DetailsFragment if CommentRepliesFragment was opened | ||||
|             // and no other CommentRepliesFragments are on top of the back stack | ||||
|             // to show the top level comments again. | ||||
|             openDetailFragmentFromCommentReplies(fm, true); | ||||
|         } else if (!NavigationHelper.tryGotoSearchFragment(fm)) { | ||||
|             // If search fragment wasn't found in the backstack go to the main fragment | ||||
|             NavigationHelper.gotoMainFragment(fm); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -653,8 +673,8 @@ public class MainActivity extends AppCompatActivity { | ||||
|         } | ||||
|         super.onCreateOptionsMenu(menu); | ||||
|  | ||||
|         final Fragment fragment | ||||
|                 = getSupportFragmentManager().findFragmentById(R.id.fragment_holder); | ||||
|         final Fragment fragment = | ||||
|                 getSupportFragmentManager().findFragmentById(R.id.fragment_holder); | ||||
|         if (!(fragment instanceof SearchFragment)) { | ||||
|             toolbarLayoutBinding.toolbarSearchContainer.getRoot().setVisibility(View.GONE); | ||||
|         } | ||||
| @@ -721,7 +741,7 @@ public class MainActivity extends AppCompatActivity { | ||||
|             if (toggle != null) { | ||||
|                 toggle.syncState(); | ||||
|                 toolbarLayoutBinding.toolbar.setNavigationOnClickListener(v -> mainBinding.getRoot() | ||||
|                         .openDrawer(GravityCompat.START)); | ||||
|                         .open()); | ||||
|                 mainBinding.getRoot().setDrawerLockMode(DrawerLayout.LOCK_MODE_UNDEFINED); | ||||
|             } | ||||
|         } else { | ||||
| @@ -834,6 +854,68 @@ public class MainActivity extends AppCompatActivity { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void openDetailFragmentFromCommentReplies( | ||||
|             @NonNull final FragmentManager fm, | ||||
|             final boolean popBackStack | ||||
|     ) { | ||||
|         // obtain the name of the fragment under the replies fragment that's going to be popped | ||||
|         @Nullable final String fragmentUnderEntryName; | ||||
|         if (fm.getBackStackEntryCount() < 2) { | ||||
|             fragmentUnderEntryName = null; | ||||
|         } else { | ||||
|             fragmentUnderEntryName = fm.getBackStackEntryAt(fm.getBackStackEntryCount() - 2) | ||||
|                     .getName(); | ||||
|         } | ||||
|  | ||||
|         // the root comment is the comment for which the user opened the replies page | ||||
|         @Nullable final CommentRepliesFragment repliesFragment = | ||||
|                 (CommentRepliesFragment) fm.findFragmentByTag(CommentRepliesFragment.TAG); | ||||
|         @Nullable final CommentsInfoItem rootComment = | ||||
|                 repliesFragment == null ? null : repliesFragment.getCommentsInfoItem(); | ||||
|  | ||||
|         // sometimes this function pops the backstack, other times it's handled by the system | ||||
|         if (popBackStack) { | ||||
|             fm.popBackStackImmediate(); | ||||
|         } | ||||
|  | ||||
|         // only expand the bottom sheet back if there are no more nested comment replies fragments | ||||
|         // stacked under the one that is currently being popped | ||||
|         if (CommentRepliesFragment.TAG.equals(fragmentUnderEntryName)) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         final BottomSheetBehavior<FragmentContainerView> behavior = BottomSheetBehavior | ||||
|                 .from(mainBinding.fragmentPlayerHolder); | ||||
|         // do not return to the comment if the details fragment was closed | ||||
|         if (behavior.getState() == BottomSheetBehavior.STATE_HIDDEN) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // scroll to the root comment once the bottom sheet expansion animation is finished | ||||
|         behavior.addBottomSheetCallback(new BottomSheetBehavior.BottomSheetCallback() { | ||||
|             @Override | ||||
|             public void onStateChanged(@NonNull final View bottomSheet, | ||||
|                                        final int newState) { | ||||
|                 if (newState == BottomSheetBehavior.STATE_EXPANDED) { | ||||
|                     final Fragment detailFragment = fm.findFragmentById( | ||||
|                             R.id.fragment_player_holder); | ||||
|                     if (detailFragment instanceof VideoDetailFragment && rootComment != null) { | ||||
|                         // should always be the case | ||||
|                         ((VideoDetailFragment) detailFragment).scrollToComment(rootComment); | ||||
|                     } | ||||
|                     behavior.removeBottomSheetCallback(this); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             @Override | ||||
|             public void onSlide(@NonNull final View bottomSheet, final float slideOffset) { | ||||
|                 // not needed, listener is removed once the sheet is expanded | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         behavior.setState(BottomSheetBehavior.STATE_EXPANDED); | ||||
|     } | ||||
|  | ||||
|     private boolean bottomSheetHiddenOrCollapsed() { | ||||
|         final BottomSheetBehavior<FrameLayout> bottomSheetBehavior = | ||||
|                 BottomSheetBehavior.from(mainBinding.fragmentPlayerHolder); | ||||
|   | ||||
| @@ -6,6 +6,9 @@ import static org.schabi.newpipe.database.Migrations.MIGRATION_2_3; | ||||
| import static org.schabi.newpipe.database.Migrations.MIGRATION_3_4; | ||||
| import static org.schabi.newpipe.database.Migrations.MIGRATION_4_5; | ||||
| import static org.schabi.newpipe.database.Migrations.MIGRATION_5_6; | ||||
| import static org.schabi.newpipe.database.Migrations.MIGRATION_6_7; | ||||
| import static org.schabi.newpipe.database.Migrations.MIGRATION_7_8; | ||||
| import static org.schabi.newpipe.database.Migrations.MIGRATION_8_9; | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.database.Cursor; | ||||
| @@ -26,7 +29,7 @@ public final class NewPipeDatabase { | ||||
|         return Room | ||||
|                 .databaseBuilder(context.getApplicationContext(), AppDatabase.class, DATABASE_NAME) | ||||
|                 .addMigrations(MIGRATION_1_2, MIGRATION_2_3, MIGRATION_3_4, MIGRATION_4_5, | ||||
|                         MIGRATION_5_6) | ||||
|                         MIGRATION_5_6, MIGRATION_6_7, MIGRATION_7_8, MIGRATION_8_9) | ||||
|                 .build(); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,26 +1,26 @@ | ||||
| package org.schabi.newpipe | ||||
|  | ||||
| import android.app.PendingIntent | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.util.Log | ||||
| import android.widget.Toast | ||||
| import androidx.core.app.NotificationCompat | ||||
| import androidx.core.app.NotificationManagerCompat | ||||
| import androidx.core.app.PendingIntentCompat | ||||
| import androidx.core.content.ContextCompat | ||||
| import androidx.core.content.edit | ||||
| import androidx.core.net.toUri | ||||
| import androidx.preference.PreferenceManager | ||||
| import androidx.work.OneTimeWorkRequest | ||||
| import androidx.work.OneTimeWorkRequestBuilder | ||||
| import androidx.work.WorkManager | ||||
| import androidx.work.WorkRequest | ||||
| import androidx.work.Worker | ||||
| import androidx.work.WorkerParameters | ||||
| import androidx.work.workDataOf | ||||
| import com.grack.nanojson.JsonParser | ||||
| import com.grack.nanojson.JsonParserException | ||||
| import org.schabi.newpipe.extractor.downloader.Response | ||||
| import org.schabi.newpipe.extractor.exceptions.ReCaptchaException | ||||
| import org.schabi.newpipe.util.ReleaseVersionUtil.coerceUpdateCheckExpiry | ||||
| import org.schabi.newpipe.util.ReleaseVersionUtil.isLastUpdateCheckExpired | ||||
| import org.schabi.newpipe.util.ReleaseVersionUtil.isReleaseApk | ||||
| import org.schabi.newpipe.util.ReleaseVersionUtil | ||||
| import java.io.IOException | ||||
|  | ||||
| class NewVersionWorker( | ||||
| @@ -42,42 +42,58 @@ class NewVersionWorker( | ||||
|         versionCode: Int | ||||
|     ) { | ||||
|         if (BuildConfig.VERSION_CODE >= versionCode) { | ||||
|             if (inputData.getBoolean(IS_MANUAL, false)) { | ||||
|                 // Show toast stating that the app is up-to-date if the update check was manual. | ||||
|                 ContextCompat.getMainExecutor(applicationContext).execute { | ||||
|                     Toast.makeText( | ||||
|                         applicationContext, R.string.app_update_unavailable_toast, | ||||
|                         Toast.LENGTH_SHORT | ||||
|                     ).show() | ||||
|                 } | ||||
|             } | ||||
|             return | ||||
|         } | ||||
|         val app = App.getApp() | ||||
|  | ||||
|         // A pending intent to open the apk location url in the browser. | ||||
|         val intent = Intent(Intent.ACTION_VIEW, apkLocationUrl?.toUri()) | ||||
|         intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) | ||||
|         val pendingIntent = PendingIntent.getActivity(app, 0, intent, 0) | ||||
|         val channelId = app.getString(R.string.app_update_notification_channel_id) | ||||
|         val notificationBuilder = NotificationCompat.Builder(app, channelId) | ||||
|         val pendingIntent = PendingIntentCompat.getActivity( | ||||
|             applicationContext, 0, intent, 0, false | ||||
|         ) | ||||
|         val channelId = applicationContext.getString(R.string.app_update_notification_channel_id) | ||||
|         val notificationBuilder = NotificationCompat.Builder(applicationContext, channelId) | ||||
|             .setSmallIcon(R.drawable.ic_newpipe_update) | ||||
|             .setVisibility(NotificationCompat.VISIBILITY_PUBLIC) | ||||
|             .setContentIntent(pendingIntent) | ||||
|             .setAutoCancel(true) | ||||
|             .setContentTitle(app.getString(R.string.app_update_notification_content_title)) | ||||
|             .setContentText( | ||||
|                 app.getString(R.string.app_update_notification_content_text) + | ||||
|                     " " + versionName | ||||
|             .setContentIntent(pendingIntent) | ||||
|             .setContentTitle( | ||||
|                 applicationContext.getString(R.string.app_update_available_notification_title) | ||||
|             ) | ||||
|         val notificationManager = NotificationManagerCompat.from(app) | ||||
|             .setContentText( | ||||
|                 applicationContext.getString( | ||||
|                     R.string.app_update_available_notification_text, versionName | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|         val notificationManager = NotificationManagerCompat.from(applicationContext) | ||||
|         notificationManager.notify(2000, notificationBuilder.build()) | ||||
|     } | ||||
|  | ||||
|     @Throws(IOException::class, ReCaptchaException::class) | ||||
|     private fun checkNewVersion() { | ||||
|         // Check if the current apk is a github one or not. | ||||
|         if (!isReleaseApk()) { | ||||
|         if (!ReleaseVersionUtil.isReleaseApk) { | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         val prefs = PreferenceManager.getDefaultSharedPreferences(applicationContext) | ||||
|         // Check if the last request has happened a certain time ago | ||||
|         // to reduce the number of API requests. | ||||
|         val expiry = prefs.getLong(applicationContext.getString(R.string.update_expiry_key), 0) | ||||
|         if (!isLastUpdateCheckExpired(expiry)) { | ||||
|             return | ||||
|         if (!inputData.getBoolean(IS_MANUAL, false)) { | ||||
|             val prefs = PreferenceManager.getDefaultSharedPreferences(applicationContext) | ||||
|             // Check if the last request has happened a certain time ago | ||||
|             // to reduce the number of API requests. | ||||
|             val expiry = prefs.getLong(applicationContext.getString(R.string.update_expiry_key), 0) | ||||
|             if (!ReleaseVersionUtil.isLastUpdateCheckExpired(expiry)) { | ||||
|                 return | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Make a network request to get latest NewPipe data. | ||||
| @@ -90,7 +106,7 @@ class NewVersionWorker( | ||||
|         try { | ||||
|             // Store a timestamp which needs to be exceeded, | ||||
|             // before a new request to the API is made. | ||||
|             val newExpiry = coerceUpdateCheckExpiry(response.getHeader("expires")) | ||||
|             val newExpiry = ReleaseVersionUtil.coerceUpdateCheckExpiry(response.getHeader("expires")) | ||||
|             prefs.edit { | ||||
|                 putLong(applicationContext.getString(R.string.update_expiry_key), newExpiry) | ||||
|             } | ||||
| @@ -102,13 +118,13 @@ class NewVersionWorker( | ||||
|  | ||||
|         // Parse the json from the response. | ||||
|         try { | ||||
|             val githubStableObject = JsonParser.`object`() | ||||
|             val newpipeVersionInfo = JsonParser.`object`() | ||||
|                 .from(response.responseBody()).getObject("flavors") | ||||
|                 .getObject("github").getObject("stable") | ||||
|                 .getObject("newpipe") | ||||
|  | ||||
|             val versionName = githubStableObject.getString("version") | ||||
|             val versionCode = githubStableObject.getInt("version_code") | ||||
|             val apkLocationUrl = githubStableObject.getString("apk") | ||||
|             val versionName = newpipeVersionInfo.getString("version") | ||||
|             val versionCode = newpipeVersionInfo.getInt("version_code") | ||||
|             val apkLocationUrl = newpipeVersionInfo.getString("apk") | ||||
|             compareAppVersionAndShowNotification(versionName, apkLocationUrl, versionCode) | ||||
|         } catch (e: JsonParserException) { | ||||
|             // Most likely something is wrong in data received from NEWPIPE_API_URL. | ||||
| @@ -120,43 +136,42 @@ class NewVersionWorker( | ||||
|     } | ||||
|  | ||||
|     override fun doWork(): Result { | ||||
|         try { | ||||
|         return try { | ||||
|             checkNewVersion() | ||||
|             Result.success() | ||||
|         } catch (e: IOException) { | ||||
|             Log.w(TAG, "Could not fetch NewPipe API: probably network problem", e) | ||||
|             return Result.failure() | ||||
|             Result.failure() | ||||
|         } catch (e: ReCaptchaException) { | ||||
|             Log.e(TAG, "ReCaptchaException should never happen here.", e) | ||||
|             return Result.failure() | ||||
|             Result.failure() | ||||
|         } | ||||
|         return Result.success() | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         private val DEBUG = MainActivity.DEBUG | ||||
|         private val TAG = NewVersionWorker::class.java.simpleName | ||||
|         private const val NEWPIPE_API_URL = "https://newpipe.net/api/data.json" | ||||
|         private const val IS_MANUAL = "isManual" | ||||
|  | ||||
|         /** | ||||
|          * Start a new worker which | ||||
|          * checks if all conditions for performing a version check are met, | ||||
|          * fetches the API endpoint [.NEWPIPE_API_URL] containing info | ||||
|          * about the latest NewPipe version | ||||
|          * and displays a notification about ana available update. | ||||
|          * Start a new worker which checks if all conditions for performing a version check are met, | ||||
|          * fetches the API endpoint [.NEWPIPE_API_URL] containing info about the latest NewPipe | ||||
|          * version and displays a notification about an available update if one is available. | ||||
|          * <br></br> | ||||
|          * Following conditions need to be met, before data is request from the server: | ||||
|          * Following conditions need to be met, before data is requested from the server: | ||||
|          * | ||||
|          *  *  The app is signed with the correct signing key (by TeamNewPipe / schabi). | ||||
|          * If the signing key differs from the one used upstream, the update cannot be installed. | ||||
|          *  * The user enabled searching for and notifying about updates in the settings. | ||||
|          *  * The app did not recently check for updates. | ||||
|          * We do not want to make unnecessary connections and DOS our servers. | ||||
|          * | ||||
|          */ | ||||
|         @JvmStatic | ||||
|         fun enqueueNewVersionCheckingWork(context: Context) { | ||||
|             val workRequest: WorkRequest = | ||||
|                 OneTimeWorkRequest.Builder(NewVersionWorker::class.java).build() | ||||
|         fun enqueueNewVersionCheckingWork(context: Context, isManual: Boolean) { | ||||
|             val workRequest = OneTimeWorkRequestBuilder<NewVersionWorker>() | ||||
|                 .setInputData(workDataOf(IS_MANUAL to isManual)) | ||||
|                 .build() | ||||
|             WorkManager.getInstance(context).enqueue(workRequest) | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package org.schabi.newpipe; | ||||
| import android.annotation.SuppressLint; | ||||
| import android.app.Activity; | ||||
| import android.content.Intent; | ||||
| import android.os.Build; | ||||
| import android.os.Bundle; | ||||
|  | ||||
| /* | ||||
| @@ -40,10 +39,6 @@ public class PanicResponderActivity extends Activity { | ||||
|             ExitActivity.exitAndRemoveFromRecentApps(this); | ||||
|         } | ||||
|  | ||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { | ||||
|             finishAndRemoveTask(); | ||||
|         } else { | ||||
|             finish(); | ||||
|         } | ||||
|         finishAndRemoveTask(); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| package org.schabi.newpipe; | ||||
|  | ||||
| import static org.schabi.newpipe.util.SparseItemUtil.fetchStreamInfoAndSaveToDatabase; | ||||
| import static org.schabi.newpipe.util.external_communication.ShareUtils.shareText; | ||||
|  | ||||
| import android.content.Context; | ||||
| @@ -10,13 +11,14 @@ import android.widget.PopupMenu; | ||||
| import androidx.fragment.app.FragmentManager; | ||||
|  | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity; | ||||
| import org.schabi.newpipe.download.DownloadDialog; | ||||
| import org.schabi.newpipe.local.dialog.PlaylistDialog; | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueue; | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueueItem; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.SparseItemUtil; | ||||
|  | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
|  | ||||
| public final class QueueItemMenuUtil { | ||||
|     private QueueItemMenuUtil() { | ||||
| @@ -53,7 +55,7 @@ public final class QueueItemMenuUtil { | ||||
|                 case R.id.menu_item_append_playlist: | ||||
|                     PlaylistDialog.createCorrespondingDialog( | ||||
|                             context, | ||||
|                             Collections.singletonList(new StreamEntity(item)), | ||||
|                             List.of(new StreamEntity(item)), | ||||
|                             dialog -> dialog.show( | ||||
|                                     fragmentManager, | ||||
|                                     "QueueItemMenuUtil@append_playlist" | ||||
| @@ -73,7 +75,15 @@ public final class QueueItemMenuUtil { | ||||
|                     return true; | ||||
|                 case R.id.menu_item_share: | ||||
|                     shareText(context, item.getTitle(), item.getUrl(), | ||||
|                             item.getThumbnailUrl()); | ||||
|                             item.getThumbnails()); | ||||
|                     return true; | ||||
|                 case R.id.menu_item_download: | ||||
|                     fetchStreamInfoAndSaveToDatabase(context, item.getServiceId(), item.getUrl(), | ||||
|                             info -> { | ||||
|                                 final DownloadDialog downloadDialog = new DownloadDialog(context, | ||||
|                                         info); | ||||
|                                 downloadDialog.show(fragmentManager, "downloadDialog"); | ||||
|                             }); | ||||
|                     return true; | ||||
|             } | ||||
|             return false; | ||||
|   | ||||
| @@ -10,12 +10,14 @@ import android.content.DialogInterface; | ||||
| import android.content.Intent; | ||||
| import android.content.SharedPreferences; | ||||
| import android.content.pm.PackageManager; | ||||
| import android.os.Build; | ||||
| import android.os.Bundle; | ||||
| import android.text.TextUtils; | ||||
| import android.view.ContextThemeWrapper; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.view.WindowManager; | ||||
| import android.widget.Button; | ||||
| import android.widget.RadioButton; | ||||
| import android.widget.RadioGroup; | ||||
| @@ -24,19 +26,26 @@ import android.widget.Toast; | ||||
| import androidx.annotation.DrawableRes; | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.annotation.StringRes; | ||||
| import androidx.appcompat.app.AlertDialog; | ||||
| import androidx.appcompat.app.AppCompatActivity; | ||||
| import androidx.appcompat.content.res.AppCompatResources; | ||||
| import androidx.core.app.NotificationCompat; | ||||
| import androidx.core.app.ServiceCompat; | ||||
| import androidx.core.widget.TextViewCompat; | ||||
| import androidx.core.math.MathUtils; | ||||
| import androidx.fragment.app.DialogFragment; | ||||
| import androidx.fragment.app.Fragment; | ||||
| import androidx.fragment.app.FragmentManager; | ||||
| import androidx.lifecycle.DefaultLifecycleObserver; | ||||
| import androidx.lifecycle.Lifecycle; | ||||
| import androidx.lifecycle.LifecycleOwner; | ||||
| import androidx.preference.PreferenceManager; | ||||
|  | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity; | ||||
| import org.schabi.newpipe.databinding.ListRadioIconItemBinding; | ||||
| import org.schabi.newpipe.databinding.SingleChoiceDialogViewBinding; | ||||
| import org.schabi.newpipe.download.DownloadDialog; | ||||
| import org.schabi.newpipe.download.LoadingDialog; | ||||
| import org.schabi.newpipe.error.ErrorInfo; | ||||
| import org.schabi.newpipe.error.ErrorUtil; | ||||
| import org.schabi.newpipe.error.ReCaptchaActivity; | ||||
| @@ -56,22 +65,23 @@ import org.schabi.newpipe.extractor.exceptions.PrivateContentException; | ||||
| import org.schabi.newpipe.extractor.exceptions.ReCaptchaException; | ||||
| import org.schabi.newpipe.extractor.exceptions.SoundCloudGoPlusContentException; | ||||
| import org.schabi.newpipe.extractor.exceptions.YoutubeMusicPremiumContentException; | ||||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandler; | ||||
| import org.schabi.newpipe.extractor.playlist.PlaylistInfo; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfo; | ||||
| import org.schabi.newpipe.extractor.stream.VideoStream; | ||||
| import org.schabi.newpipe.ktx.ExceptionUtils; | ||||
| import org.schabi.newpipe.local.dialog.PlaylistDialog; | ||||
| import org.schabi.newpipe.player.MainPlayer; | ||||
| import org.schabi.newpipe.player.PlayerType; | ||||
| import org.schabi.newpipe.player.helper.PlayerHelper; | ||||
| import org.schabi.newpipe.player.helper.PlayerHolder; | ||||
| import org.schabi.newpipe.player.playqueue.ChannelPlayQueue; | ||||
| import org.schabi.newpipe.player.playqueue.ChannelTabPlayQueue; | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueue; | ||||
| import org.schabi.newpipe.player.playqueue.PlaylistPlayQueue; | ||||
| import org.schabi.newpipe.player.playqueue.SinglePlayQueue; | ||||
| import org.schabi.newpipe.util.ChannelTabHelper; | ||||
| import org.schabi.newpipe.util.Constants; | ||||
| import org.schabi.newpipe.util.DeviceUtils; | ||||
| import org.schabi.newpipe.util.ExtractorHelper; | ||||
| import org.schabi.newpipe.util.ListHelper; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.PermissionHelper; | ||||
| import org.schabi.newpipe.util.ThemeHelper; | ||||
| @@ -80,10 +90,13 @@ import org.schabi.newpipe.util.urlfinder.UrlFinder; | ||||
| import org.schabi.newpipe.views.FocusOverlayView; | ||||
|  | ||||
| import java.io.Serializable; | ||||
| import java.lang.ref.Reference; | ||||
| import java.lang.ref.WeakReference; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
| import java.util.Optional; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| import icepick.Icepick; | ||||
| import icepick.State; | ||||
| @@ -92,7 +105,6 @@ import io.reactivex.rxjava3.core.Observable; | ||||
| import io.reactivex.rxjava3.core.Single; | ||||
| import io.reactivex.rxjava3.disposables.CompositeDisposable; | ||||
| import io.reactivex.rxjava3.disposables.Disposable; | ||||
| import io.reactivex.rxjava3.functions.Consumer; | ||||
| import io.reactivex.rxjava3.schedulers.Schedulers; | ||||
|  | ||||
| /** | ||||
| @@ -112,12 +124,57 @@ public class RouterActivity extends AppCompatActivity { | ||||
|     private boolean selectionIsDownload = false; | ||||
|     private boolean selectionIsAddToPlaylist = false; | ||||
|     private AlertDialog alertDialogChoice = null; | ||||
|     private FragmentManager.FragmentLifecycleCallbacks dismissListener = null; | ||||
|  | ||||
|     @Override | ||||
|     protected void onCreate(final Bundle savedInstanceState) { | ||||
|         ThemeHelper.setDayNightMode(this); | ||||
|         setTheme(ThemeHelper.isLightThemeSelected(this) | ||||
|                 ? R.style.RouterActivityThemeLight : R.style.RouterActivityThemeDark); | ||||
|         Localization.assureCorrectAppLanguage(this); | ||||
|  | ||||
|         // Pass-through touch events to background activities | ||||
|         // so that our transparent window won't lock UI in the mean time | ||||
|         // network request is underway before showing PlaylistDialog or DownloadDialog | ||||
|         // (ref: https://stackoverflow.com/a/10606141) | ||||
|         getWindow().addFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | ||||
|                 | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | ||||
|                 | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE); | ||||
|  | ||||
|         // Android never fails to impress us with a list of new restrictions per API. | ||||
|         // Starting with S (Android 12) one of the prerequisite conditions has to be met | ||||
|         // before the FLAG_NOT_TOUCHABLE flag is allowed to kick in: | ||||
|         // @see WindowManager.LayoutParams#FLAG_NOT_TOUCHABLE | ||||
|         // For our present purpose it seems we can just set LayoutParams.alpha to 0 | ||||
|         // on the strength of "4. Fully transparent windows" without affecting the scrim of dialogs | ||||
|         final WindowManager.LayoutParams params = getWindow().getAttributes(); | ||||
|         params.alpha = 0f; | ||||
|         getWindow().setAttributes(params); | ||||
|  | ||||
|         super.onCreate(savedInstanceState); | ||||
|         Icepick.restoreInstanceState(this, savedInstanceState); | ||||
|  | ||||
|         // FragmentManager will take care to recreate (Playlist|Download)Dialog when screen rotates | ||||
|         // We used to .setOnDismissListener(dialog -> finish()); when creating these DialogFragments | ||||
|         // but those callbacks won't survive a config change | ||||
|         // Try an alternate approach to hook into FragmentManager instead, to that effect | ||||
|         // (ref: https://stackoverflow.com/a/44028453) | ||||
|         final FragmentManager fm = getSupportFragmentManager(); | ||||
|         if (dismissListener == null) { | ||||
|             dismissListener = new FragmentManager.FragmentLifecycleCallbacks() { | ||||
|                 @Override | ||||
|                 public void onFragmentDestroyed(@NonNull final FragmentManager fm, | ||||
|                                                 @NonNull final Fragment f) { | ||||
|                     super.onFragmentDestroyed(fm, f); | ||||
|                     if (f instanceof DialogFragment && fm.getFragments().isEmpty()) { | ||||
|                         // No more DialogFragments, we're done | ||||
|                         finish(); | ||||
|                     } | ||||
|                 } | ||||
|             }; | ||||
|         } | ||||
|         fm.registerFragmentLifecycleCallbacks(dismissListener, false); | ||||
|  | ||||
|         if (TextUtils.isEmpty(currentUrl)) { | ||||
|             currentUrl = getUrl(getIntent()); | ||||
|  | ||||
| @@ -126,9 +183,6 @@ public class RouterActivity extends AppCompatActivity { | ||||
|                 finish(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         setTheme(ThemeHelper.isLightThemeSelected(this) | ||||
|                 ? R.style.RouterActivityThemeLight : R.style.RouterActivityThemeDark); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -150,16 +204,34 @@ public class RouterActivity extends AppCompatActivity { | ||||
|     protected void onStart() { | ||||
|         super.onStart(); | ||||
|  | ||||
|         handleUrl(currentUrl); | ||||
|         // Don't overlap the DialogFragment after rotating the screen | ||||
|         // If there's no DialogFragment, we're either starting afresh | ||||
|         // or we didn't make it to PlaylistDialog or DownloadDialog before the orientation change | ||||
|         if (getSupportFragmentManager().getFragments().isEmpty()) { | ||||
|             // Start over from scratch | ||||
|             handleUrl(currentUrl); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void onDestroy() { | ||||
|         super.onDestroy(); | ||||
|  | ||||
|         if (dismissListener != null) { | ||||
|             getSupportFragmentManager().unregisterFragmentLifecycleCallbacks(dismissListener); | ||||
|         } | ||||
|  | ||||
|         disposables.clear(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void finish() { | ||||
|         // allow the activity to recreate in case orientation changes | ||||
|         if (!isChangingConfigurations()) { | ||||
|             super.finish(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void handleUrl(final String url) { | ||||
|         disposables.add(Observable | ||||
|                 .fromCallable(() -> { | ||||
| @@ -239,7 +311,7 @@ public class RouterActivity extends AppCompatActivity { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void showUnsupportedUrlDialog(final String url) { | ||||
|     protected void showUnsupportedUrlDialog(final String url) { | ||||
|         final Context context = getThemeWrapperContext(); | ||||
|         new AlertDialog.Builder(context) | ||||
|                 .setTitle(R.string.unsupported_url) | ||||
| @@ -257,80 +329,122 @@ public class RouterActivity extends AppCompatActivity { | ||||
|     protected void onSuccess() { | ||||
|         final SharedPreferences preferences = PreferenceManager | ||||
|                 .getDefaultSharedPreferences(this); | ||||
|         final String selectedChoiceKey = preferences | ||||
|                 .getString(getString(R.string.preferred_open_action_key), | ||||
|                         getString(R.string.preferred_open_action_default)); | ||||
|  | ||||
|         final String showInfoKey = getString(R.string.show_info_key); | ||||
|         final String videoPlayerKey = getString(R.string.video_player_key); | ||||
|         final String backgroundPlayerKey = getString(R.string.background_player_key); | ||||
|         final String popupPlayerKey = getString(R.string.popup_player_key); | ||||
|         final String downloadKey = getString(R.string.download_key); | ||||
|         final String alwaysAskKey = getString(R.string.always_ask_open_action_key); | ||||
|         final ChoiceAvailabilityChecker choiceChecker = new ChoiceAvailabilityChecker( | ||||
|                 getChoicesForService(currentService, currentLinkType), | ||||
|                 preferences.getString(getString(R.string.preferred_open_action_key), | ||||
|                         getString(R.string.preferred_open_action_default))); | ||||
|  | ||||
|         if (selectedChoiceKey.equals(alwaysAskKey)) { | ||||
|             final List<AdapterChoiceItem> choices | ||||
|                     = getChoicesForService(currentService, currentLinkType); | ||||
|         // Check for non-player related choices | ||||
|         if (choiceChecker.isAvailableAndSelected( | ||||
|                 R.string.show_info_key, | ||||
|                 R.string.download_key, | ||||
|                 R.string.add_to_playlist_key)) { | ||||
|             handleChoice(choiceChecker.getSelectedChoiceKey()); | ||||
|             return; | ||||
|         } | ||||
|         // Check if the choice is player related | ||||
|         if (choiceChecker.isAvailableAndSelected( | ||||
|                 R.string.video_player_key, | ||||
|                 R.string.background_player_key, | ||||
|                 R.string.popup_player_key)) { | ||||
|  | ||||
|             final String selectedChoice = choiceChecker.getSelectedChoiceKey(); | ||||
|  | ||||
|             switch (choices.size()) { | ||||
|                 case 1: | ||||
|                     handleChoice(choices.get(0).key); | ||||
|                     break; | ||||
|                 case 0: | ||||
|                     handleChoice(showInfoKey); | ||||
|                     break; | ||||
|                 default: | ||||
|                     showDialog(choices); | ||||
|                     break; | ||||
|             } | ||||
|         } else if (selectedChoiceKey.equals(showInfoKey)) { | ||||
|             handleChoice(showInfoKey); | ||||
|         } else if (selectedChoiceKey.equals(downloadKey)) { | ||||
|             handleChoice(downloadKey); | ||||
|         } else { | ||||
|             final boolean isExtVideoEnabled = preferences.getBoolean( | ||||
|                     getString(R.string.use_external_video_player_key), false); | ||||
|             final boolean isExtAudioEnabled = preferences.getBoolean( | ||||
|                     getString(R.string.use_external_audio_player_key), false); | ||||
|             final boolean isVideoPlayerSelected = selectedChoiceKey.equals(videoPlayerKey) | ||||
|                     || selectedChoiceKey.equals(popupPlayerKey); | ||||
|             final boolean isAudioPlayerSelected = selectedChoiceKey.equals(backgroundPlayerKey); | ||||
|             final boolean isVideoPlayerSelected = | ||||
|                     selectedChoice.equals(getString(R.string.video_player_key)) | ||||
|                             || selectedChoice.equals(getString(R.string.popup_player_key)); | ||||
|             final boolean isAudioPlayerSelected = | ||||
|                     selectedChoice.equals(getString(R.string.background_player_key)); | ||||
|  | ||||
|             if (currentLinkType != LinkType.STREAM) { | ||||
|                 if (isExtAudioEnabled && isAudioPlayerSelected | ||||
|                         || isExtVideoEnabled && isVideoPlayerSelected) { | ||||
|                     Toast.makeText(this, R.string.external_player_unsupported_link_type, | ||||
|                             Toast.LENGTH_LONG).show(); | ||||
|                     handleChoice(showInfoKey); | ||||
|                     return; | ||||
|                 } | ||||
|             if (currentLinkType != LinkType.STREAM | ||||
|                     && ((isExtAudioEnabled && isAudioPlayerSelected) | ||||
|                     || (isExtVideoEnabled && isVideoPlayerSelected)) | ||||
|             ) { | ||||
|                 Toast.makeText(this, R.string.external_player_unsupported_link_type, | ||||
|                         Toast.LENGTH_LONG).show(); | ||||
|                 handleChoice(getString(R.string.show_info_key)); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             final List<StreamingService.ServiceInfo.MediaCapability> capabilities | ||||
|                     = currentService.getServiceInfo().getMediaCapabilities(); | ||||
|             final List<StreamingService.ServiceInfo.MediaCapability> capabilities = | ||||
|                     currentService.getServiceInfo().getMediaCapabilities(); | ||||
|  | ||||
|             boolean serviceSupportsChoice = false; | ||||
|             if (isVideoPlayerSelected) { | ||||
|                 serviceSupportsChoice = capabilities.contains(VIDEO); | ||||
|             } else if (selectedChoiceKey.equals(backgroundPlayerKey)) { | ||||
|                 serviceSupportsChoice = capabilities.contains(AUDIO); | ||||
|             } | ||||
|  | ||||
|             if (serviceSupportsChoice) { | ||||
|                 handleChoice(selectedChoiceKey); | ||||
|             // Check if the service supports the choice | ||||
|             if ((isVideoPlayerSelected && capabilities.contains(VIDEO)) | ||||
|                     || (isAudioPlayerSelected && capabilities.contains(AUDIO))) { | ||||
|                 handleChoice(selectedChoice); | ||||
|             } else { | ||||
|                 handleChoice(showInfoKey); | ||||
|                 handleChoice(getString(R.string.show_info_key)); | ||||
|             } | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Default / Ask always | ||||
|         final List<AdapterChoiceItem> availableChoices = choiceChecker.getAvailableChoices(); | ||||
|         switch (availableChoices.size()) { | ||||
|             case 1: | ||||
|                 handleChoice(availableChoices.get(0).key); | ||||
|                 break; | ||||
|             case 0: | ||||
|                 handleChoice(getString(R.string.show_info_key)); | ||||
|                 break; | ||||
|             default: | ||||
|                 showDialog(availableChoices); | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * This is a helper class for checking if the choices are available and/or selected. | ||||
|      */ | ||||
|     class ChoiceAvailabilityChecker { | ||||
|         private final List<AdapterChoiceItem> availableChoices; | ||||
|         private final String selectedChoiceKey; | ||||
|  | ||||
|         ChoiceAvailabilityChecker( | ||||
|                 @NonNull final List<AdapterChoiceItem> availableChoices, | ||||
|                 @NonNull final String selectedChoiceKey) { | ||||
|             this.availableChoices = availableChoices; | ||||
|             this.selectedChoiceKey = selectedChoiceKey; | ||||
|         } | ||||
|  | ||||
|         public List<AdapterChoiceItem> getAvailableChoices() { | ||||
|             return availableChoices; | ||||
|         } | ||||
|  | ||||
|         public String getSelectedChoiceKey() { | ||||
|             return selectedChoiceKey; | ||||
|         } | ||||
|  | ||||
|         public boolean isAvailableAndSelected(@StringRes final int... wantedKeys) { | ||||
|             return Arrays.stream(wantedKeys).anyMatch(this::isAvailableAndSelected); | ||||
|         } | ||||
|  | ||||
|         public boolean isAvailableAndSelected(@StringRes final int wantedKey) { | ||||
|             final String wanted = getString(wantedKey); | ||||
|             // Check if the wanted option is selected | ||||
|             if (!selectedChoiceKey.equals(wanted)) { | ||||
|                 return false; | ||||
|             } | ||||
|             // Check if it's available | ||||
|             return availableChoices.stream().anyMatch(item -> wanted.equals(item.key)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void showDialog(final List<AdapterChoiceItem> choices) { | ||||
|         final SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(this); | ||||
|         final Context themeWrapperContext = getThemeWrapperContext(); | ||||
|  | ||||
|         final LayoutInflater inflater = LayoutInflater.from(themeWrapperContext); | ||||
|         final RadioGroup radioGroup = SingleChoiceDialogViewBinding.inflate(getLayoutInflater()) | ||||
|                 .list; | ||||
|         final Context themeWrapperContext = getThemeWrapperContext(); | ||||
|         final LayoutInflater layoutInflater = LayoutInflater.from(themeWrapperContext); | ||||
|  | ||||
|         final SingleChoiceDialogViewBinding binding = | ||||
|                 SingleChoiceDialogViewBinding.inflate(layoutInflater); | ||||
|         final RadioGroup radioGroup = binding.list; | ||||
|  | ||||
|         final DialogInterface.OnClickListener dialogButtonsClickListener = (dialog, which) -> { | ||||
|             final int indexOfChild = radioGroup.indexOfChild( | ||||
| @@ -349,21 +463,19 @@ public class RouterActivity extends AppCompatActivity { | ||||
|  | ||||
|         alertDialogChoice = new AlertDialog.Builder(themeWrapperContext) | ||||
|                 .setTitle(R.string.preferred_open_action_share_menu_title) | ||||
|                 .setView(radioGroup) | ||||
|                 .setView(binding.getRoot()) | ||||
|                 .setCancelable(true) | ||||
|                 .setNegativeButton(R.string.just_once, dialogButtonsClickListener) | ||||
|                 .setPositiveButton(R.string.always, dialogButtonsClickListener) | ||||
|                 .setOnDismissListener((dialog) -> { | ||||
|                 .setOnDismissListener(dialog -> { | ||||
|                     if (!selectionIsDownload && !selectionIsAddToPlaylist) { | ||||
|                         finish(); | ||||
|                     } | ||||
|                 }) | ||||
|                 .create(); | ||||
|  | ||||
|         //noinspection CodeBlock2Expr | ||||
|         alertDialogChoice.setOnShowListener(dialog -> { | ||||
|             setDialogButtonsState(alertDialogChoice, radioGroup.getCheckedRadioButtonId() != -1); | ||||
|         }); | ||||
|         alertDialogChoice.setOnShowListener(dialog -> setDialogButtonsState( | ||||
|                 alertDialogChoice, radioGroup.getCheckedRadioButtonId() != -1)); | ||||
|  | ||||
|         radioGroup.setOnCheckedChangeListener((group, checkedId) -> | ||||
|                 setDialogButtonsState(alertDialogChoice, true)); | ||||
| @@ -383,9 +495,10 @@ public class RouterActivity extends AppCompatActivity { | ||||
|  | ||||
|         int id = 12345; | ||||
|         for (final AdapterChoiceItem item : choices) { | ||||
|             final RadioButton radioButton = ListRadioIconItemBinding.inflate(inflater).getRoot(); | ||||
|             final RadioButton radioButton = ListRadioIconItemBinding.inflate(layoutInflater) | ||||
|                     .getRoot(); | ||||
|             radioButton.setText(item.description); | ||||
|             TextViewCompat.setCompoundDrawablesRelativeWithIntrinsicBounds(radioButton, | ||||
|             radioButton.setCompoundDrawablesRelativeWithIntrinsicBounds( | ||||
|                     AppCompatResources.getDrawable(themeWrapperContext, item.icon), | ||||
|                     null, null, null); | ||||
|             radioButton.setChecked(false); | ||||
| @@ -410,7 +523,7 @@ public class RouterActivity extends AppCompatActivity { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         selectedRadioPosition = Math.min(Math.max(-1, selectedRadioPosition), choices.size() - 1); | ||||
|         selectedRadioPosition = MathUtils.clamp(selectedRadioPosition, -1, choices.size() - 1); | ||||
|         if (selectedRadioPosition != -1) { | ||||
|             ((RadioButton) radioGroup.getChildAt(selectedRadioPosition)).setChecked(true); | ||||
|         } | ||||
| @@ -425,90 +538,67 @@ public class RouterActivity extends AppCompatActivity { | ||||
|  | ||||
|     private List<AdapterChoiceItem> getChoicesForService(final StreamingService service, | ||||
|                                                          final LinkType linkType) { | ||||
|         final Context context = getThemeWrapperContext(); | ||||
|  | ||||
|         final List<AdapterChoiceItem> returnList = new ArrayList<>(); | ||||
|         final List<StreamingService.ServiceInfo.MediaCapability> capabilities | ||||
|                 = service.getServiceInfo().getMediaCapabilities(); | ||||
|  | ||||
|         final SharedPreferences preferences = PreferenceManager | ||||
|                 .getDefaultSharedPreferences(this); | ||||
|         final boolean isExtVideoEnabled = preferences.getBoolean( | ||||
|                 getString(R.string.use_external_video_player_key), false); | ||||
|         final boolean isExtAudioEnabled = preferences.getBoolean( | ||||
|                 getString(R.string.use_external_audio_player_key), false); | ||||
|  | ||||
|         final AdapterChoiceItem videoPlayer = new AdapterChoiceItem( | ||||
|                 getString(R.string.video_player_key), getString(R.string.video_player), | ||||
|                 R.drawable.ic_play_arrow); | ||||
|         final AdapterChoiceItem showInfo = new AdapterChoiceItem( | ||||
|                 getString(R.string.show_info_key), getString(R.string.show_info), | ||||
|                 R.drawable.ic_info_outline); | ||||
|         final AdapterChoiceItem popupPlayer = new AdapterChoiceItem( | ||||
|                 getString(R.string.popup_player_key), getString(R.string.popup_player), | ||||
|                 R.drawable.ic_picture_in_picture); | ||||
|         final AdapterChoiceItem videoPlayer = new AdapterChoiceItem( | ||||
|                 getString(R.string.video_player_key), getString(R.string.video_player), | ||||
|                 R.drawable.ic_play_arrow); | ||||
|         final AdapterChoiceItem backgroundPlayer = new AdapterChoiceItem( | ||||
|                 getString(R.string.background_player_key), getString(R.string.background_player), | ||||
|                 R.drawable.ic_headset); | ||||
|         final AdapterChoiceItem addToPlaylist = new AdapterChoiceItem( | ||||
|                 getString(R.string.add_to_playlist_key), getString(R.string.add_to_playlist), | ||||
|                 R.drawable.ic_add); | ||||
|         final AdapterChoiceItem popupPlayer = new AdapterChoiceItem( | ||||
|                 getString(R.string.popup_player_key), getString(R.string.popup_player), | ||||
|                 R.drawable.ic_picture_in_picture); | ||||
|  | ||||
|         final List<AdapterChoiceItem> returnedItems = new ArrayList<>(); | ||||
|         returnedItems.add(showInfo); // Always present | ||||
|  | ||||
|         final List<StreamingService.ServiceInfo.MediaCapability> capabilities = | ||||
|                 service.getServiceInfo().getMediaCapabilities(); | ||||
|  | ||||
|         if (linkType == LinkType.STREAM) { | ||||
|             if (isExtVideoEnabled) { | ||||
|                 // show both "show info" and "video player", they are two different activities | ||||
|                 returnList.add(showInfo); | ||||
|                 returnList.add(videoPlayer); | ||||
|             } else { | ||||
|                 final MainPlayer.PlayerType playerType = PlayerHolder.getInstance().getType(); | ||||
|                 if (capabilities.contains(VIDEO) | ||||
|                         && PlayerHelper.isAutoplayAllowedByUser(context) | ||||
|                         && playerType == null || playerType == MainPlayer.PlayerType.VIDEO) { | ||||
|                     // show only "video player" since the details activity will be opened and the | ||||
|                     // video will be auto played there. Since "show info" would do the exact same | ||||
|                     // thing, use that as a key to let VideoDetailFragment load the stream instead | ||||
|                     // of using FetcherService (see comment in handleChoice()) | ||||
|                     returnList.add(new AdapterChoiceItem( | ||||
|                             showInfo.key, videoPlayer.description, videoPlayer.icon)); | ||||
|                 } else { | ||||
|                     // show only "show info" if video player is not applicable, auto play is | ||||
|                     // disabled or a video is playing in a player different than the main one | ||||
|                     returnList.add(showInfo); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (capabilities.contains(VIDEO)) { | ||||
|                 returnList.add(popupPlayer); | ||||
|                 returnedItems.add(videoPlayer); | ||||
|                 returnedItems.add(popupPlayer); | ||||
|             } | ||||
|             if (capabilities.contains(AUDIO)) { | ||||
|                 returnList.add(backgroundPlayer); | ||||
|                 returnedItems.add(backgroundPlayer); | ||||
|             } | ||||
|             // download is redundant for linkType CHANNEL AND PLAYLIST (till playlist downloading is | ||||
|             // not supported ) | ||||
|             returnList.add(new AdapterChoiceItem(getString(R.string.download_key), | ||||
|             returnedItems.add(new AdapterChoiceItem(getString(R.string.download_key), | ||||
|                     getString(R.string.download), | ||||
|                     R.drawable.ic_file_download)); | ||||
|  | ||||
|             // Add to playlist is not necessary for CHANNEL and PLAYLIST linkType since those can | ||||
|             // not be added to a playlist | ||||
|             returnList.add(addToPlaylist); | ||||
|  | ||||
|             returnedItems.add(new AdapterChoiceItem(getString(R.string.add_to_playlist_key), | ||||
|                     getString(R.string.add_to_playlist), | ||||
|                     R.drawable.ic_add)); | ||||
|         } else { | ||||
|             returnList.add(showInfo); | ||||
|             // LinkType.NONE is never present because it's filtered out before | ||||
|             // channels and playlist can be played as they contain a list of videos | ||||
|             final SharedPreferences preferences = PreferenceManager | ||||
|                     .getDefaultSharedPreferences(this); | ||||
|             final boolean isExtVideoEnabled = preferences.getBoolean( | ||||
|                     getString(R.string.use_external_video_player_key), false); | ||||
|             final boolean isExtAudioEnabled = preferences.getBoolean( | ||||
|                     getString(R.string.use_external_audio_player_key), false); | ||||
|  | ||||
|             if (capabilities.contains(VIDEO) && !isExtVideoEnabled) { | ||||
|                 returnList.add(videoPlayer); | ||||
|                 returnList.add(popupPlayer); | ||||
|                 returnedItems.add(videoPlayer); | ||||
|                 returnedItems.add(popupPlayer); | ||||
|             } | ||||
|             if (capabilities.contains(AUDIO) && !isExtAudioEnabled) { | ||||
|                 returnList.add(backgroundPlayer); | ||||
|                 returnedItems.add(backgroundPlayer); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return returnList; | ||||
|         return returnedItems; | ||||
|     } | ||||
|  | ||||
|     private Context getThemeWrapperContext() { | ||||
|     protected Context getThemeWrapperContext() { | ||||
|         return new ContextThemeWrapper(this, ThemeHelper.isLightThemeSelected(this) | ||||
|                 ? R.style.LightTheme : R.style.DarkTheme); | ||||
|     } | ||||
| @@ -544,8 +634,7 @@ public class RouterActivity extends AppCompatActivity { | ||||
|         } | ||||
|  | ||||
|         if (selectedChoiceKey.equals(getString(R.string.popup_player_key)) | ||||
|                 && !PermissionHelper.isPopupEnabled(this)) { | ||||
|             PermissionHelper.showPopupEnablementToast(this); | ||||
|                 && !PermissionHelper.isPopupEnabledElseAsk(this)) { | ||||
|             finish(); | ||||
|             return; | ||||
|         } | ||||
| @@ -567,7 +656,8 @@ public class RouterActivity extends AppCompatActivity { | ||||
|  | ||||
|         // stop and bypass FetcherService if InfoScreen was selected since | ||||
|         // StreamDetailFragment can fetch data itself | ||||
|         if (selectedChoiceKey.equals(getString(R.string.show_info_key))) { | ||||
|         if (selectedChoiceKey.equals(getString(R.string.show_info_key)) | ||||
|                 || canHandleChoiceLikeShowInfo(selectedChoiceKey)) { | ||||
|             disposables.add(Observable | ||||
|                     .fromCallable(() -> NavigationHelper.getIntentByLink(this, currentUrl)) | ||||
|                     .subscribeOn(Schedulers.io()) | ||||
| @@ -590,63 +680,208 @@ public class RouterActivity extends AppCompatActivity { | ||||
|         finish(); | ||||
|     } | ||||
|  | ||||
|     private void openAddToPlaylistDialog() { | ||||
|         // Getting the stream info usually takes a moment | ||||
|         // Notifying the user here to ensure that no confusion arises | ||||
|         Toast.makeText( | ||||
|                 getApplicationContext(), | ||||
|                 getString(R.string.processing_may_take_a_moment), | ||||
|                 Toast.LENGTH_SHORT) | ||||
|                 .show(); | ||||
|     private boolean canHandleChoiceLikeShowInfo(final String selectedChoiceKey) { | ||||
|         if (!selectedChoiceKey.equals(getString(R.string.video_player_key))) { | ||||
|             return false; | ||||
|         } | ||||
|         // "video player" can be handled like "show info" (because VideoDetailFragment can load | ||||
|         // the stream instead of FetcherService) when... | ||||
|  | ||||
|         disposables.add(ExtractorHelper.getStreamInfo(currentServiceId, currentUrl, false) | ||||
|                 .subscribeOn(Schedulers.io()) | ||||
|                 .observeOn(AndroidSchedulers.mainThread()) | ||||
|                 .subscribe( | ||||
|                         info -> PlaylistDialog.createCorrespondingDialog( | ||||
|                                 getThemeWrapperContext(), | ||||
|                                 Collections.singletonList(new StreamEntity(info)), | ||||
|                                 playlistDialog -> { | ||||
|                                     playlistDialog.setOnDismissListener(dialog -> finish()); | ||||
|         // ...Autoplay is enabled | ||||
|         if (!PlayerHelper.isAutoplayAllowedByUser(getThemeWrapperContext())) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|                                     playlistDialog.show( | ||||
|                                             this.getSupportFragmentManager(), | ||||
|                                             "addToPlaylistDialog" | ||||
|                                     ); | ||||
|                                 } | ||||
|                         ), | ||||
|                         throwable -> handleError(this, new ErrorInfo( | ||||
|                                 throwable, | ||||
|                                 UserAction.REQUESTED_STREAM, | ||||
|                                 "Tried to add " + currentUrl + " to a playlist", | ||||
|                                 currentService.getServiceId()) | ||||
|                         ) | ||||
|                 ) | ||||
|         ); | ||||
|         final boolean isExtVideoEnabled = PreferenceManager.getDefaultSharedPreferences(this) | ||||
|                 .getBoolean(getString(R.string.use_external_video_player_key), false); | ||||
|         // ...it's not done via an external player | ||||
|         if (isExtVideoEnabled) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         // ...the player is not running or in normal Video-mode/type | ||||
|         final PlayerType playerType = PlayerHolder.getInstance().getType(); | ||||
|         return playerType == null || playerType == PlayerType.MAIN; | ||||
|     } | ||||
|  | ||||
|     @SuppressLint("CheckResult") | ||||
|     private void openDownloadDialog() { | ||||
|         disposables.add(ExtractorHelper.getStreamInfo(currentServiceId, currentUrl, true) | ||||
|                 .subscribeOn(Schedulers.io()) | ||||
|                 .observeOn(AndroidSchedulers.mainThread()) | ||||
|                 .subscribe(result -> { | ||||
|                     final List<VideoStream> sortedVideoStreams = ListHelper | ||||
|                             .getSortedStreamVideosList(this, result.getVideoStreams(), | ||||
|                                     result.getVideoOnlyStreams(), false, false); | ||||
|                     final int selectedVideoStreamIndex = ListHelper | ||||
|                             .getDefaultResolutionIndex(this, sortedVideoStreams); | ||||
|     public static class PersistentFragment extends Fragment { | ||||
|         private WeakReference<AppCompatActivity> weakContext; | ||||
|         private final CompositeDisposable disposables = new CompositeDisposable(); | ||||
|         private int running = 0; | ||||
|  | ||||
|                     final FragmentManager fm = getSupportFragmentManager(); | ||||
|                     final DownloadDialog downloadDialog = DownloadDialog.newInstance(result); | ||||
|                     downloadDialog.setVideoStreams(sortedVideoStreams); | ||||
|                     downloadDialog.setAudioStreams(result.getAudioStreams()); | ||||
|                     downloadDialog.setSelectedVideoStream(selectedVideoStreamIndex); | ||||
|                     downloadDialog.setOnDismissListener(dialog -> finish()); | ||||
|                     downloadDialog.show(fm, "downloadDialog"); | ||||
|                     fm.executePendingTransactions(); | ||||
|                 }, throwable -> | ||||
|                         showUnsupportedUrlDialog(currentUrl))); | ||||
|         private synchronized void inFlight(final boolean started) { | ||||
|             if (started) { | ||||
|                 running++; | ||||
|             } else { | ||||
|                 running--; | ||||
|                 if (running <= 0) { | ||||
|                     getActivityContext().ifPresent(context -> context.getSupportFragmentManager() | ||||
|                             .beginTransaction().remove(this).commit()); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void onAttach(@NonNull final Context activityContext) { | ||||
|             super.onAttach(activityContext); | ||||
|             weakContext = new WeakReference<>((AppCompatActivity) activityContext); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void onDetach() { | ||||
|             super.onDetach(); | ||||
|             weakContext = null; | ||||
|         } | ||||
|  | ||||
|         @SuppressWarnings("deprecation") | ||||
|         @Override | ||||
|         public void onCreate(final Bundle savedInstanceState) { | ||||
|             super.onCreate(savedInstanceState); | ||||
|             setRetainInstance(true); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public void onDestroy() { | ||||
|             super.onDestroy(); | ||||
|             disposables.clear(); | ||||
|         } | ||||
|  | ||||
|         /** | ||||
|          * @return the activity context, if there is one and the activity is not finishing | ||||
|          */ | ||||
|         private Optional<AppCompatActivity> getActivityContext() { | ||||
|             return Optional.ofNullable(weakContext) | ||||
|                     .map(Reference::get) | ||||
|                     .filter(context -> !context.isFinishing()); | ||||
|         } | ||||
|  | ||||
|         // guard against IllegalStateException in calling DialogFragment.show() whilst in background | ||||
|         // (which could happen, say, when the user pressed the home button while waiting for | ||||
|         // the network request to return) when it internally calls FragmentTransaction.commit() | ||||
|         // after the FragmentManager has saved its states (isStateSaved() == true) | ||||
|         // (ref: https://stackoverflow.com/a/39813506) | ||||
|         private void runOnVisible(final Consumer<AppCompatActivity> runnable) { | ||||
|             getActivityContext().ifPresentOrElse(context -> { | ||||
|                 if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.STARTED)) { | ||||
|                     context.runOnUiThread(() -> { | ||||
|                         runnable.accept(context); | ||||
|                         inFlight(false); | ||||
|                     }); | ||||
|                 } else { | ||||
|                     getLifecycle().addObserver(new DefaultLifecycleObserver() { | ||||
|                         @Override | ||||
|                         public void onResume(@NonNull final LifecycleOwner owner) { | ||||
|                             getLifecycle().removeObserver(this); | ||||
|                             getActivityContext().ifPresentOrElse(context -> | ||||
|                                     context.runOnUiThread(() -> { | ||||
|                                         runnable.accept(context); | ||||
|                                         inFlight(false); | ||||
|                                     }), | ||||
|                                     () -> inFlight(false) | ||||
|                             ); | ||||
|                         } | ||||
|                     }); | ||||
|                     // this trick doesn't seem to work on Android 10+ (API 29) | ||||
|                     // which places restrictions on starting activities from the background | ||||
|                     if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q | ||||
|                             && !context.isChangingConfigurations()) { | ||||
|                         // try to bring the activity back to front if minimised | ||||
|                         final Intent i = new Intent(context, RouterActivity.class); | ||||
|                         i.setFlags(Intent.FLAG_ACTIVITY_REORDER_TO_FRONT); | ||||
|                         startActivity(i); | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|             }, () -> | ||||
|                 // this branch is executed if there is no activity context | ||||
|                 inFlight(false) | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         <T> Single<T> pleaseWait(final Single<T> single) { | ||||
|             // 'abuse' ambWith() here to cancel the toast for us when the wait is over | ||||
|             return single.ambWith(Single.create(emitter -> getActivityContext().ifPresent(context -> | ||||
|                     context.runOnUiThread(() -> { | ||||
|                         // Getting the stream info usually takes a moment | ||||
|                         // Notifying the user here to ensure that no confusion arises | ||||
|                         final Toast toast = Toast.makeText(context, | ||||
|                                 getString(R.string.processing_may_take_a_moment), | ||||
|                                 Toast.LENGTH_LONG); | ||||
|                         toast.show(); | ||||
|                         emitter.setCancellable(toast::cancel); | ||||
|             })))); | ||||
|         } | ||||
|  | ||||
|         @SuppressLint("CheckResult") | ||||
|         private void openDownloadDialog(final int currentServiceId, final String currentUrl) { | ||||
|             inFlight(true); | ||||
|             final LoadingDialog loadingDialog = new LoadingDialog(R.string.loading_metadata_title); | ||||
|             loadingDialog.show(getParentFragmentManager(), "loadingDialog"); | ||||
|             disposables.add(ExtractorHelper.getStreamInfo(currentServiceId, currentUrl, true) | ||||
|                     .subscribeOn(Schedulers.io()) | ||||
|                     .observeOn(AndroidSchedulers.mainThread()) | ||||
|                     .compose(this::pleaseWait) | ||||
|                     .subscribe(result -> | ||||
|                         runOnVisible(ctx -> { | ||||
|                             loadingDialog.dismiss(); | ||||
|                             final FragmentManager fm = ctx.getSupportFragmentManager(); | ||||
|                             final DownloadDialog downloadDialog = new DownloadDialog(ctx, result); | ||||
|                             // dismiss listener to be handled by FragmentManager | ||||
|                             downloadDialog.show(fm, "downloadDialog"); | ||||
|                         } | ||||
|                         ), throwable -> runOnVisible(ctx -> { | ||||
|                         loadingDialog.dismiss(); | ||||
|                         ((RouterActivity) ctx).showUnsupportedUrlDialog(currentUrl); | ||||
|                     }))); | ||||
|         } | ||||
|  | ||||
|         private void openAddToPlaylistDialog(final int currentServiceId, final String currentUrl) { | ||||
|             inFlight(true); | ||||
|             disposables.add(ExtractorHelper.getStreamInfo(currentServiceId, currentUrl, false) | ||||
|                     .subscribeOn(Schedulers.io()) | ||||
|                     .observeOn(AndroidSchedulers.mainThread()) | ||||
|                     .compose(this::pleaseWait) | ||||
|                     .subscribe( | ||||
|                             info -> getActivityContext().ifPresent(context -> | ||||
|                                     PlaylistDialog.createCorrespondingDialog(context, | ||||
|                                             List.of(new StreamEntity(info)), | ||||
|                                             playlistDialog -> runOnVisible(ctx -> { | ||||
|                                                 // dismiss listener to be handled by FragmentManager | ||||
|                                                 final FragmentManager fm = | ||||
|                                                         ctx.getSupportFragmentManager(); | ||||
|                                                 playlistDialog.show(fm, "addToPlaylistDialog"); | ||||
|                                             }) | ||||
|                                     )), | ||||
|                             throwable -> runOnVisible(ctx -> handleError(ctx, new ErrorInfo( | ||||
|                                     throwable, | ||||
|                                     UserAction.REQUESTED_STREAM, | ||||
|                                     "Tried to add " + currentUrl + " to a playlist", | ||||
|                                     ((RouterActivity) ctx).currentService.getServiceId()) | ||||
|                             )) | ||||
|                     ) | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void openAddToPlaylistDialog() { | ||||
|         getPersistFragment().openAddToPlaylistDialog(currentServiceId, currentUrl); | ||||
|     } | ||||
|  | ||||
|     private void openDownloadDialog() { | ||||
|         getPersistFragment().openDownloadDialog(currentServiceId, currentUrl); | ||||
|     } | ||||
|  | ||||
|     private PersistentFragment getPersistFragment() { | ||||
|         final FragmentManager fm = getSupportFragmentManager(); | ||||
|         PersistentFragment persistFragment = | ||||
|                 (PersistentFragment) fm.findFragmentByTag("PERSIST_FRAGMENT"); | ||||
|         if (persistFragment == null) { | ||||
|             persistFragment = new PersistentFragment(); | ||||
|             fm.beginTransaction() | ||||
|                     .add(persistFragment, "PERSIST_FRAGMENT") | ||||
|                     .commitNow(); | ||||
|         } | ||||
|         return persistFragment; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -672,8 +907,8 @@ public class RouterActivity extends AppCompatActivity { | ||||
|         final int icon; | ||||
|  | ||||
|         AdapterChoiceItem(final String key, final String description, final int icon) { | ||||
|             this.description = description; | ||||
|             this.key = key; | ||||
|             this.description = description; | ||||
|             this.icon = icon; | ||||
|         } | ||||
|     } | ||||
| @@ -789,7 +1024,16 @@ public class RouterActivity extends AppCompatActivity { | ||||
|                     } | ||||
|                     playQueue = new SinglePlayQueue((StreamInfo) info); | ||||
|                 } else if (info instanceof ChannelInfo) { | ||||
|                     playQueue = new ChannelPlayQueue((ChannelInfo) info); | ||||
|                     final Optional<ListLinkHandler> playableTab = ((ChannelInfo) info).getTabs() | ||||
|                             .stream() | ||||
|                             .filter(ChannelTabHelper::isStreamsTab) | ||||
|                             .findFirst(); | ||||
|  | ||||
|                     if (playableTab.isPresent()) { | ||||
|                         playQueue = new ChannelTabPlayQueue(info.getServiceId(), playableTab.get()); | ||||
|                     } else { | ||||
|                         return; // there is no playable tab | ||||
|                     } | ||||
|                 } else if (info instanceof PlaylistInfo) { | ||||
|                     playQueue = new PlaylistPlayQueue((PlaylistInfo) info); | ||||
|                 } else { | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import android.view.MenuItem | ||||
| import android.view.View | ||||
| import android.view.ViewGroup | ||||
| import android.widget.Button | ||||
| import androidx.annotation.StringRes | ||||
| import androidx.appcompat.app.AppCompatActivity | ||||
| import androidx.fragment.app.Fragment | ||||
| import androidx.fragment.app.FragmentActivity | ||||
| @@ -57,13 +58,9 @@ class AboutActivity : AppCompatActivity() { | ||||
|      * A placeholder fragment containing a simple view. | ||||
|      */ | ||||
|     class AboutFragment : Fragment() { | ||||
|         private fun Button.openLink(url: Int) { | ||||
|         private fun Button.openLink(@StringRes url: Int) { | ||||
|             setOnClickListener { | ||||
|                 ShareUtils.openUrlInBrowser( | ||||
|                     context, | ||||
|                     requireContext().getString(url), | ||||
|                     false | ||||
|                 ) | ||||
|                 ShareUtils.openUrlInApp(context, requireContext().getString(url)) | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -78,6 +75,7 @@ class AboutActivity : AppCompatActivity() { | ||||
|                 aboutDonationLink.openLink(R.string.donation_url) | ||||
|                 aboutWebsiteLink.openLink(R.string.website_url) | ||||
|                 aboutPrivacyPolicyLink.openLink(R.string.privacy_policy_url) | ||||
|                 faqLink.openLink(R.string.faq_url) | ||||
|                 return root | ||||
|             } | ||||
|         } | ||||
| @@ -118,7 +116,7 @@ class AboutActivity : AppCompatActivity() { | ||||
|         /** | ||||
|          * List of all software components. | ||||
|          */ | ||||
|         private val SOFTWARE_COMPONENTS = arrayOf( | ||||
|         private val SOFTWARE_COMPONENTS = arrayListOf( | ||||
|             SoftwareComponent( | ||||
|                 "ACRA", "2013", "Kevin Gaudin", | ||||
|                 "https://github.com/ACRA/acra", StandardLicenses.APACHE2 | ||||
|   | ||||
| @@ -1,31 +1,40 @@ | ||||
| package org.schabi.newpipe.about | ||||
|  | ||||
| import android.os.Bundle | ||||
| import android.util.Base64 | ||||
| import android.view.LayoutInflater | ||||
| import android.view.View | ||||
| import android.view.ViewGroup | ||||
| import android.webkit.WebView | ||||
| import androidx.appcompat.app.AlertDialog | ||||
| import androidx.core.os.bundleOf | ||||
| import androidx.fragment.app.Fragment | ||||
| import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers | ||||
| import io.reactivex.rxjava3.core.Observable | ||||
| import io.reactivex.rxjava3.disposables.CompositeDisposable | ||||
| import io.reactivex.rxjava3.disposables.Disposable | ||||
| import io.reactivex.rxjava3.schedulers.Schedulers | ||||
| import org.schabi.newpipe.BuildConfig | ||||
| import org.schabi.newpipe.R | ||||
| import org.schabi.newpipe.about.LicenseFragmentHelper.showLicense | ||||
| import org.schabi.newpipe.databinding.FragmentLicensesBinding | ||||
| import org.schabi.newpipe.databinding.ItemSoftwareComponentBinding | ||||
| import org.schabi.newpipe.ktx.parcelableArrayList | ||||
| import org.schabi.newpipe.util.Localization | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils | ||||
|  | ||||
| /** | ||||
|  * Fragment containing the software licenses. | ||||
|  */ | ||||
| class LicenseFragment : Fragment() { | ||||
|     private lateinit var softwareComponents: Array<SoftwareComponent> | ||||
|     private var activeLicense: License? = null | ||||
|     private lateinit var softwareComponents: List<SoftwareComponent> | ||||
|     private var activeSoftwareComponent: SoftwareComponent? = null | ||||
|     private val compositeDisposable = CompositeDisposable() | ||||
|  | ||||
|     override fun onCreate(savedInstanceState: Bundle?) { | ||||
|         super.onCreate(savedInstanceState) | ||||
|         softwareComponents = arguments?.getParcelableArray(ARG_COMPONENTS) as Array<SoftwareComponent> | ||||
|         activeLicense = savedInstanceState?.getSerializable(LICENSE_KEY) as? License | ||||
|         // Sort components by name | ||||
|         softwareComponents.sortBy { it.name } | ||||
|         softwareComponents = arguments?.parcelableArrayList<SoftwareComponent>(ARG_COMPONENTS)!! | ||||
|             .sortedBy { it.name } // Sort components by name | ||||
|         activeSoftwareComponent = savedInstanceState?.getSerializable(SOFTWARE_COMPONENT_KEY) as? SoftwareComponent | ||||
|     } | ||||
|  | ||||
|     override fun onDestroy() { | ||||
| @@ -40,9 +49,8 @@ class LicenseFragment : Fragment() { | ||||
|     ): View { | ||||
|         val binding = FragmentLicensesBinding.inflate(inflater, container, false) | ||||
|         binding.licensesAppReadLicense.setOnClickListener { | ||||
|             activeLicense = StandardLicenses.GPL3 | ||||
|             compositeDisposable.add( | ||||
|                 showLicense(activity, StandardLicenses.GPL3) | ||||
|                 showLicense(NEWPIPE_SOFTWARE_COMPONENT) | ||||
|             ) | ||||
|         } | ||||
|         for (component in softwareComponents) { | ||||
| @@ -58,27 +66,72 @@ class LicenseFragment : Fragment() { | ||||
|             val root: View = componentBinding.root | ||||
|             root.tag = component | ||||
|             root.setOnClickListener { | ||||
|                 activeLicense = component.license | ||||
|                 compositeDisposable.add( | ||||
|                     showLicense(activity, component) | ||||
|                     showLicense(component) | ||||
|                 ) | ||||
|             } | ||||
|             binding.licensesSoftwareComponents.addView(root) | ||||
|             registerForContextMenu(root) | ||||
|         } | ||||
|         activeLicense?.let { compositeDisposable.add(showLicense(activity, it)) } | ||||
|         activeSoftwareComponent?.let { compositeDisposable.add(showLicense(it)) } | ||||
|         return binding.root | ||||
|     } | ||||
|  | ||||
|     override fun onSaveInstanceState(savedInstanceState: Bundle) { | ||||
|         super.onSaveInstanceState(savedInstanceState) | ||||
|         activeLicense?.let { savedInstanceState.putSerializable(LICENSE_KEY, it) } | ||||
|         activeSoftwareComponent?.let { savedInstanceState.putSerializable(SOFTWARE_COMPONENT_KEY, it) } | ||||
|     } | ||||
|  | ||||
|     private fun showLicense( | ||||
|         softwareComponent: SoftwareComponent | ||||
|     ): Disposable { | ||||
|         return if (context == null) { | ||||
|             Disposable.empty() | ||||
|         } else { | ||||
|             val context = requireContext() | ||||
|             activeSoftwareComponent = softwareComponent | ||||
|             Observable.fromCallable { getFormattedLicense(context, softwareComponent.license) } | ||||
|                 .subscribeOn(Schedulers.io()) | ||||
|                 .observeOn(AndroidSchedulers.mainThread()) | ||||
|                 .subscribe { formattedLicense -> | ||||
|                     val webViewData = Base64.encodeToString( | ||||
|                         formattedLicense.toByteArray(), Base64.NO_PADDING | ||||
|                     ) | ||||
|                     val webView = WebView(context) | ||||
|                     webView.loadData(webViewData, "text/html; charset=UTF-8", "base64") | ||||
|  | ||||
|                     Localization.assureCorrectAppLanguage(context) | ||||
|                     val builder = AlertDialog.Builder(requireContext()) | ||||
|                         .setTitle(softwareComponent.name) | ||||
|                         .setView(webView) | ||||
|                         .setOnCancelListener { activeSoftwareComponent = null } | ||||
|                         .setOnDismissListener { activeSoftwareComponent = null } | ||||
|                         .setPositiveButton(R.string.done) { dialog, _ -> dialog.dismiss() } | ||||
|  | ||||
|                     if (softwareComponent != NEWPIPE_SOFTWARE_COMPONENT) { | ||||
|                         builder.setNeutralButton(R.string.open_website_license) { _, _ -> | ||||
|                             ShareUtils.openUrlInApp(requireContext(), softwareComponent.link) | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     builder.show() | ||||
|                 } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         private const val ARG_COMPONENTS = "components" | ||||
|         private const val LICENSE_KEY = "ACTIVE_LICENSE" | ||||
|         fun newInstance(softwareComponents: Array<SoftwareComponent>): LicenseFragment { | ||||
|         private const val SOFTWARE_COMPONENT_KEY = "ACTIVE_SOFTWARE_COMPONENT" | ||||
|         private val NEWPIPE_SOFTWARE_COMPONENT = SoftwareComponent( | ||||
|             "NewPipe", | ||||
|             "2014-2023", | ||||
|             "Team NewPipe", | ||||
|             "https://newpipe.net/", | ||||
|             StandardLicenses.GPL3, | ||||
|             BuildConfig.VERSION_NAME | ||||
|         ) | ||||
|  | ||||
|         fun newInstance(softwareComponents: ArrayList<SoftwareComponent>): LicenseFragment { | ||||
|             val fragment = LicenseFragment() | ||||
|             fragment.arguments = bundleOf(ARG_COMPONENTS to softwareComponents) | ||||
|             return fragment | ||||
|   | ||||
| @@ -1,137 +1,52 @@ | ||||
| package org.schabi.newpipe.about | ||||
|  | ||||
| import android.content.Context | ||||
| import android.util.Base64 | ||||
| import android.webkit.WebView | ||||
| import androidx.appcompat.app.AlertDialog | ||||
| import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers | ||||
| import io.reactivex.rxjava3.core.Observable | ||||
| import io.reactivex.rxjava3.disposables.Disposable | ||||
| import io.reactivex.rxjava3.schedulers.Schedulers | ||||
| import org.schabi.newpipe.R | ||||
| import org.schabi.newpipe.util.Localization | ||||
| import org.schabi.newpipe.util.ThemeHelper | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils | ||||
| import java.io.BufferedReader | ||||
| import java.io.IOException | ||||
| import java.io.InputStreamReader | ||||
| import java.nio.charset.StandardCharsets | ||||
|  | ||||
| object LicenseFragmentHelper { | ||||
|     /** | ||||
|      * @param context the context to use | ||||
|      * @param license the license | ||||
|      * @return String which contains a HTML formatted license page | ||||
|      * styled according to the context's theme | ||||
|      */ | ||||
|     private fun getFormattedLicense(context: Context, license: License): String { | ||||
|         val licenseContent = StringBuilder() | ||||
|         val webViewData: String | ||||
|         try { | ||||
|             BufferedReader( | ||||
|                 InputStreamReader( | ||||
|                     context.assets.open(license.filename), | ||||
|                     StandardCharsets.UTF_8 | ||||
|                 ) | ||||
|             ).use { `in` -> | ||||
|                 var str: String? | ||||
|                 while (`in`.readLine().also { str = it } != null) { | ||||
|                     licenseContent.append(str) | ||||
|                 } | ||||
|  | ||||
|                 // split the HTML file and insert the stylesheet into the HEAD of the file | ||||
|                 webViewData = "$licenseContent".replace( | ||||
|                     "</head>", | ||||
|                     "<style>" + getLicenseStylesheet(context) + "</style></head>" | ||||
|                 ) | ||||
|             } | ||||
|         } catch (e: IOException) { | ||||
|             throw IllegalArgumentException( | ||||
|                 "Could not get license file: " + license.filename, e | ||||
|             ) | ||||
|         } | ||||
|         return webViewData | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param context the Android context | ||||
|      * @return String which is a CSS stylesheet according to the context's theme | ||||
|      */ | ||||
|     private fun getLicenseStylesheet(context: Context): String { | ||||
|         val isLightTheme = ThemeHelper.isLightThemeSelected(context) | ||||
|         return ( | ||||
|             "body{padding:12px 15px;margin:0;" + "background:#" + getHexRGBColor( | ||||
|                 context, | ||||
|                 if (isLightTheme) R.color.light_license_background_color | ||||
|                 else R.color.dark_license_background_color | ||||
|             ) + ";" + "color:#" + getHexRGBColor( | ||||
|                 context, | ||||
|                 if (isLightTheme) R.color.light_license_text_color | ||||
|                 else R.color.dark_license_text_color | ||||
|             ) + "}" + "a[href]{color:#" + getHexRGBColor( | ||||
|                 context, | ||||
|                 if (isLightTheme) R.color.light_youtube_primary_color | ||||
|                 else R.color.dark_youtube_primary_color | ||||
|             ) + "}" + "pre{white-space:pre-wrap}" | ||||
|             ) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Cast R.color to a hexadecimal color value. | ||||
|      * | ||||
|      * @param context the context to use | ||||
|      * @param color   the color number from R.color | ||||
|      * @return a six characters long String with hexadecimal RGB values | ||||
|      */ | ||||
|     private fun getHexRGBColor(context: Context, color: Int): String { | ||||
|         return context.getString(color).substring(3) | ||||
|     } | ||||
|  | ||||
|     fun showLicense(context: Context?, license: License): Disposable { | ||||
|         return showLicense(context, license) { alertDialog -> | ||||
|             alertDialog.setPositiveButton(R.string.ok) { dialog, _ -> | ||||
|                 dialog.dismiss() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun showLicense(context: Context?, component: SoftwareComponent): Disposable { | ||||
|         return showLicense(context, component.license) { alertDialog -> | ||||
|             alertDialog.setPositiveButton(R.string.dismiss) { dialog, _ -> | ||||
|                 dialog.dismiss() | ||||
|             } | ||||
|             alertDialog.setNeutralButton(R.string.open_website_license) { _, _ -> | ||||
|                 ShareUtils.openUrlInBrowser(context!!, component.link) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun showLicense( | ||||
|         context: Context?, | ||||
|         license: License, | ||||
|         block: (AlertDialog.Builder) -> Unit | ||||
|     ): Disposable { | ||||
|         return if (context == null) { | ||||
|             Disposable.empty() | ||||
|         } else { | ||||
|             Observable.fromCallable { getFormattedLicense(context, license) } | ||||
|                 .subscribeOn(Schedulers.io()) | ||||
|                 .observeOn(AndroidSchedulers.mainThread()) | ||||
|                 .subscribe { formattedLicense -> | ||||
|                     val webViewData = Base64.encodeToString( | ||||
|                         formattedLicense.toByteArray(StandardCharsets.UTF_8), Base64.NO_PADDING | ||||
|                     ) | ||||
|                     val webView = WebView(context) | ||||
|                     webView.loadData(webViewData, "text/html; charset=UTF-8", "base64") | ||||
|  | ||||
|                     AlertDialog.Builder(context).apply { | ||||
|                         setTitle(license.name) | ||||
|                         setView(webView) | ||||
|                         Localization.assureCorrectAppLanguage(context) | ||||
|                         block(this) | ||||
|                         show() | ||||
|                     } | ||||
|                 } | ||||
|         } | ||||
| /** | ||||
|  * @param context the context to use | ||||
|  * @param license the license | ||||
|  * @return String which contains a HTML formatted license page | ||||
|  * styled according to the context's theme | ||||
|  */ | ||||
| fun getFormattedLicense(context: Context, license: License): String { | ||||
|     try { | ||||
|         return context.assets.open(license.filename).bufferedReader().use { it.readText() } | ||||
|             // split the HTML file and insert the stylesheet into the HEAD of the file | ||||
|             .replace("</head>", "<style>${getLicenseStylesheet(context)}</style></head>") | ||||
|     } catch (e: IOException) { | ||||
|         throw IllegalArgumentException("Could not get license file: ${license.filename}", e) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * @param context the Android context | ||||
|  * @return String which is a CSS stylesheet according to the context's theme | ||||
|  */ | ||||
| fun getLicenseStylesheet(context: Context): String { | ||||
|     val isLightTheme = ThemeHelper.isLightThemeSelected(context) | ||||
|     val licenseBackgroundColor = getHexRGBColor( | ||||
|         context, if (isLightTheme) R.color.light_license_background_color else R.color.dark_license_background_color | ||||
|     ) | ||||
|     val licenseTextColor = getHexRGBColor( | ||||
|         context, if (isLightTheme) R.color.light_license_text_color else R.color.dark_license_text_color | ||||
|     ) | ||||
|     val youtubePrimaryColor = getHexRGBColor( | ||||
|         context, if (isLightTheme) R.color.light_youtube_primary_color else R.color.dark_youtube_primary_color | ||||
|     ) | ||||
|     return "body{padding:12px 15px;margin:0;background:#$licenseBackgroundColor;color:#$licenseTextColor}" + | ||||
|         "a[href]{color:#$youtubePrimaryColor}pre{white-space:pre-wrap}" | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Cast R.color to a hexadecimal color value. | ||||
|  * | ||||
|  * @param context the context to use | ||||
|  * @param color   the color number from R.color | ||||
|  * @return a six characters long String with hexadecimal RGB values | ||||
|  */ | ||||
| fun getHexRGBColor(context: Context, color: Int): String { | ||||
|     return context.getString(color).substring(3) | ||||
| } | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package org.schabi.newpipe.about | ||||
|  | ||||
| import android.os.Parcelable | ||||
| import kotlinx.parcelize.Parcelize | ||||
| import java.io.Serializable | ||||
|  | ||||
| @Parcelize | ||||
| class SoftwareComponent | ||||
| @@ -13,4 +14,4 @@ constructor( | ||||
|     val link: String, | ||||
|     val license: License, | ||||
|     val version: String? = null | ||||
| ) : Parcelable | ||||
| ) : Parcelable, Serializable | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| package org.schabi.newpipe.database; | ||||
|  | ||||
| import static org.schabi.newpipe.database.Migrations.DB_VER_6; | ||||
| import static org.schabi.newpipe.database.Migrations.DB_VER_9; | ||||
|  | ||||
| import androidx.room.Database; | ||||
| import androidx.room.RoomDatabase; | ||||
| @@ -38,7 +38,7 @@ import org.schabi.newpipe.database.subscription.SubscriptionEntity; | ||||
|                 FeedEntity.class, FeedGroupEntity.class, FeedGroupSubscriptionEntity.class, | ||||
|                 FeedLastUpdatedEntity.class | ||||
|         }, | ||||
|         version = DB_VER_6 | ||||
|         version = DB_VER_9 | ||||
| ) | ||||
| public abstract class AppDatabase extends RoomDatabase { | ||||
|     public static final String DATABASE_NAME = "newpipe.db"; | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package org.schabi.newpipe.database; | ||||
| import androidx.room.Dao; | ||||
| import androidx.room.Delete; | ||||
| import androidx.room.Insert; | ||||
| import androidx.room.OnConflictStrategy; | ||||
| import androidx.room.Update; | ||||
|  | ||||
| import java.util.Collection; | ||||
| @@ -14,13 +13,10 @@ import io.reactivex.rxjava3.core.Flowable; | ||||
| @Dao | ||||
| public interface BasicDAO<Entity> { | ||||
|     /* Inserts */ | ||||
|     @Insert(onConflict = OnConflictStrategy.ABORT) | ||||
|     @Insert | ||||
|     long insert(Entity entity); | ||||
|  | ||||
|     @Insert(onConflict = OnConflictStrategy.ABORT) | ||||
|     List<Long> insertAll(Entity... entities); | ||||
|  | ||||
|     @Insert(onConflict = OnConflictStrategy.ABORT) | ||||
|     @Insert | ||||
|     List<Long> insertAll(Collection<Entity> entities); | ||||
|  | ||||
|     /* Searches */ | ||||
| @@ -32,9 +28,6 @@ public interface BasicDAO<Entity> { | ||||
|     @Delete | ||||
|     void delete(Entity entity); | ||||
|  | ||||
|     @Delete | ||||
|     int delete(Collection<Entity> entities); | ||||
|  | ||||
|     int deleteAll(); | ||||
|  | ||||
|     /* Updates */ | ||||
|   | ||||
| @@ -7,7 +7,7 @@ import java.time.Instant | ||||
| import java.time.OffsetDateTime | ||||
| import java.time.ZoneOffset | ||||
|  | ||||
| object Converters { | ||||
| class Converters { | ||||
|     /** | ||||
|      * Convert a long value to a [OffsetDateTime]. | ||||
|      * | ||||
| @@ -47,6 +47,6 @@ object Converters { | ||||
|  | ||||
|     @TypeConverter | ||||
|     fun feedGroupIconOf(id: Int): FeedGroupIcon { | ||||
|         return FeedGroupIcon.values().first { it.id == id } | ||||
|         return FeedGroupIcon.entries.first { it.id == id } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -24,6 +24,9 @@ public final class Migrations { | ||||
|     public static final int DB_VER_4 = 4; | ||||
|     public static final int DB_VER_5 = 5; | ||||
|     public static final int DB_VER_6 = 6; | ||||
|     public static final int DB_VER_7 = 7; | ||||
|     public static final int DB_VER_8 = 8; | ||||
|     public static final int DB_VER_9 = 9; | ||||
|  | ||||
|     private static final String TAG = Migrations.class.getName(); | ||||
|     public static final boolean DEBUG = MainActivity.DEBUG; | ||||
| @@ -190,6 +193,60 @@ public final class Migrations { | ||||
|     }; | ||||
|  | ||||
|     public static final Migration MIGRATION_5_6 = new Migration(DB_VER_5, DB_VER_6) { | ||||
|         @Override | ||||
|         public void migrate(@NonNull final SupportSQLiteDatabase database) { | ||||
|             database.execSQL("ALTER TABLE `playlists` ADD COLUMN `is_thumbnail_permanent` " | ||||
|                     + "INTEGER NOT NULL DEFAULT 0"); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     public static final Migration MIGRATION_6_7 = new Migration(DB_VER_6, DB_VER_7) { | ||||
|         @Override | ||||
|         public void migrate(@NonNull final SupportSQLiteDatabase database) { | ||||
|             // Create a new column thumbnail_stream_id | ||||
|             database.execSQL("ALTER TABLE `playlists` ADD COLUMN `thumbnail_stream_id` " | ||||
|                     + "INTEGER NOT NULL DEFAULT -1"); | ||||
|  | ||||
|             // Migrate the thumbnail_url to the thumbnail_stream_id | ||||
|             database.execSQL("UPDATE playlists SET thumbnail_stream_id = (" | ||||
|                     + " SELECT CASE WHEN COUNT(*) != 0 then stream_uid ELSE -1 END" | ||||
|                     + " FROM (" | ||||
|                     + " SELECT p.uid AS playlist_uid, s.uid AS stream_uid" | ||||
|                     + " FROM playlists p" | ||||
|                     + " LEFT JOIN playlist_stream_join ps ON p.uid = ps.playlist_id" | ||||
|                     + " LEFT JOIN streams s ON s.uid = ps.stream_id" | ||||
|                     + " WHERE s.thumbnail_url = p.thumbnail_url) AS temporary_table" | ||||
|                     + " WHERE playlist_uid = playlists.uid)"); | ||||
|  | ||||
|             // Remove the thumbnail_url field in the playlist table | ||||
|             database.execSQL("CREATE TABLE IF NOT EXISTS `playlists_new`" | ||||
|                     + "(uid INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, " | ||||
|                     + "name TEXT, " | ||||
|                     + "is_thumbnail_permanent INTEGER NOT NULL, " | ||||
|                     + "thumbnail_stream_id INTEGER NOT NULL)"); | ||||
|  | ||||
|             database.execSQL("INSERT INTO playlists_new" | ||||
|                     + " SELECT uid, name, is_thumbnail_permanent, thumbnail_stream_id " | ||||
|                     + " FROM playlists"); | ||||
|  | ||||
|  | ||||
|             database.execSQL("DROP TABLE playlists"); | ||||
|             database.execSQL("ALTER TABLE playlists_new RENAME TO playlists"); | ||||
|             database.execSQL("CREATE INDEX IF NOT EXISTS " | ||||
|                     + "`index_playlists_name` ON `playlists` (`name`)"); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     public static final Migration MIGRATION_7_8 = new Migration(DB_VER_7, DB_VER_8) { | ||||
|         @Override | ||||
|         public void migrate(@NonNull final SupportSQLiteDatabase database) { | ||||
|             database.execSQL("DELETE FROM search_history WHERE id NOT IN (SELECT id FROM (SELECT " | ||||
|                     + "MIN(id) as id FROM search_history GROUP BY trim(search), service_id ) tmp)"); | ||||
|             database.execSQL("UPDATE search_history SET search = trim(search)"); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     public static final Migration MIGRATION_8_9 = new Migration(DB_VER_8, DB_VER_9) { | ||||
|         @Override | ||||
|         public void migrate(@NonNull final SupportSQLiteDatabase database) { | ||||
|             try { | ||||
| @@ -199,10 +256,13 @@ public final class Migrations { | ||||
|                 // Create a temp table to initialize display_index. | ||||
|                 database.execSQL("CREATE TABLE `playlists_tmp` " | ||||
|                         + "(`uid` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, " | ||||
|                         + "`name` TEXT, `thumbnail_url` TEXT," | ||||
|                         + "`name` TEXT, `is_thumbnail_permanent` INTEGER NOT NULL, " | ||||
|                         + "`thumbnail_stream_id` INTEGER NOT NULL, " | ||||
|                         + "`display_index` INTEGER NOT NULL DEFAULT 0)"); | ||||
|                 database.execSQL("INSERT INTO `playlists_tmp` (`uid`, `name`, `thumbnail_url`)" | ||||
|                         + "SELECT `uid`, `name`, `thumbnail_url` FROM `playlists`"); | ||||
|                 database.execSQL("INSERT INTO `playlists_tmp` " | ||||
|                         + "(`uid`, `name`, `is_thumbnail_permanent`, `thumbnail_stream_id`) " | ||||
|                         + "SELECT `uid`, `name`, `is_thumbnail_permanent`, `thumbnail_stream_id` " | ||||
|                         + "FROM `playlists`"); | ||||
|  | ||||
|                 // Replace the old table. | ||||
|                 database.execSQL("DROP TABLE `playlists`"); | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import androidx.room.Update | ||||
| import io.reactivex.rxjava3.core.Flowable | ||||
| import io.reactivex.rxjava3.core.Maybe | ||||
| import org.schabi.newpipe.database.feed.model.FeedEntity | ||||
| import org.schabi.newpipe.database.feed.model.FeedGroupEntity | ||||
| import org.schabi.newpipe.database.feed.model.FeedLastUpdatedEntity | ||||
| import org.schabi.newpipe.database.stream.StreamWithState | ||||
| import org.schabi.newpipe.database.stream.model.StreamStateEntity | ||||
| @@ -21,56 +22,17 @@ abstract class FeedDAO { | ||||
|     @Query("DELETE FROM feed") | ||||
|     abstract fun deleteAll(): Int | ||||
|  | ||||
|     @Query( | ||||
|         """ | ||||
|         SELECT s.*, sst.progress_time | ||||
|         FROM streams s | ||||
|  | ||||
|         LEFT JOIN stream_state sst | ||||
|         ON s.uid = sst.stream_id | ||||
|  | ||||
|         LEFT JOIN stream_history sh | ||||
|         ON s.uid = sh.stream_id | ||||
|  | ||||
|         INNER JOIN feed f | ||||
|         ON s.uid = f.stream_id | ||||
|  | ||||
|         ORDER BY s.upload_date IS NULL DESC, s.upload_date DESC, s.uploader ASC | ||||
|         LIMIT 500 | ||||
|         """ | ||||
|     ) | ||||
|     abstract fun getAllStreams(): Maybe<List<StreamWithState>> | ||||
|  | ||||
|     @Query( | ||||
|         """ | ||||
|         SELECT s.*, sst.progress_time | ||||
|         FROM streams s | ||||
|  | ||||
|         LEFT JOIN stream_state sst | ||||
|         ON s.uid = sst.stream_id | ||||
|  | ||||
|         LEFT JOIN stream_history sh | ||||
|         ON s.uid = sh.stream_id | ||||
|  | ||||
|         INNER JOIN feed f | ||||
|         ON s.uid = f.stream_id | ||||
|  | ||||
|         INNER JOIN feed_group_subscription_join fgs | ||||
|         ON fgs.subscription_id = f.subscription_id | ||||
|  | ||||
|         WHERE fgs.group_id = :groupId | ||||
|  | ||||
|         ORDER BY s.upload_date IS NULL DESC, s.upload_date DESC, s.uploader ASC | ||||
|         LIMIT 500 | ||||
|         """ | ||||
|     ) | ||||
|     abstract fun getAllStreamsForGroup(groupId: Long): Maybe<List<StreamWithState>> | ||||
|  | ||||
|     /** | ||||
|      * @param groupId          the group id to get feed streams of; use | ||||
|      *                         [FeedGroupEntity.GROUP_ALL_ID] to not filter by group | ||||
|      * @param includePlayed    if false, only return all of the live, never-played or non-finished | ||||
|      *                         feed streams (see `@see` items); if true no filter is applied | ||||
|      * @param uploadDateBefore get only streams uploaded before this date (useful to filter out | ||||
|      *                         future streams); use null to not filter by upload date | ||||
|      * @return the feed streams filtered according to the conditions provided in the parameters | ||||
|      * @see StreamStateEntity.isFinished() | ||||
|      * @see StreamStateEntity.PLAYBACK_FINISHED_END_MILLISECONDS | ||||
|      * @return all of the non-live, never-played and non-finished streams in the feed | ||||
|      *         (all of the cited conditions must hold for a stream to be in the returned list) | ||||
|      * @see StreamStateEntity.PLAYBACK_SAVE_THRESHOLD_START_MILLISECONDS | ||||
|      */ | ||||
|     @Query( | ||||
|         """ | ||||
| @@ -79,80 +41,82 @@ abstract class FeedDAO { | ||||
|  | ||||
|         LEFT JOIN stream_state sst | ||||
|         ON s.uid = sst.stream_id | ||||
|  | ||||
|          | ||||
|         LEFT JOIN stream_history sh | ||||
|         ON s.uid = sh.stream_id     | ||||
|              | ||||
|         ON s.uid = sh.stream_id | ||||
|          | ||||
|         INNER JOIN feed f | ||||
|         ON s.uid = f.stream_id | ||||
|  | ||||
|         LEFT JOIN feed_group_subscription_join fgs | ||||
|         ON ( | ||||
|             :groupId <> ${FeedGroupEntity.GROUP_ALL_ID} | ||||
|             AND fgs.subscription_id = f.subscription_id | ||||
|         ) | ||||
|  | ||||
|         WHERE ( | ||||
|             sh.stream_id IS NULL | ||||
|             :groupId = ${FeedGroupEntity.GROUP_ALL_ID} | ||||
|             OR fgs.group_id = :groupId | ||||
|         ) | ||||
|         AND ( | ||||
|             :includePlayed | ||||
|             OR sh.stream_id IS NULL | ||||
|             OR sst.stream_id IS NULL | ||||
|             OR sst.progress_time < s.duration * 1000 - ${StreamStateEntity.PLAYBACK_FINISHED_END_MILLISECONDS} | ||||
|             OR sst.progress_time < s.duration * 1000 * 3 / 4 | ||||
|             OR s.stream_type = 'LIVE_STREAM' | ||||
|             OR s.stream_type = 'AUDIO_LIVE_STREAM' | ||||
|         ) | ||||
|         AND ( | ||||
|             :includePartiallyPlayed | ||||
|             OR sh.stream_id IS NULL | ||||
|             OR sst.stream_id IS NULL | ||||
|             OR (sst.progress_time <= ${StreamStateEntity.PLAYBACK_SAVE_THRESHOLD_START_MILLISECONDS} | ||||
|             AND sst.progress_time <= s.duration * 1000 / 4) | ||||
|             OR (sst.progress_time >= s.duration * 1000 - ${StreamStateEntity.PLAYBACK_FINISHED_END_MILLISECONDS} | ||||
|             AND sst.progress_time >= s.duration * 1000 * 3 / 4) | ||||
|         ) | ||||
|         AND ( | ||||
|             :uploadDateBefore IS NULL | ||||
|             OR s.upload_date IS NULL | ||||
|             OR s.upload_date < :uploadDateBefore | ||||
|         ) | ||||
|  | ||||
|         ORDER BY s.upload_date IS NULL DESC, s.upload_date DESC, s.uploader ASC | ||||
|         LIMIT 500 | ||||
|         """ | ||||
|     ) | ||||
|     abstract fun getLiveOrNotPlayedStreams(): Maybe<List<StreamWithState>> | ||||
|     abstract fun getStreams( | ||||
|         groupId: Long, | ||||
|         includePlayed: Boolean, | ||||
|         includePartiallyPlayed: Boolean, | ||||
|         uploadDateBefore: OffsetDateTime? | ||||
|     ): Maybe<List<StreamWithState>> | ||||
|  | ||||
|     /** | ||||
|      * @see StreamStateEntity.isFinished() | ||||
|      * @see StreamStateEntity.PLAYBACK_FINISHED_END_MILLISECONDS | ||||
|      * @param groupId the group id to get streams of | ||||
|      * @return all of the non-live, never-played and non-finished streams for the given feed group | ||||
|      *         (all of the cited conditions must hold for a stream to be in the returned list) | ||||
|      * Remove links to streams that are older than the given date | ||||
|      * **but keep at least one stream per uploader**. | ||||
|      * | ||||
|      * One stream per uploader is kept because it is needed as reference | ||||
|      * when fetching new streams to check if they are new or not. | ||||
|      * @param offsetDateTime the newest date to keep, older streams are removed | ||||
|      */ | ||||
|     @Query( | ||||
|         """ | ||||
|         SELECT s.*, sst.progress_time | ||||
|         FROM streams s | ||||
|  | ||||
|         LEFT JOIN stream_state sst | ||||
|         ON s.uid = sst.stream_id | ||||
|         DELETE FROM feed | ||||
|         WHERE feed.stream_id IN (SELECT uid from ( | ||||
|               SELECT s.uid, | ||||
|               (SELECT MAX(upload_date) | ||||
|                     FROM streams s1 | ||||
|                     INNER JOIN feed f1 | ||||
|                     ON s1.uid = f1.stream_id | ||||
|                     WHERE f1.subscription_id = f.subscription_id) max_upload_date | ||||
|               FROM streams s | ||||
|               INNER JOIN feed f | ||||
|               ON s.uid = f.stream_id | ||||
|          | ||||
|         LEFT JOIN stream_history sh | ||||
|         ON s.uid = sh.stream_id | ||||
|          | ||||
|         INNER JOIN feed f | ||||
|         ON s.uid = f.stream_id | ||||
|  | ||||
|         INNER JOIN feed_group_subscription_join fgs | ||||
|         ON fgs.subscription_id = f.subscription_id | ||||
|  | ||||
|         WHERE fgs.group_id = :groupId | ||||
|         AND ( | ||||
|             sh.stream_id IS NULL | ||||
|             OR sst.stream_id IS NULL | ||||
|             OR sst.progress_time < s.duration * 1000 - ${StreamStateEntity.PLAYBACK_FINISHED_END_MILLISECONDS} | ||||
|             OR sst.progress_time < s.duration * 1000 * 3 / 4 | ||||
|             OR s.stream_type = 'LIVE_STREAM' | ||||
|             OR s.stream_type = 'AUDIO_LIVE_STREAM' | ||||
|         ) | ||||
|  | ||||
|         ORDER BY s.upload_date IS NULL DESC, s.upload_date DESC, s.uploader ASC | ||||
|         LIMIT 500 | ||||
|         """ | ||||
|     ) | ||||
|     abstract fun getLiveOrNotPlayedStreamsForGroup(groupId: Long): Maybe<List<StreamWithState>> | ||||
|  | ||||
|     @Query( | ||||
|         """ | ||||
|         DELETE FROM feed WHERE | ||||
|  | ||||
|         feed.stream_id IN ( | ||||
|             SELECT s.uid FROM streams s | ||||
|  | ||||
|             INNER JOIN feed f | ||||
|             ON s.uid = f.stream_id | ||||
|  | ||||
|             WHERE s.upload_date < :offsetDateTime | ||||
|         ) | ||||
|               WHERE s.upload_date < :offsetDateTime | ||||
|               AND   s.upload_date <> max_upload_date)) | ||||
|         """ | ||||
|     ) | ||||
|     abstract fun unlinkStreamsOlderThan(offsetDateTime: OffsetDateTime) | ||||
|   | ||||
| @@ -3,7 +3,6 @@ package org.schabi.newpipe.database.feed.model | ||||
| import androidx.room.ColumnInfo | ||||
| import androidx.room.Entity | ||||
| import androidx.room.ForeignKey | ||||
| import androidx.room.ForeignKey.CASCADE | ||||
| import androidx.room.Index | ||||
| import org.schabi.newpipe.database.feed.model.FeedGroupSubscriptionEntity.Companion.FEED_GROUP_SUBSCRIPTION_TABLE | ||||
| import org.schabi.newpipe.database.feed.model.FeedGroupSubscriptionEntity.Companion.GROUP_ID | ||||
| @@ -19,14 +18,14 @@ import org.schabi.newpipe.database.subscription.SubscriptionEntity | ||||
|             entity = FeedGroupEntity::class, | ||||
|             parentColumns = [FeedGroupEntity.ID], | ||||
|             childColumns = [GROUP_ID], | ||||
|             onDelete = CASCADE, onUpdate = CASCADE, deferred = true | ||||
|             onDelete = ForeignKey.CASCADE, onUpdate = ForeignKey.CASCADE, deferred = true | ||||
|         ), | ||||
|  | ||||
|         ForeignKey( | ||||
|             entity = SubscriptionEntity::class, | ||||
|             parentColumns = [SubscriptionEntity.SUBSCRIPTION_UID], | ||||
|             childColumns = [SUBSCRIPTION_ID], | ||||
|             onDelete = CASCADE, onUpdate = CASCADE, deferred = true | ||||
|             onDelete = ForeignKey.CASCADE, onUpdate = ForeignKey.CASCADE, deferred = true | ||||
|         ) | ||||
|     ] | ||||
| ) | ||||
|   | ||||
| @@ -4,7 +4,6 @@ import androidx.annotation.NonNull; | ||||
| import androidx.room.ColumnInfo; | ||||
| import androidx.room.Entity; | ||||
| import androidx.room.ForeignKey; | ||||
| import androidx.room.Ignore; | ||||
| import androidx.room.Index; | ||||
|  | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity; | ||||
| @@ -42,18 +41,19 @@ public class StreamHistoryEntity { | ||||
|     @ColumnInfo(name = STREAM_REPEAT_COUNT) | ||||
|     private long repeatCount; | ||||
|  | ||||
|     public StreamHistoryEntity(final long streamUid, @NonNull final OffsetDateTime accessDate, | ||||
|     /** | ||||
|      * @param streamUid the stream id this history item will refer to | ||||
|      * @param accessDate the last time the stream was accessed | ||||
|      * @param repeatCount the total number of views this stream received | ||||
|      */ | ||||
|     public StreamHistoryEntity(final long streamUid, | ||||
|                                @NonNull final OffsetDateTime accessDate, | ||||
|                                final long repeatCount) { | ||||
|         this.streamUid = streamUid; | ||||
|         this.accessDate = accessDate; | ||||
|         this.repeatCount = repeatCount; | ||||
|     } | ||||
|  | ||||
|     @Ignore | ||||
|     public StreamHistoryEntity(final long streamUid, @NonNull final OffsetDateTime accessDate) { | ||||
|         this(streamUid, accessDate, 1); | ||||
|     } | ||||
|  | ||||
|     public long getStreamUid() { | ||||
|         return streamUid; | ||||
|     } | ||||
|   | ||||
| @@ -0,0 +1,28 @@ | ||||
| package org.schabi.newpipe.database.playlist; | ||||
|  | ||||
| import androidx.room.ColumnInfo; | ||||
|  | ||||
| /** | ||||
|  * This class adds a field to {@link PlaylistMetadataEntry} that contains an integer representing | ||||
|  * how many times a specific stream is already contained inside a local playlist. Used to be able | ||||
|  * to grey out playlists which already contain the current stream in the playlist append dialog. | ||||
|  * @see org.schabi.newpipe.local.playlist.LocalPlaylistManager#getPlaylistDuplicates(String) | ||||
|  */ | ||||
| public class PlaylistDuplicatesEntry extends PlaylistMetadataEntry { | ||||
|     public static final String PLAYLIST_TIMES_STREAM_IS_CONTAINED = "timesStreamIsContained"; | ||||
|     @ColumnInfo(name = PLAYLIST_TIMES_STREAM_IS_CONTAINED) | ||||
|     public final long timesStreamIsContained; | ||||
|  | ||||
|     public PlaylistDuplicatesEntry(final long uid, | ||||
|                                    final String name, | ||||
|                                    final String thumbnailUrl, | ||||
|                                    final boolean isThumbnailPermanent, | ||||
|                                    final long thumbnailStreamId, | ||||
|                                    final long displayIndex, | ||||
|                                    final long streamCount, | ||||
|                                    final long timesStreamIsContained) { | ||||
|         super(uid, name, thumbnailUrl, isThumbnailPermanent, thumbnailStreamId, displayIndex, | ||||
|                 streamCount); | ||||
|         this.timesStreamIsContained = timesStreamIsContained; | ||||
|     } | ||||
| } | ||||
| @@ -28,7 +28,6 @@ public interface PlaylistLocalItem extends LocalItem { | ||||
|     static List<PlaylistLocalItem> merge( | ||||
|             final List<PlaylistMetadataEntry> localPlaylists, | ||||
|             final List<PlaylistRemoteEntity> remotePlaylists) { | ||||
|  | ||||
|         Collections.sort(localPlaylists, | ||||
|                 Comparator.comparingLong(PlaylistMetadataEntry::getDisplayIndex)); | ||||
|         Collections.sort(remotePlaylists, | ||||
|   | ||||
| @@ -5,6 +5,8 @@ import androidx.room.ColumnInfo; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_DISPLAY_INDEX; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_ID; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_NAME; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_THUMBNAIL_PERMANENT; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_THUMBNAIL_STREAM_ID; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_THUMBNAIL_URL; | ||||
|  | ||||
| public class PlaylistMetadataEntry implements PlaylistLocalItem { | ||||
| @@ -14,6 +16,10 @@ public class PlaylistMetadataEntry implements PlaylistLocalItem { | ||||
|     private final long uid; | ||||
|     @ColumnInfo(name = PLAYLIST_NAME) | ||||
|     public final String name; | ||||
|     @ColumnInfo(name = PLAYLIST_THUMBNAIL_PERMANENT) | ||||
|     private final boolean isThumbnailPermanent; | ||||
|     @ColumnInfo(name = PLAYLIST_THUMBNAIL_STREAM_ID) | ||||
|     private final long thumbnailStreamId; | ||||
|     @ColumnInfo(name = PLAYLIST_THUMBNAIL_URL) | ||||
|     public final String thumbnailUrl; | ||||
|     @ColumnInfo(name = PLAYLIST_DISPLAY_INDEX) | ||||
| @@ -22,10 +28,13 @@ public class PlaylistMetadataEntry implements PlaylistLocalItem { | ||||
|     public final long streamCount; | ||||
|  | ||||
|     public PlaylistMetadataEntry(final long uid, final String name, final String thumbnailUrl, | ||||
|                                  final boolean isThumbnailPermanent, final long thumbnailStreamId, | ||||
|                                  final long displayIndex, final long streamCount) { | ||||
|         this.uid = uid; | ||||
|         this.name = name; | ||||
|         this.thumbnailUrl = thumbnailUrl; | ||||
|         this.isThumbnailPermanent = isThumbnailPermanent; | ||||
|         this.thumbnailStreamId = thumbnailStreamId; | ||||
|         this.displayIndex = displayIndex; | ||||
|         this.streamCount = streamCount; | ||||
|     } | ||||
| @@ -40,6 +49,14 @@ public class PlaylistMetadataEntry implements PlaylistLocalItem { | ||||
|         return name; | ||||
|     } | ||||
|  | ||||
|     public boolean isThumbnailPermanent() { | ||||
|         return isThumbnailPermanent; | ||||
|     } | ||||
|  | ||||
|     public long getThumbnailStreamId() { | ||||
|         return thumbnailStreamId; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public long getDisplayIndex() { | ||||
|         return displayIndex; | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import org.schabi.newpipe.database.playlist.model.PlaylistStreamEntity | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity | ||||
| import org.schabi.newpipe.database.stream.model.StreamStateEntity | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem | ||||
| import org.schabi.newpipe.util.image.ImageStrategy | ||||
|  | ||||
| data class PlaylistStreamEntry( | ||||
|     @Embedded | ||||
| @@ -28,7 +29,7 @@ data class PlaylistStreamEntry( | ||||
|         item.duration = streamEntity.duration | ||||
|         item.uploaderName = streamEntity.uploader | ||||
|         item.uploaderUrl = streamEntity.uploaderUrl | ||||
|         item.thumbnailUrl = streamEntity.thumbnailUrl | ||||
|         item.thumbnails = ImageStrategy.dbUrlToImageList(streamEntity.thumbnailUrl) | ||||
|  | ||||
|         return item | ||||
|     } | ||||
|   | ||||
| @@ -6,19 +6,25 @@ import androidx.room.RewriteQueriesToDropUnusedColumns; | ||||
| import androidx.room.Transaction; | ||||
|  | ||||
| import org.schabi.newpipe.database.BasicDAO; | ||||
| import org.schabi.newpipe.database.playlist.PlaylistDuplicatesEntry; | ||||
| import org.schabi.newpipe.database.playlist.PlaylistMetadataEntry; | ||||
| import org.schabi.newpipe.database.playlist.PlaylistStreamEntry; | ||||
| import org.schabi.newpipe.database.playlist.model.PlaylistEntity; | ||||
| import org.schabi.newpipe.database.playlist.model.PlaylistStreamEntity; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import io.reactivex.rxjava3.core.Flowable; | ||||
|  | ||||
| import static org.schabi.newpipe.database.playlist.PlaylistDuplicatesEntry.PLAYLIST_TIMES_STREAM_IS_CONTAINED; | ||||
| import static org.schabi.newpipe.database.playlist.PlaylistMetadataEntry.PLAYLIST_STREAM_COUNT; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_DISPLAY_INDEX; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.DEFAULT_THUMBNAIL; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_ID; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_NAME; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_TABLE; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_THUMBNAIL_PERMANENT; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_THUMBNAIL_STREAM_ID; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_THUMBNAIL_URL; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistStreamEntity.JOIN_INDEX; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistStreamEntity.JOIN_PLAYLIST_ID; | ||||
| @@ -26,6 +32,8 @@ import static org.schabi.newpipe.database.playlist.model.PlaylistStreamEntity.JO | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistStreamEntity.PLAYLIST_STREAM_JOIN_TABLE; | ||||
| import static org.schabi.newpipe.database.stream.model.StreamEntity.STREAM_ID; | ||||
| import static org.schabi.newpipe.database.stream.model.StreamEntity.STREAM_TABLE; | ||||
| import static org.schabi.newpipe.database.stream.model.StreamEntity.STREAM_THUMBNAIL_URL; | ||||
| import static org.schabi.newpipe.database.stream.model.StreamEntity.STREAM_URL; | ||||
| import static org.schabi.newpipe.database.stream.model.StreamStateEntity.JOIN_STREAM_ID_ALIAS; | ||||
| import static org.schabi.newpipe.database.stream.model.StreamStateEntity.STREAM_PROGRESS_MILLIS; | ||||
| import static org.schabi.newpipe.database.stream.model.StreamStateEntity.STREAM_STATE_TABLE; | ||||
| @@ -54,6 +62,16 @@ public interface PlaylistStreamDAO extends BasicDAO<PlaylistStreamEntity> { | ||||
|             + " WHERE " + JOIN_PLAYLIST_ID + " = :playlistId") | ||||
|     Flowable<Integer> getMaximumIndexOf(long playlistId); | ||||
|  | ||||
|     @Query("SELECT CASE WHEN COUNT(*) != 0 then " + STREAM_ID | ||||
|             + " ELSE " + PlaylistEntity.DEFAULT_THUMBNAIL_ID + " END" | ||||
|             + " FROM " + STREAM_TABLE | ||||
|             + " LEFT JOIN " + PLAYLIST_STREAM_JOIN_TABLE | ||||
|             + " ON " + STREAM_ID + " = " + JOIN_STREAM_ID | ||||
|             + " WHERE " + JOIN_PLAYLIST_ID + " = :playlistId " | ||||
|             + " LIMIT 1" | ||||
|     ) | ||||
|     Flowable<Long> getAutomaticThumbnailStreamId(long playlistId); | ||||
|  | ||||
|     @RewriteQueriesToDropUnusedColumns | ||||
|     @Transaction | ||||
|     @Query("SELECT * FROM " + STREAM_TABLE + " INNER JOIN " | ||||
| @@ -75,26 +93,87 @@ public interface PlaylistStreamDAO extends BasicDAO<PlaylistStreamEntity> { | ||||
|     Flowable<List<PlaylistStreamEntry>> getOrderedStreamsOf(long playlistId); | ||||
|  | ||||
|     @Transaction | ||||
|     @Query("SELECT " + PLAYLIST_ID + ", " + PLAYLIST_NAME + ", " + PLAYLIST_THUMBNAIL_URL + ", " | ||||
|     @Query("SELECT " + PLAYLIST_ID + ", " + PLAYLIST_NAME + ", " | ||||
|             + PLAYLIST_THUMBNAIL_PERMANENT + ", " + PLAYLIST_THUMBNAIL_STREAM_ID + ", " | ||||
|             + PLAYLIST_DISPLAY_INDEX + ", " | ||||
|             + "COALESCE(COUNT(" + JOIN_PLAYLIST_ID + "), 0) AS " + PLAYLIST_STREAM_COUNT | ||||
|  | ||||
|             + " CASE WHEN " + PLAYLIST_THUMBNAIL_STREAM_ID + " = " | ||||
|             + PlaylistEntity.DEFAULT_THUMBNAIL_ID + " THEN " + "'" + DEFAULT_THUMBNAIL + "'" | ||||
|             + " ELSE (SELECT " + STREAM_THUMBNAIL_URL | ||||
|             + " FROM " + STREAM_TABLE | ||||
|             + " WHERE " + STREAM_TABLE + "." + STREAM_ID + " = " + PLAYLIST_THUMBNAIL_STREAM_ID | ||||
|             + " ) END AS " + PLAYLIST_THUMBNAIL_URL + ", " | ||||
|  | ||||
|             + "COALESCE(COUNT(" + JOIN_PLAYLIST_ID + "), 0) AS " + PLAYLIST_STREAM_COUNT | ||||
|             + " FROM " + PLAYLIST_TABLE | ||||
|             + " LEFT JOIN " + PLAYLIST_STREAM_JOIN_TABLE | ||||
|             + " ON " + PLAYLIST_ID + " = " + JOIN_PLAYLIST_ID | ||||
|             + " ON " + PLAYLIST_TABLE + "." + PLAYLIST_ID + " = " + JOIN_PLAYLIST_ID | ||||
|             + " GROUP BY " + PLAYLIST_ID | ||||
|             + " ORDER BY " + PLAYLIST_NAME + " COLLATE NOCASE ASC") | ||||
|     Flowable<List<PlaylistMetadataEntry>> getPlaylistMetadata(); | ||||
|  | ||||
|     @Transaction | ||||
|     @Query("SELECT " + PLAYLIST_ID + ", " + PLAYLIST_NAME + ", " + PLAYLIST_THUMBNAIL_URL + ", " | ||||
|     @Query("SELECT " + PLAYLIST_ID + ", " + PLAYLIST_NAME + ", " | ||||
|             + PLAYLIST_THUMBNAIL_PERMANENT + ", " + PLAYLIST_THUMBNAIL_STREAM_ID + ", " | ||||
|             + PLAYLIST_DISPLAY_INDEX + ", " | ||||
|             + "COALESCE(COUNT(" + JOIN_PLAYLIST_ID + "), 0) AS " + PLAYLIST_STREAM_COUNT | ||||
|  | ||||
|             + " CASE WHEN " + PLAYLIST_THUMBNAIL_STREAM_ID + " = " | ||||
|             + PlaylistEntity.DEFAULT_THUMBNAIL_ID + " THEN " + "'" + DEFAULT_THUMBNAIL + "'" | ||||
|             + " ELSE (SELECT " + STREAM_THUMBNAIL_URL | ||||
|             + " FROM " + STREAM_TABLE | ||||
|             + " WHERE " + STREAM_TABLE + "." + STREAM_ID + " = " + PLAYLIST_THUMBNAIL_STREAM_ID | ||||
|             + " ) END AS " + PLAYLIST_THUMBNAIL_URL + ", " | ||||
|  | ||||
|             + "COALESCE(COUNT(" + JOIN_PLAYLIST_ID + "), 0) AS " + PLAYLIST_STREAM_COUNT | ||||
|             + " FROM " + PLAYLIST_TABLE | ||||
|             + " LEFT JOIN " + PLAYLIST_STREAM_JOIN_TABLE | ||||
|             + " ON " + PLAYLIST_ID + " = " + JOIN_PLAYLIST_ID | ||||
|             + " ON " + PLAYLIST_TABLE + "." + PLAYLIST_ID + " = " + JOIN_PLAYLIST_ID | ||||
|             + " GROUP BY " + PLAYLIST_ID | ||||
|             + " ORDER BY " + PLAYLIST_DISPLAY_INDEX) | ||||
|     Flowable<List<PlaylistMetadataEntry>> getDisplayIndexOrderedPlaylistMetadata(); | ||||
|  | ||||
|     @RewriteQueriesToDropUnusedColumns | ||||
|     @Transaction | ||||
|     @Query("SELECT *, MIN(" + JOIN_INDEX + ")" | ||||
|             + " FROM " + STREAM_TABLE + " INNER JOIN" | ||||
|             + " (SELECT " + JOIN_STREAM_ID + "," + JOIN_INDEX | ||||
|             + " FROM " + PLAYLIST_STREAM_JOIN_TABLE | ||||
|             + " WHERE " + JOIN_PLAYLIST_ID + " = :playlistId)" | ||||
|             + " ON " + STREAM_ID + " = " + JOIN_STREAM_ID | ||||
|             + " LEFT JOIN " | ||||
|             + "(SELECT " + JOIN_STREAM_ID + " AS " + JOIN_STREAM_ID_ALIAS + ", " | ||||
|             + STREAM_PROGRESS_MILLIS | ||||
|             + " FROM " + STREAM_STATE_TABLE + " )" | ||||
|             + " ON " + STREAM_ID + " = " + JOIN_STREAM_ID_ALIAS | ||||
|             + " GROUP BY " + STREAM_ID | ||||
|             + " ORDER BY MIN(" + JOIN_INDEX + ") ASC") | ||||
|     Flowable<List<PlaylistStreamEntry>> getStreamsWithoutDuplicates(long playlistId); | ||||
|  | ||||
|     @Transaction | ||||
|     @Query("SELECT " + PLAYLIST_TABLE + "." + PLAYLIST_ID + ", " + PLAYLIST_NAME + ", " | ||||
|             + PLAYLIST_THUMBNAIL_PERMANENT + ", " + PLAYLIST_THUMBNAIL_STREAM_ID + ", " | ||||
|             + PLAYLIST_DISPLAY_INDEX + ", " | ||||
|  | ||||
|             + " CASE WHEN " + PLAYLIST_THUMBNAIL_STREAM_ID + " = " | ||||
|             + PlaylistEntity.DEFAULT_THUMBNAIL_ID + " THEN " + "'" + DEFAULT_THUMBNAIL + "'" | ||||
|             + " ELSE (SELECT " + STREAM_THUMBNAIL_URL | ||||
|             + " FROM " + STREAM_TABLE | ||||
|             + " WHERE " + STREAM_TABLE + "." + STREAM_ID + " = " + PLAYLIST_THUMBNAIL_STREAM_ID | ||||
|             + " ) END AS " + PLAYLIST_THUMBNAIL_URL + ", " | ||||
|  | ||||
|             + "COALESCE(COUNT(" + JOIN_PLAYLIST_ID + "), 0) AS " + PLAYLIST_STREAM_COUNT + ", " | ||||
|             + "COALESCE(SUM(" + STREAM_URL + " = :streamUrl), 0) AS " | ||||
|                 + PLAYLIST_TIMES_STREAM_IS_CONTAINED | ||||
|  | ||||
|             + " FROM " + PLAYLIST_TABLE | ||||
|             + " LEFT JOIN " + PLAYLIST_STREAM_JOIN_TABLE | ||||
|             + " ON " + PLAYLIST_TABLE + "." + PLAYLIST_ID + " = " + JOIN_PLAYLIST_ID | ||||
|  | ||||
|             + " LEFT JOIN " + STREAM_TABLE | ||||
|             + " ON " + STREAM_TABLE + "." + STREAM_ID + " = " + JOIN_STREAM_ID | ||||
|             + " AND :streamUrl = :streamUrl" | ||||
|  | ||||
|             + " GROUP BY " + JOIN_PLAYLIST_ID | ||||
|             + " ORDER BY " + PLAYLIST_NAME + " COLLATE NOCASE ASC") | ||||
|     Flowable<List<PlaylistDuplicatesEntry>> getPlaylistDuplicatesMetadata(String streamUrl); | ||||
| } | ||||
|   | ||||
| @@ -9,16 +9,24 @@ import androidx.room.PrimaryKey; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_NAME; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistEntity.PLAYLIST_TABLE; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.database.playlist.PlaylistMetadataEntry; | ||||
|  | ||||
| @Entity(tableName = PLAYLIST_TABLE, | ||||
|         indices = {@Index(value = {PLAYLIST_NAME})}) | ||||
| public class PlaylistEntity { | ||||
|  | ||||
|     public static final String DEFAULT_THUMBNAIL = "drawable://" | ||||
|             + R.drawable.placeholder_thumbnail_playlist; | ||||
|     public static final long DEFAULT_THUMBNAIL_ID = -1; | ||||
|  | ||||
|     public static final String PLAYLIST_TABLE = "playlists"; | ||||
|     public static final String PLAYLIST_ID = "uid"; | ||||
|     public static final String PLAYLIST_NAME = "name"; | ||||
|     public static final String PLAYLIST_THUMBNAIL_URL = "thumbnail_url"; | ||||
|     public static final String PLAYLIST_DISPLAY_INDEX = "display_index"; | ||||
|     public static final String PLAYLIST_THUMBNAIL_PERMANENT = "is_thumbnail_permanent"; | ||||
|     public static final String PLAYLIST_THUMBNAIL_STREAM_ID = "thumbnail_stream_id"; | ||||
|  | ||||
|     @PrimaryKey(autoGenerate = true) | ||||
|     @ColumnInfo(name = PLAYLIST_ID) | ||||
| @@ -27,15 +35,20 @@ public class PlaylistEntity { | ||||
|     @ColumnInfo(name = PLAYLIST_NAME) | ||||
|     private String name; | ||||
|  | ||||
|     @ColumnInfo(name = PLAYLIST_THUMBNAIL_URL) | ||||
|     private String thumbnailUrl; | ||||
|     @ColumnInfo(name = PLAYLIST_THUMBNAIL_PERMANENT) | ||||
|     private boolean isThumbnailPermanent; | ||||
|  | ||||
|     @ColumnInfo(name = PLAYLIST_THUMBNAIL_STREAM_ID) | ||||
|     private long thumbnailStreamId; | ||||
|  | ||||
|     @ColumnInfo(name = PLAYLIST_DISPLAY_INDEX) | ||||
|     private long displayIndex; | ||||
|  | ||||
|     public PlaylistEntity(final String name, final String thumbnailUrl, final long displayIndex) { | ||||
|     public PlaylistEntity(final String name, final boolean isThumbnailPermanent, | ||||
|                           final long thumbnailStreamId, final long displayIndex) { | ||||
|         this.name = name; | ||||
|         this.thumbnailUrl = thumbnailUrl; | ||||
|         this.isThumbnailPermanent = isThumbnailPermanent; | ||||
|         this.thumbnailStreamId = thumbnailStreamId; | ||||
|         this.displayIndex = displayIndex; | ||||
|     } | ||||
|  | ||||
| @@ -43,7 +56,8 @@ public class PlaylistEntity { | ||||
|     public PlaylistEntity(final PlaylistMetadataEntry item) { | ||||
|         this.uid = item.getUid(); | ||||
|         this.name = item.name; | ||||
|         this.thumbnailUrl = item.thumbnailUrl; | ||||
|         this.isThumbnailPermanent = item.isThumbnailPermanent(); | ||||
|         this.thumbnailStreamId = item.getThumbnailStreamId(); | ||||
|         this.displayIndex = item.getDisplayIndex(); | ||||
|     } | ||||
|  | ||||
| @@ -63,12 +77,20 @@ public class PlaylistEntity { | ||||
|         this.name = name; | ||||
|     } | ||||
|  | ||||
|     public String getThumbnailUrl() { | ||||
|         return thumbnailUrl; | ||||
|     public long getThumbnailStreamId() { | ||||
|         return thumbnailStreamId; | ||||
|     } | ||||
|  | ||||
|     public void setThumbnailUrl(final String thumbnailUrl) { | ||||
|         this.thumbnailUrl = thumbnailUrl; | ||||
|     public void setThumbnailStreamId(final long thumbnailStreamId) { | ||||
|         this.thumbnailStreamId = thumbnailStreamId; | ||||
|     } | ||||
|  | ||||
|     public boolean getIsThumbnailPermanent() { | ||||
|         return isThumbnailPermanent; | ||||
|     } | ||||
|  | ||||
|     public void setIsThumbnailPermanent(final boolean isThumbnailSet) { | ||||
|         this.isThumbnailPermanent = isThumbnailSet; | ||||
|     } | ||||
|  | ||||
|     public long getDisplayIndex() { | ||||
|   | ||||
| @@ -11,6 +11,7 @@ import androidx.room.PrimaryKey; | ||||
| import org.schabi.newpipe.database.playlist.PlaylistLocalItem; | ||||
| import org.schabi.newpipe.extractor.playlist.PlaylistInfo; | ||||
| import org.schabi.newpipe.util.Constants; | ||||
| import org.schabi.newpipe.util.image.ImageStrategy; | ||||
|  | ||||
| import static org.schabi.newpipe.database.LocalItem.LocalItemType.PLAYLIST_REMOTE_ITEM; | ||||
| import static org.schabi.newpipe.database.playlist.model.PlaylistRemoteEntity.REMOTE_PLAYLIST_NAME; | ||||
| @@ -86,8 +87,9 @@ public class PlaylistRemoteEntity implements PlaylistLocalItem { | ||||
|     @Ignore | ||||
|     public PlaylistRemoteEntity(final PlaylistInfo info) { | ||||
|         this(info.getServiceId(), info.getName(), info.getUrl(), | ||||
|                 info.getThumbnailUrl() == null | ||||
|                         ? info.getUploaderAvatarUrl() : info.getThumbnailUrl(), | ||||
|                 // use uploader avatar when no thumbnail is available | ||||
|                 ImageStrategy.imageListToDbUrl(info.getThumbnails().isEmpty() | ||||
|                         ? info.getUploaderAvatars() : info.getThumbnails()), | ||||
|                 info.getUploaderName(), info.getStreamCount()); | ||||
|     } | ||||
|  | ||||
| @@ -101,7 +103,10 @@ public class PlaylistRemoteEntity implements PlaylistLocalItem { | ||||
|                 && getStreamCount() == info.getStreamCount() | ||||
|                 && TextUtils.equals(getName(), info.getName()) | ||||
|                 && TextUtils.equals(getUrl(), info.getUrl()) | ||||
|                 && TextUtils.equals(getThumbnailUrl(), info.getThumbnailUrl()) | ||||
|                 // we want to update the local playlist data even when either the remote thumbnail | ||||
|                 // URL changes, or the preferred image quality setting is changed by the user | ||||
|                 && TextUtils.equals(getThumbnailUrl(), | ||||
|                 ImageStrategy.imageListToDbUrl(info.getThumbnails())) | ||||
|                 && TextUtils.equals(getUploader(), info.getUploaderName()); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import org.schabi.newpipe.database.history.model.StreamHistoryEntity | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity | ||||
| import org.schabi.newpipe.database.stream.model.StreamStateEntity.STREAM_PROGRESS_MILLIS | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem | ||||
| import org.schabi.newpipe.util.image.ImageStrategy | ||||
| import java.time.OffsetDateTime | ||||
|  | ||||
| class StreamStatisticsEntry( | ||||
| @@ -30,7 +31,7 @@ class StreamStatisticsEntry( | ||||
|         item.duration = streamEntity.duration | ||||
|         item.uploaderName = streamEntity.uploader | ||||
|         item.uploaderUrl = streamEntity.uploaderUrl | ||||
|         item.thumbnailUrl = streamEntity.thumbnailUrl | ||||
|         item.thumbnails = ImageStrategy.dbUrlToImageList(streamEntity.thumbnailUrl) | ||||
|  | ||||
|         return item | ||||
|     } | ||||
|   | ||||
| @@ -12,8 +12,7 @@ import org.schabi.newpipe.database.BasicDAO | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity.Companion.STREAM_ID | ||||
| import org.schabi.newpipe.extractor.stream.StreamType | ||||
| import org.schabi.newpipe.extractor.stream.StreamType.AUDIO_LIVE_STREAM | ||||
| import org.schabi.newpipe.extractor.stream.StreamType.LIVE_STREAM | ||||
| import org.schabi.newpipe.util.StreamTypeUtil | ||||
| import java.time.OffsetDateTime | ||||
|  | ||||
| @Dao | ||||
| @@ -91,8 +90,7 @@ abstract class StreamDAO : BasicDAO<StreamEntity> { | ||||
|             ?: throw IllegalStateException("Stream cannot be null just after insertion.") | ||||
|         newerStream.uid = existentMinimalStream.uid | ||||
|  | ||||
|         val isNewerStreamLive = newerStream.streamType == AUDIO_LIVE_STREAM || newerStream.streamType == LIVE_STREAM | ||||
|         if (!isNewerStreamLive) { | ||||
|         if (!StreamTypeUtil.isLiveStream(newerStream.streamType)) { | ||||
|  | ||||
|             // Use the existent upload date if the newer stream does not have a better precision | ||||
|             // (i.e. is an approximation). This is done to prevent unnecessary changes. | ||||
|   | ||||
| @@ -13,6 +13,7 @@ import org.schabi.newpipe.extractor.stream.StreamInfo | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem | ||||
| import org.schabi.newpipe.extractor.stream.StreamType | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueueItem | ||||
| import org.schabi.newpipe.util.image.ImageStrategy | ||||
| import java.io.Serializable | ||||
| import java.time.OffsetDateTime | ||||
|  | ||||
| @@ -67,7 +68,8 @@ data class StreamEntity( | ||||
|     constructor(item: StreamInfoItem) : this( | ||||
|         serviceId = item.serviceId, url = item.url, title = item.name, | ||||
|         streamType = item.streamType, duration = item.duration, uploader = item.uploaderName, | ||||
|         uploaderUrl = item.uploaderUrl, thumbnailUrl = item.thumbnailUrl, viewCount = item.viewCount, | ||||
|         uploaderUrl = item.uploaderUrl, | ||||
|         thumbnailUrl = ImageStrategy.imageListToDbUrl(item.thumbnails), viewCount = item.viewCount, | ||||
|         textualUploadDate = item.textualUploadDate, uploadDate = item.uploadDate?.offsetDateTime(), | ||||
|         isUploadDateApproximation = item.uploadDate?.isApproximation | ||||
|     ) | ||||
| @@ -76,7 +78,8 @@ data class StreamEntity( | ||||
|     constructor(info: StreamInfo) : this( | ||||
|         serviceId = info.serviceId, url = info.url, title = info.name, | ||||
|         streamType = info.streamType, duration = info.duration, uploader = info.uploaderName, | ||||
|         uploaderUrl = info.uploaderUrl, thumbnailUrl = info.thumbnailUrl, viewCount = info.viewCount, | ||||
|         uploaderUrl = info.uploaderUrl, | ||||
|         thumbnailUrl = ImageStrategy.imageListToDbUrl(info.thumbnails), viewCount = info.viewCount, | ||||
|         textualUploadDate = info.textualUploadDate, uploadDate = info.uploadDate?.offsetDateTime(), | ||||
|         isUploadDateApproximation = info.uploadDate?.isApproximation | ||||
|     ) | ||||
| @@ -85,7 +88,8 @@ data class StreamEntity( | ||||
|     constructor(item: PlayQueueItem) : this( | ||||
|         serviceId = item.serviceId, url = item.url, title = item.title, | ||||
|         streamType = item.streamType, duration = item.duration, uploader = item.uploader, | ||||
|         uploaderUrl = item.uploaderUrl, thumbnailUrl = item.thumbnailUrl | ||||
|         uploaderUrl = item.uploaderUrl, | ||||
|         thumbnailUrl = ImageStrategy.imageListToDbUrl(item.thumbnails) | ||||
|     ) | ||||
|  | ||||
|     fun toStreamInfoItem(): StreamInfoItem { | ||||
| @@ -93,7 +97,7 @@ data class StreamEntity( | ||||
|         item.duration = duration | ||||
|         item.uploaderName = uploader | ||||
|         item.uploaderUrl = uploaderUrl | ||||
|         item.thumbnailUrl = thumbnailUrl | ||||
|         item.thumbnails = ImageStrategy.dbUrlToImageList(thumbnailUrl) | ||||
|  | ||||
|         if (viewCount != null) item.viewCount = viewCount as Long | ||||
|         item.textualUploadDate = textualUploadDate | ||||
|   | ||||
| @@ -30,7 +30,7 @@ public class StreamStateEntity { | ||||
|     /** | ||||
|      * Playback state will not be saved, if playback time is less than this threshold (5000ms = 5s). | ||||
|      */ | ||||
|     private static final long PLAYBACK_SAVE_THRESHOLD_START_MILLISECONDS = 5000; | ||||
|     public static final long PLAYBACK_SAVE_THRESHOLD_START_MILLISECONDS = 5000; | ||||
|  | ||||
|     /** | ||||
|      * Stream will be considered finished if the playback time left exceeds this threshold | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import androidx.room.PrimaryKey; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfo; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfoItem; | ||||
| import org.schabi.newpipe.util.Constants; | ||||
| import org.schabi.newpipe.util.image.ImageStrategy; | ||||
|  | ||||
| import static org.schabi.newpipe.database.subscription.SubscriptionEntity.SUBSCRIPTION_SERVICE_ID; | ||||
| import static org.schabi.newpipe.database.subscription.SubscriptionEntity.SUBSCRIPTION_TABLE; | ||||
| @@ -57,8 +58,8 @@ public class SubscriptionEntity { | ||||
|         final SubscriptionEntity result = new SubscriptionEntity(); | ||||
|         result.setServiceId(info.getServiceId()); | ||||
|         result.setUrl(info.getUrl()); | ||||
|         result.setData(info.getName(), info.getAvatarUrl(), info.getDescription(), | ||||
|                 info.getSubscriberCount()); | ||||
|         result.setData(info.getName(), ImageStrategy.imageListToDbUrl(info.getAvatars()), | ||||
|                 info.getDescription(), info.getSubscriberCount()); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
| @@ -138,7 +139,7 @@ public class SubscriptionEntity { | ||||
|     @Ignore | ||||
|     public ChannelInfoItem toChannelInfoItem() { | ||||
|         final ChannelInfoItem item = new ChannelInfoItem(getServiceId(), getUrl(), getName()); | ||||
|         item.setThumbnailUrl(getAvatarUrl()); | ||||
|         item.setThumbnails(ImageStrategy.dbUrlToImageList(getAvatarUrl())); | ||||
|         item.setSubscriberCount(getSubscriberCount()); | ||||
|         item.setDescription(getDescription()); | ||||
|         return item; | ||||
|   | ||||
| @@ -1,10 +1,12 @@ | ||||
| package org.schabi.newpipe.download; | ||||
|  | ||||
| import static org.schabi.newpipe.extractor.stream.DeliveryMethod.PROGRESSIVE_HTTP; | ||||
| import static org.schabi.newpipe.util.ListHelper.getStreamsOfSpecifiedDelivery; | ||||
| import static org.schabi.newpipe.util.Localization.assureCorrectAppLanguage; | ||||
|  | ||||
| import android.app.Activity; | ||||
| import android.content.ComponentName; | ||||
| import android.content.Context; | ||||
| import android.content.DialogInterface; | ||||
| import android.content.DialogInterface.OnDismissListener; | ||||
| import android.content.Intent; | ||||
| import android.content.ServiceConnection; | ||||
| import android.content.SharedPreferences; | ||||
| @@ -12,8 +14,8 @@ import android.net.Uri; | ||||
| import android.os.Bundle; | ||||
| import android.os.Environment; | ||||
| import android.os.IBinder; | ||||
| import android.provider.Settings; | ||||
| import android.util.Log; | ||||
| import android.util.SparseArray; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| @@ -32,6 +34,7 @@ import androidx.annotation.StringRes; | ||||
| import androidx.appcompat.app.AlertDialog; | ||||
| import androidx.appcompat.view.menu.ActionMenuItemView; | ||||
| import androidx.appcompat.widget.Toolbar; | ||||
| import androidx.collection.SparseArrayCompat; | ||||
| import androidx.documentfile.provider.DocumentFile; | ||||
| import androidx.fragment.app.DialogFragment; | ||||
| import androidx.preference.PreferenceManager; | ||||
| @@ -63,7 +66,9 @@ import org.schabi.newpipe.util.PermissionHelper; | ||||
| import org.schabi.newpipe.util.SecondaryStreamHelper; | ||||
| import org.schabi.newpipe.util.SimpleOnSeekBarChangeListener; | ||||
| import org.schabi.newpipe.util.StreamItemAdapter; | ||||
| import org.schabi.newpipe.util.StreamItemAdapter.StreamSizeWrapper; | ||||
| import org.schabi.newpipe.util.StreamItemAdapter.StreamInfoWrapper; | ||||
| import org.schabi.newpipe.util.AudioTrackAdapter; | ||||
| import org.schabi.newpipe.util.AudioTrackAdapter.AudioTracksWrapper; | ||||
| import org.schabi.newpipe.util.ThemeHelper; | ||||
|  | ||||
| import java.io.File; | ||||
| @@ -71,6 +76,8 @@ import java.io.IOException; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.Locale; | ||||
| import java.util.Objects; | ||||
| import java.util.Optional; | ||||
|  | ||||
| import icepick.Icepick; | ||||
| import icepick.State; | ||||
| @@ -82,8 +89,6 @@ import us.shandian.giga.service.DownloadManagerService; | ||||
| import us.shandian.giga.service.DownloadManagerService.DownloadManagerBinder; | ||||
| import us.shandian.giga.service.MissionState; | ||||
|  | ||||
| import static org.schabi.newpipe.util.Localization.assureCorrectAppLanguage; | ||||
|  | ||||
| public class DownloadDialog extends DialogFragment | ||||
|         implements RadioGroup.OnCheckedChangeListener, AdapterView.OnItemSelectedListener { | ||||
|     private static final String TAG = "DialogFragment"; | ||||
| @@ -92,28 +97,28 @@ public class DownloadDialog extends DialogFragment | ||||
|     @State | ||||
|     StreamInfo currentInfo; | ||||
|     @State | ||||
|     StreamSizeWrapper<AudioStream> wrappedAudioStreams = StreamSizeWrapper.empty(); | ||||
|     StreamInfoWrapper<VideoStream> wrappedVideoStreams; | ||||
|     @State | ||||
|     StreamSizeWrapper<VideoStream> wrappedVideoStreams = StreamSizeWrapper.empty(); | ||||
|     StreamInfoWrapper<SubtitlesStream> wrappedSubtitleStreams; | ||||
|     @State | ||||
|     StreamSizeWrapper<SubtitlesStream> wrappedSubtitleStreams = StreamSizeWrapper.empty(); | ||||
|     AudioTracksWrapper wrappedAudioTracks; | ||||
|     @State | ||||
|     int selectedVideoIndex = 0; | ||||
|     int selectedAudioTrackIndex; | ||||
|     @State | ||||
|     int selectedAudioIndex = 0; | ||||
|     int selectedVideoIndex; // set in the constructor | ||||
|     @State | ||||
|     int selectedSubtitleIndex = 0; | ||||
|  | ||||
|     @Nullable | ||||
|     private OnDismissListener onDismissListener = null; | ||||
|     int selectedAudioIndex = 0; // default to the first item | ||||
|     @State | ||||
|     int selectedSubtitleIndex = 0; // default to the first item | ||||
|  | ||||
|     private StoredDirectoryHelper mainStorageAudio = null; | ||||
|     private StoredDirectoryHelper mainStorageVideo = null; | ||||
|     private DownloadManager downloadManager = null; | ||||
|     private ActionMenuItemView okButton = null; | ||||
|     private Context context; | ||||
|     private Context context = null; | ||||
|     private boolean askForSavePath; | ||||
|  | ||||
|     private AudioTrackAdapter audioTrackAdapter; | ||||
|     private StreamItemAdapter<AudioStream, Stream> audioStreamsAdapter; | ||||
|     private StreamItemAdapter<VideoStream, AudioStream> videoStreamsAdapter; | ||||
|     private StreamItemAdapter<SubtitlesStream, Stream> subtitleStreamsAdapter; | ||||
| @@ -138,81 +143,53 @@ public class DownloadDialog extends DialogFragment | ||||
|             registerForActivityResult( | ||||
|                     new StartActivityForResult(), this::requestDownloadPickVideoFolderResult); | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Instance creation | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     public static DownloadDialog newInstance(final StreamInfo info) { | ||||
|         final DownloadDialog dialog = new DownloadDialog(); | ||||
|         dialog.setInfo(info); | ||||
|         return dialog; | ||||
|     public DownloadDialog() { | ||||
|         // Just an empty default no-arg ctor to keep Fragment.instantiate() happy | ||||
|         // otherwise InstantiationException will be thrown when fragment is recreated | ||||
|         // TODO: Maybe use a custom FragmentFactory instead? | ||||
|     } | ||||
|  | ||||
|     public static DownloadDialog newInstance(final Context context, final StreamInfo info) { | ||||
|         final ArrayList<VideoStream> streamsList = new ArrayList<>(ListHelper | ||||
|                 .getSortedStreamVideosList(context, info.getVideoStreams(), | ||||
|                         info.getVideoOnlyStreams(), false, false)); | ||||
|         final int selectedStreamIndex = ListHelper.getDefaultResolutionIndex(context, streamsList); | ||||
|  | ||||
|         final DownloadDialog instance = newInstance(info); | ||||
|         instance.setVideoStreams(streamsList); | ||||
|         instance.setSelectedVideoStream(selectedStreamIndex); | ||||
|         instance.setAudioStreams(info.getAudioStreams()); | ||||
|         instance.setSubtitleStreams(info.getSubtitles()); | ||||
|  | ||||
|         return instance; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Setters | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     private void setInfo(final StreamInfo info) { | ||||
|     /** | ||||
|      * Create a new download dialog with the video, audio and subtitle streams from the provided | ||||
|      * stream info. Video streams and video-only streams will be put into a single list menu, | ||||
|      * sorted according to their resolution and the default video resolution will be selected. | ||||
|      * | ||||
|      * @param context the context to use just to obtain preferences and strings (will not be stored) | ||||
|      * @param info    the info from which to obtain downloadable streams and other info (e.g. title) | ||||
|      */ | ||||
|     public DownloadDialog(@NonNull final Context context, @NonNull final StreamInfo info) { | ||||
|         this.currentInfo = info; | ||||
|  | ||||
|         final List<AudioStream> audioStreams = | ||||
|                 getStreamsOfSpecifiedDelivery(info.getAudioStreams(), PROGRESSIVE_HTTP); | ||||
|         final List<List<AudioStream>> groupedAudioStreams = | ||||
|                 ListHelper.getGroupedAudioStreams(context, audioStreams); | ||||
|         this.wrappedAudioTracks = new AudioTracksWrapper(groupedAudioStreams, context); | ||||
|         this.selectedAudioTrackIndex = | ||||
|                 ListHelper.getDefaultAudioTrackGroup(context, groupedAudioStreams); | ||||
|  | ||||
|         // TODO: Adapt this code when the downloader support other types of stream deliveries | ||||
|         final List<VideoStream> videoStreams = ListHelper.getSortedStreamVideosList( | ||||
|                 context, | ||||
|                 getStreamsOfSpecifiedDelivery(info.getVideoStreams(), PROGRESSIVE_HTTP), | ||||
|                 getStreamsOfSpecifiedDelivery(info.getVideoOnlyStreams(), PROGRESSIVE_HTTP), | ||||
|                 false, | ||||
|                 // If there are multiple languages available, prefer streams without audio | ||||
|                 // to allow language selection | ||||
|                 wrappedAudioTracks.size() > 1 | ||||
|         ); | ||||
|  | ||||
|         this.wrappedVideoStreams = new StreamInfoWrapper<>(videoStreams, context); | ||||
|         this.wrappedSubtitleStreams = new StreamInfoWrapper<>( | ||||
|                 getStreamsOfSpecifiedDelivery(info.getSubtitles(), PROGRESSIVE_HTTP), context); | ||||
|  | ||||
|         this.selectedVideoIndex = ListHelper.getDefaultResolutionIndex(context, videoStreams); | ||||
|     } | ||||
|  | ||||
|     public void setAudioStreams(final List<AudioStream> audioStreams) { | ||||
|         setAudioStreams(new StreamSizeWrapper<>(audioStreams, getContext())); | ||||
|     } | ||||
|  | ||||
|     public void setAudioStreams(final StreamSizeWrapper<AudioStream> was) { | ||||
|         this.wrappedAudioStreams = was; | ||||
|     } | ||||
|  | ||||
|     public void setVideoStreams(final List<VideoStream> videoStreams) { | ||||
|         setVideoStreams(new StreamSizeWrapper<>(videoStreams, getContext())); | ||||
|     } | ||||
|  | ||||
|     public void setVideoStreams(final StreamSizeWrapper<VideoStream> wvs) { | ||||
|         this.wrappedVideoStreams = wvs; | ||||
|     } | ||||
|  | ||||
|     public void setSubtitleStreams(final List<SubtitlesStream> subtitleStreams) { | ||||
|         setSubtitleStreams(new StreamSizeWrapper<>(subtitleStreams, getContext())); | ||||
|     } | ||||
|  | ||||
|     public void setSubtitleStreams( | ||||
|             final StreamSizeWrapper<SubtitlesStream> wss) { | ||||
|         this.wrappedSubtitleStreams = wss; | ||||
|     } | ||||
|  | ||||
|     public void setSelectedVideoStream(final int svi) { | ||||
|         this.selectedVideoIndex = svi; | ||||
|     } | ||||
|  | ||||
|     public void setSelectedAudioStream(final int sai) { | ||||
|         this.selectedAudioIndex = sai; | ||||
|     } | ||||
|  | ||||
|     public void setSelectedSubtitleStream(final int ssi) { | ||||
|         this.selectedSubtitleIndex = ssi; | ||||
|     } | ||||
|  | ||||
|     public void setOnDismissListener(@Nullable final OnDismissListener onDismissListener) { | ||||
|         this.onDismissListener = onDismissListener; | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Android lifecycle | ||||
| @@ -232,35 +209,16 @@ public class DownloadDialog extends DialogFragment | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // context will remain null if dismiss() was called above, allowing to check whether the | ||||
|         // dialog is being dismissed in onViewCreated() | ||||
|         context = getContext(); | ||||
|  | ||||
|         setStyle(STYLE_NO_TITLE, ThemeHelper.getDialogTheme(context)); | ||||
|         Icepick.restoreInstanceState(this, savedInstanceState); | ||||
|  | ||||
|         final SparseArray<SecondaryStreamHelper<AudioStream>> secondaryStreams | ||||
|                 = new SparseArray<>(4); | ||||
|         final List<VideoStream> videoStreams = wrappedVideoStreams.getStreamsList(); | ||||
|  | ||||
|         for (int i = 0; i < videoStreams.size(); i++) { | ||||
|             if (!videoStreams.get(i).isVideoOnly()) { | ||||
|                 continue; | ||||
|             } | ||||
|             final AudioStream audioStream = SecondaryStreamHelper | ||||
|                     .getAudioStreamFor(wrappedAudioStreams.getStreamsList(), videoStreams.get(i)); | ||||
|  | ||||
|             if (audioStream != null) { | ||||
|                 secondaryStreams | ||||
|                         .append(i, new SecondaryStreamHelper<>(wrappedAudioStreams, audioStream)); | ||||
|             } else if (DEBUG) { | ||||
|                 Log.w(TAG, "No audio stream candidates for video format " | ||||
|                         + videoStreams.get(i).getFormat().name()); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this.videoStreamsAdapter = new StreamItemAdapter<>(context, wrappedVideoStreams, | ||||
|                 secondaryStreams); | ||||
|         this.audioStreamsAdapter = new StreamItemAdapter<>(context, wrappedAudioStreams); | ||||
|         this.subtitleStreamsAdapter = new StreamItemAdapter<>(context, wrappedSubtitleStreams); | ||||
|         this.audioTrackAdapter = new AudioTrackAdapter(wrappedAudioTracks); | ||||
|         this.subtitleStreamsAdapter = new StreamItemAdapter<>(wrappedSubtitleStreams); | ||||
|         updateSecondaryStreams(); | ||||
|  | ||||
|         final Intent intent = new Intent(context, DownloadManagerService.class); | ||||
|         context.startService(intent); | ||||
| @@ -287,8 +245,42 @@ public class DownloadDialog extends DialogFragment | ||||
|         }, Context.BIND_AUTO_CREATE); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Update the displayed video streams based on the selected audio track. | ||||
|      */ | ||||
|     private void updateSecondaryStreams() { | ||||
|         final StreamInfoWrapper<AudioStream> audioStreams = getWrappedAudioStreams(); | ||||
|         final var secondaryStreams = new SparseArrayCompat<SecondaryStreamHelper<AudioStream>>(4); | ||||
|         final List<VideoStream> videoStreams = wrappedVideoStreams.getStreamsList(); | ||||
|         wrappedVideoStreams.resetInfo(); | ||||
|  | ||||
|         for (int i = 0; i < videoStreams.size(); i++) { | ||||
|             if (!videoStreams.get(i).isVideoOnly()) { | ||||
|                 continue; | ||||
|             } | ||||
|             final AudioStream audioStream = SecondaryStreamHelper.getAudioStreamFor( | ||||
|                     context, audioStreams.getStreamsList(), videoStreams.get(i)); | ||||
|  | ||||
|             if (audioStream != null) { | ||||
|                 secondaryStreams.append(i, new SecondaryStreamHelper<>(audioStreams, audioStream)); | ||||
|             } else if (DEBUG) { | ||||
|                 final MediaFormat mediaFormat = videoStreams.get(i).getFormat(); | ||||
|                 if (mediaFormat != null) { | ||||
|                     Log.w(TAG, "No audio stream candidates for video format " | ||||
|                             + mediaFormat.name()); | ||||
|                 } else { | ||||
|                     Log.w(TAG, "No audio stream candidates for unknown video format"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this.videoStreamsAdapter = new StreamItemAdapter<>(wrappedVideoStreams, secondaryStreams); | ||||
|         this.audioStreamsAdapter = new StreamItemAdapter<>(audioStreams); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, final ViewGroup container, | ||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, | ||||
|                              final ViewGroup container, | ||||
|                              final Bundle savedInstanceState) { | ||||
|         if (DEBUG) { | ||||
|             Log.d(TAG, "onCreateView() called with: " | ||||
| @@ -299,19 +291,24 @@ public class DownloadDialog extends DialogFragment | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onViewCreated(@NonNull final View view, @Nullable final Bundle savedInstanceState) { | ||||
|     public void onViewCreated(@NonNull final View view, | ||||
|                               @Nullable final Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
|         dialogBinding = DownloadDialogBinding.bind(view); | ||||
|         if (context == null) { | ||||
|             return; // the dialog is being dismissed, see the call to dismiss() in onCreate() | ||||
|         } | ||||
|  | ||||
|         dialogBinding.fileName.setText(FilenameUtils.createFilename(getContext(), | ||||
|                 currentInfo.getName())); | ||||
|         selectedAudioIndex = ListHelper | ||||
|                 .getDefaultAudioFormat(getContext(), currentInfo.getAudioStreams()); | ||||
|         selectedAudioIndex = ListHelper.getDefaultAudioFormat(getContext(), | ||||
|                 getWrappedAudioStreams().getStreamsList()); | ||||
|  | ||||
|         selectedSubtitleIndex = getSubtitleIndexBy(subtitleStreamsAdapter.getAll()); | ||||
|  | ||||
|         dialogBinding.qualitySpinner.setOnItemSelectedListener(this); | ||||
|  | ||||
|         dialogBinding.audioStreamSpinner.setOnItemSelectedListener(this); | ||||
|         dialogBinding.audioTrackSpinner.setOnItemSelectedListener(this); | ||||
|         dialogBinding.videoAudioGroup.setOnCheckedChangeListener(this); | ||||
|  | ||||
|         initToolbar(dialogBinding.toolbarLayout.toolbar); | ||||
| @@ -324,7 +321,8 @@ public class DownloadDialog extends DialogFragment | ||||
|         dialogBinding.threads.setProgress(threads - 1); | ||||
|         dialogBinding.threads.setOnSeekBarChangeListener(new SimpleOnSeekBarChangeListener() { | ||||
|             @Override | ||||
|             public void onProgressChanged(@NonNull final SeekBar seekbar, final int progress, | ||||
|             public void onProgressChanged(@NonNull final SeekBar seekbar, | ||||
|                                           final int progress, | ||||
|                                           final boolean fromUser) { | ||||
|                 final int newProgress = progress + 1; | ||||
|                 prefs.edit().putInt(getString(R.string.default_download_threads), newProgress) | ||||
| @@ -359,14 +357,6 @@ public class DownloadDialog extends DialogFragment | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDismiss(@NonNull final DialogInterface dialog) { | ||||
|         super.onDismiss(dialog); | ||||
|         if (onDismissListener != null) { | ||||
|             onDismissListener.onDismiss(dialog); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroy() { | ||||
|         super.onDestroy(); | ||||
| @@ -392,7 +382,7 @@ public class DownloadDialog extends DialogFragment | ||||
|  | ||||
|     private void fetchStreamsSize() { | ||||
|         disposables.clear(); | ||||
|         disposables.add(StreamSizeWrapper.fetchSizeForWrapper(wrappedVideoStreams) | ||||
|         disposables.add(StreamInfoWrapper.fetchMoreInfoForWrapper(wrappedVideoStreams) | ||||
|                 .subscribe(result -> { | ||||
|                     if (dialogBinding.videoAudioGroup.getCheckedRadioButtonId() | ||||
|                             == R.id.video_button) { | ||||
| @@ -402,7 +392,7 @@ public class DownloadDialog extends DialogFragment | ||||
|                         new ErrorInfo(throwable, UserAction.DOWNLOAD_OPEN_DIALOG, | ||||
|                                 "Downloading video stream size", | ||||
|                                 currentInfo.getServiceId())))); | ||||
|         disposables.add(StreamSizeWrapper.fetchSizeForWrapper(wrappedAudioStreams) | ||||
|         disposables.add(StreamInfoWrapper.fetchMoreInfoForWrapper(getWrappedAudioStreams()) | ||||
|                 .subscribe(result -> { | ||||
|                     if (dialogBinding.videoAudioGroup.getCheckedRadioButtonId() | ||||
|                             == R.id.audio_button) { | ||||
| @@ -412,7 +402,7 @@ public class DownloadDialog extends DialogFragment | ||||
|                         new ErrorInfo(throwable, UserAction.DOWNLOAD_OPEN_DIALOG, | ||||
|                                 "Downloading audio stream size", | ||||
|                                 currentInfo.getServiceId())))); | ||||
|         disposables.add(StreamSizeWrapper.fetchSizeForWrapper(wrappedSubtitleStreams) | ||||
|         disposables.add(StreamInfoWrapper.fetchMoreInfoForWrapper(wrappedSubtitleStreams) | ||||
|                 .subscribe(result -> { | ||||
|                     if (dialogBinding.videoAudioGroup.getCheckedRadioButtonId() | ||||
|                             == R.id.subtitle_button) { | ||||
| @@ -424,14 +414,28 @@ public class DownloadDialog extends DialogFragment | ||||
|                                 currentInfo.getServiceId())))); | ||||
|     } | ||||
|  | ||||
|     private void setupAudioTrackSpinner() { | ||||
|         if (getContext() == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         dialogBinding.audioTrackSpinner.setAdapter(audioTrackAdapter); | ||||
|         dialogBinding.audioTrackSpinner.setSelection(selectedAudioTrackIndex); | ||||
|     } | ||||
|  | ||||
|     private void setupAudioSpinner() { | ||||
|         if (getContext() == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         dialogBinding.qualitySpinner.setAdapter(audioStreamsAdapter); | ||||
|         dialogBinding.qualitySpinner.setSelection(selectedAudioIndex); | ||||
|         dialogBinding.qualitySpinner.setVisibility(View.GONE); | ||||
|         setRadioButtonsState(true); | ||||
|         dialogBinding.audioStreamSpinner.setAdapter(audioStreamsAdapter); | ||||
|         dialogBinding.audioStreamSpinner.setSelection(selectedAudioIndex); | ||||
|         dialogBinding.audioStreamSpinner.setVisibility(View.VISIBLE); | ||||
|         dialogBinding.audioTrackSpinner.setVisibility( | ||||
|                 wrappedAudioTracks.size() > 1 ? View.VISIBLE : View.GONE); | ||||
|         dialogBinding.audioTrackPresentInVideoText.setVisibility(View.GONE); | ||||
|     } | ||||
|  | ||||
|     private void setupVideoSpinner() { | ||||
| @@ -441,7 +445,19 @@ public class DownloadDialog extends DialogFragment | ||||
|  | ||||
|         dialogBinding.qualitySpinner.setAdapter(videoStreamsAdapter); | ||||
|         dialogBinding.qualitySpinner.setSelection(selectedVideoIndex); | ||||
|         dialogBinding.qualitySpinner.setVisibility(View.VISIBLE); | ||||
|         setRadioButtonsState(true); | ||||
|         dialogBinding.audioStreamSpinner.setVisibility(View.GONE); | ||||
|         onVideoStreamSelected(); | ||||
|     } | ||||
|  | ||||
|     private void onVideoStreamSelected() { | ||||
|         final boolean isVideoOnly = videoStreamsAdapter.getItem(selectedVideoIndex).isVideoOnly(); | ||||
|  | ||||
|         dialogBinding.audioTrackSpinner.setVisibility( | ||||
|                 isVideoOnly && wrappedAudioTracks.size() > 1 ? View.VISIBLE : View.GONE); | ||||
|         dialogBinding.audioTrackPresentInVideoText.setVisibility( | ||||
|                 !isVideoOnly && wrappedAudioTracks.size() > 1 ? View.VISIBLE : View.GONE); | ||||
|     } | ||||
|  | ||||
|     private void setupSubtitleSpinner() { | ||||
| @@ -451,7 +467,11 @@ public class DownloadDialog extends DialogFragment | ||||
|  | ||||
|         dialogBinding.qualitySpinner.setAdapter(subtitleStreamsAdapter); | ||||
|         dialogBinding.qualitySpinner.setSelection(selectedSubtitleIndex); | ||||
|         dialogBinding.qualitySpinner.setVisibility(View.VISIBLE); | ||||
|         setRadioButtonsState(true); | ||||
|         dialogBinding.audioStreamSpinner.setVisibility(View.GONE); | ||||
|         dialogBinding.audioTrackSpinner.setVisibility(View.GONE); | ||||
|         dialogBinding.audioTrackPresentInVideoText.setVisibility(View.GONE); | ||||
|     } | ||||
|  | ||||
|  | ||||
| @@ -469,7 +489,7 @@ public class DownloadDialog extends DialogFragment | ||||
|                 result, getString(R.string.download_path_video_key), DownloadManager.TAG_VIDEO); | ||||
|     } | ||||
|  | ||||
|     private void requestDownloadSaveAsResult(final ActivityResult result) { | ||||
|     private void requestDownloadSaveAsResult(@NonNull final ActivityResult result) { | ||||
|         if (result.getResultCode() != Activity.RESULT_OK) { | ||||
|             return; | ||||
|         } | ||||
| @@ -486,8 +506,8 @@ public class DownloadDialog extends DialogFragment | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         final DocumentFile docFile | ||||
|                 = DocumentFile.fromSingleUri(context, result.getData().getData()); | ||||
|         final DocumentFile docFile = DocumentFile.fromSingleUri(context, | ||||
|                 result.getData().getData()); | ||||
|         if (docFile == null) { | ||||
|             showFailedDialog(R.string.general_error); | ||||
|             return; | ||||
| @@ -498,7 +518,7 @@ public class DownloadDialog extends DialogFragment | ||||
|                 docFile.getType()); | ||||
|     } | ||||
|  | ||||
|     private void requestDownloadPickFolderResult(final ActivityResult result, | ||||
|     private void requestDownloadPickFolderResult(@NonNull final ActivityResult result, | ||||
|                                                  final String key, | ||||
|                                                  final String tag) { | ||||
|         if (result.getResultCode() != Activity.RESULT_OK) { | ||||
| @@ -518,12 +538,11 @@ public class DownloadDialog extends DialogFragment | ||||
|                     StoredDirectoryHelper.PERMISSION_FLAGS); | ||||
|         } | ||||
|  | ||||
|         PreferenceManager.getDefaultSharedPreferences(context).edit() | ||||
|                 .putString(key, uri.toString()).apply(); | ||||
|         PreferenceManager.getDefaultSharedPreferences(context).edit().putString(key, | ||||
|                 uri.toString()).apply(); | ||||
|  | ||||
|         try { | ||||
|             final StoredDirectoryHelper mainStorage | ||||
|                     = new StoredDirectoryHelper(context, uri, tag); | ||||
|             final StoredDirectoryHelper mainStorage = new StoredDirectoryHelper(context, uri, tag); | ||||
|             checkSelectedDownload(mainStorage, mainStorage.findFile(filenameTmp), | ||||
|                     filenameTmp, mimeTmp); | ||||
|         } catch (final IOException e) { | ||||
| @@ -531,7 +550,6 @@ public class DownloadDialog extends DialogFragment | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Listeners | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
| @@ -561,23 +579,71 @@ public class DownloadDialog extends DialogFragment | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onItemSelected(final AdapterView<?> parent, final View view, | ||||
|                                final int position, final long id) { | ||||
|     public void onItemSelected(final AdapterView<?> parent, | ||||
|                                final View view, | ||||
|                                final int position, | ||||
|                                final long id) { | ||||
|         if (DEBUG) { | ||||
|             Log.d(TAG, "onItemSelected() called with: " | ||||
|                     + "parent = [" + parent + "], view = [" + view + "], " | ||||
|                     + "position = [" + position + "], id = [" + id + "]"); | ||||
|         } | ||||
|         switch (dialogBinding.videoAudioGroup.getCheckedRadioButtonId()) { | ||||
|             case R.id.audio_button: | ||||
|  | ||||
|         switch (parent.getId()) { | ||||
|             case R.id.quality_spinner: | ||||
|                 switch (dialogBinding.videoAudioGroup.getCheckedRadioButtonId()) { | ||||
|                     case R.id.video_button: | ||||
|                         selectedVideoIndex = position; | ||||
|                         onVideoStreamSelected(); | ||||
|                         break; | ||||
|                     case R.id.subtitle_button: | ||||
|                         selectedSubtitleIndex = position; | ||||
|                         break; | ||||
|                 } | ||||
|                 onItemSelectedSetFileName(); | ||||
|                 break; | ||||
|             case R.id.audio_track_spinner: | ||||
|                 final boolean trackChanged = selectedAudioTrackIndex != position; | ||||
|                 selectedAudioTrackIndex = position; | ||||
|                 if (trackChanged) { | ||||
|                     updateSecondaryStreams(); | ||||
|                     fetchStreamsSize(); | ||||
|                 } | ||||
|                 break; | ||||
|             case R.id.audio_stream_spinner: | ||||
|                 selectedAudioIndex = position; | ||||
|                 break; | ||||
|             case R.id.video_button: | ||||
|                 selectedVideoIndex = position; | ||||
|                 break; | ||||
|             case R.id.subtitle_button: | ||||
|                 selectedSubtitleIndex = position; | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void onItemSelectedSetFileName() { | ||||
|         final String fileName = FilenameUtils.createFilename(getContext(), currentInfo.getName()); | ||||
|         final String prevFileName = Optional.ofNullable(dialogBinding.fileName.getText()) | ||||
|                 .map(Object::toString) | ||||
|                 .orElse(""); | ||||
|  | ||||
|         if (prevFileName.isEmpty() | ||||
|                 || prevFileName.equals(fileName) | ||||
|                 || prevFileName.startsWith(getString(R.string.caption_file_name, fileName, ""))) { | ||||
|             // only update the file name field if it was not edited by the user | ||||
|  | ||||
|             switch (dialogBinding.videoAudioGroup.getCheckedRadioButtonId()) { | ||||
|                 case R.id.audio_button: | ||||
|                 case R.id.video_button: | ||||
|                     if (!prevFileName.equals(fileName)) { | ||||
|                         // since the user might have switched between audio and video, the correct | ||||
|                         // text might already be in place, so avoid resetting the cursor position | ||||
|                         dialogBinding.fileName.setText(fileName); | ||||
|                     } | ||||
|                     break; | ||||
|  | ||||
|                 case R.id.subtitle_button: | ||||
|                     final String setSubtitleLanguageCode = subtitleStreamsAdapter | ||||
|                             .getItem(selectedSubtitleIndex).getLanguageTag(); | ||||
|                     // this will reset the cursor position, which is bad UX, but it can't be avoided | ||||
|                     dialogBinding.fileName.setText(getString( | ||||
|                             R.string.caption_file_name, fileName, setSubtitleLanguageCode)); | ||||
|                     break; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -592,19 +658,22 @@ public class DownloadDialog extends DialogFragment | ||||
|  | ||||
|     protected void setupDownloadOptions() { | ||||
|         setRadioButtonsState(false); | ||||
|         setupAudioTrackSpinner(); | ||||
|  | ||||
|         final boolean isVideoStreamsAvailable = videoStreamsAdapter.getCount() > 0; | ||||
|         final boolean isAudioStreamsAvailable = audioStreamsAdapter.getCount() > 0; | ||||
|         final boolean isSubtitleStreamsAvailable = subtitleStreamsAdapter.getCount() > 0; | ||||
|  | ||||
|         dialogBinding.audioButton.setVisibility(isAudioStreamsAvailable ? View.VISIBLE : View.GONE); | ||||
|         dialogBinding.videoButton.setVisibility(isVideoStreamsAvailable ? View.VISIBLE : View.GONE); | ||||
|         dialogBinding.audioButton.setVisibility(isAudioStreamsAvailable ? View.VISIBLE | ||||
|                 : View.GONE); | ||||
|         dialogBinding.videoButton.setVisibility(isVideoStreamsAvailable ? View.VISIBLE | ||||
|                 : View.GONE); | ||||
|         dialogBinding.subtitleButton.setVisibility(isSubtitleStreamsAvailable | ||||
|                 ? View.VISIBLE : View.GONE); | ||||
|  | ||||
|         prefs = PreferenceManager.getDefaultSharedPreferences(requireContext()); | ||||
|         final String defaultMedia = prefs.getString(getString(R.string.last_used_download_type), | ||||
|                     getString(R.string.last_download_type_video_key)); | ||||
|                 getString(R.string.last_download_type_video_key)); | ||||
|  | ||||
|         if (isVideoStreamsAvailable | ||||
|                 && (defaultMedia.equals(getString(R.string.last_download_type_video_key)))) { | ||||
| @@ -640,7 +709,14 @@ public class DownloadDialog extends DialogFragment | ||||
|         dialogBinding.subtitleButton.setEnabled(enabled); | ||||
|     } | ||||
|  | ||||
|     private int getSubtitleIndexBy(final List<SubtitlesStream> streams) { | ||||
|     private StreamInfoWrapper<AudioStream> getWrappedAudioStreams() { | ||||
|         if (selectedAudioTrackIndex < 0 || selectedAudioTrackIndex > wrappedAudioTracks.size()) { | ||||
|             return StreamInfoWrapper.empty(); | ||||
|         } | ||||
|         return wrappedAudioTracks.getTracksList().get(selectedAudioTrackIndex); | ||||
|     } | ||||
|  | ||||
|     private int getSubtitleIndexBy(@NonNull final List<SubtitlesStream> streams) { | ||||
|         final Localization preferredLocalization = NewPipe.getPreferredLocalization(); | ||||
|  | ||||
|         int candidate = 0; | ||||
| @@ -666,35 +742,33 @@ public class DownloadDialog extends DialogFragment | ||||
|         return candidate; | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     private String getNameEditText() { | ||||
|         final String str = dialogBinding.fileName.getText().toString().trim(); | ||||
|         final String str = Objects.requireNonNull(dialogBinding.fileName.getText()).toString() | ||||
|                 .trim(); | ||||
|  | ||||
|         return FilenameUtils.createFilename(context, str.isEmpty() ? currentInfo.getName() : str); | ||||
|     } | ||||
|  | ||||
|     private void showFailedDialog(@StringRes final int msg) { | ||||
|         assureCorrectAppLanguage(getContext()); | ||||
|         assureCorrectAppLanguage(requireContext()); | ||||
|         new AlertDialog.Builder(context) | ||||
|                 .setTitle(R.string.general_error) | ||||
|                 .setMessage(msg) | ||||
|                 .setNegativeButton(getString(R.string.ok), null) | ||||
|                 .create() | ||||
|                 .show(); | ||||
|     } | ||||
|  | ||||
|     private void launchDirectoryPicker(final ActivityResultLauncher<Intent> launcher) { | ||||
|         NoFileManagerSafeGuard.launchSafe( | ||||
|                 launcher, | ||||
|                 StoredDirectoryHelper.getPicker(context), | ||||
|                 TAG, | ||||
|                 context | ||||
|         ); | ||||
|         NoFileManagerSafeGuard.launchSafe(launcher, StoredDirectoryHelper.getPicker(context), TAG, | ||||
|                 context); | ||||
|     } | ||||
|  | ||||
|     private void prepareSelectedDownload() { | ||||
|         final StoredDirectoryHelper mainStorage; | ||||
|         final MediaFormat format; | ||||
|         final String selectedMediaType; | ||||
|         final long size; | ||||
|  | ||||
|         // first, build the filename and get the output folder (if possible) | ||||
|         // later, run a very very very large file checking logic | ||||
| @@ -706,34 +780,45 @@ public class DownloadDialog extends DialogFragment | ||||
|                 selectedMediaType = getString(R.string.last_download_type_audio_key); | ||||
|                 mainStorage = mainStorageAudio; | ||||
|                 format = audioStreamsAdapter.getItem(selectedAudioIndex).getFormat(); | ||||
|                 size = getWrappedAudioStreams().getSizeInBytes(selectedAudioIndex); | ||||
|                 if (format == MediaFormat.WEBMA_OPUS) { | ||||
|                     mimeTmp = "audio/ogg"; | ||||
|                     filenameTmp += "opus"; | ||||
|                 } else { | ||||
|                 } else if (format != null) { | ||||
|                     mimeTmp = format.mimeType; | ||||
|                     filenameTmp += format.suffix; | ||||
|                     filenameTmp += format.getSuffix(); | ||||
|                 } | ||||
|                 break; | ||||
|             case R.id.video_button: | ||||
|                 selectedMediaType = getString(R.string.last_download_type_video_key); | ||||
|                 mainStorage = mainStorageVideo; | ||||
|                 format = videoStreamsAdapter.getItem(selectedVideoIndex).getFormat(); | ||||
|                 mimeTmp = format.mimeType; | ||||
|                 filenameTmp += format.suffix; | ||||
|                 size = wrappedVideoStreams.getSizeInBytes(selectedVideoIndex); | ||||
|                 if (format != null) { | ||||
|                     mimeTmp = format.mimeType; | ||||
|                     filenameTmp += format.getSuffix(); | ||||
|                 } | ||||
|                 break; | ||||
|             case R.id.subtitle_button: | ||||
|                 selectedMediaType = getString(R.string.last_download_type_subtitle_key); | ||||
|                 mainStorage = mainStorageVideo; // subtitle & video files go together | ||||
|                 format = subtitleStreamsAdapter.getItem(selectedSubtitleIndex).getFormat(); | ||||
|                 mimeTmp = format.mimeType; | ||||
|                 filenameTmp += (format == MediaFormat.TTML ? MediaFormat.SRT : format).suffix; | ||||
|                 size = wrappedSubtitleStreams.getSizeInBytes(selectedSubtitleIndex); | ||||
|                 if (format != null) { | ||||
|                     mimeTmp = format.mimeType; | ||||
|                 } | ||||
|  | ||||
|                 if (format == MediaFormat.TTML) { | ||||
|                     filenameTmp += MediaFormat.SRT.getSuffix(); | ||||
|                 } else if (format != null) { | ||||
|                     filenameTmp += format.getSuffix(); | ||||
|                 } | ||||
|                 break; | ||||
|             default: | ||||
|                 throw new RuntimeException("No stream selected"); | ||||
|         } | ||||
|  | ||||
|         if (!askForSavePath | ||||
|                 && (mainStorage == null | ||||
|         if (!askForSavePath && (mainStorage == null | ||||
|                 || mainStorage.isDirect() == NewPipeSettings.useStorageAccessFramework(context) | ||||
|                 || mainStorage.isInvalidSafStorage())) { | ||||
|             // Pick new download folder if one of: | ||||
| @@ -767,18 +852,32 @@ public class DownloadDialog extends DialogFragment | ||||
|                 initialPath = Uri.parse(initialSavePath.getAbsolutePath()); | ||||
|             } | ||||
|  | ||||
|             NoFileManagerSafeGuard.launchSafe( | ||||
|                     requestDownloadSaveAsLauncher, | ||||
|                     StoredFileHelper.getNewPicker(context, filenameTmp, mimeTmp, initialPath), | ||||
|                     TAG, | ||||
|                     context | ||||
|             ); | ||||
|             NoFileManagerSafeGuard.launchSafe(requestDownloadSaveAsLauncher, | ||||
|                     StoredFileHelper.getNewPicker(context, filenameTmp, mimeTmp, initialPath), TAG, | ||||
|                     context); | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Check for free memory space (for api 24 and up) | ||||
|         if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) { | ||||
|             final long freeSpace = mainStorage.getFreeMemory(); | ||||
|             if (freeSpace <= size) { | ||||
|                 Toast.makeText(context, getString(R. | ||||
|                         string.error_insufficient_storage), Toast.LENGTH_LONG).show(); | ||||
|                 // move the user to storage setting tab | ||||
|                 final Intent storageSettingsIntent = new Intent(Settings. | ||||
|                         ACTION_INTERNAL_STORAGE_SETTINGS); | ||||
|                 if (storageSettingsIntent.resolveActivity(context.getPackageManager()) != null) { | ||||
|                     startActivity(storageSettingsIntent); | ||||
|                 } | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // check for existing file with the same name | ||||
|         checkSelectedDownload(mainStorage, mainStorage.findFile(filenameTmp), filenameTmp, mimeTmp); | ||||
|         checkSelectedDownload(mainStorage, mainStorage.findFile(filenameTmp), filenameTmp, | ||||
|                 mimeTmp); | ||||
|  | ||||
|         // remember the last media type downloaded by the user | ||||
|         prefs.edit().putString(getString(R.string.last_used_download_type), selectedMediaType) | ||||
| @@ -786,7 +885,8 @@ public class DownloadDialog extends DialogFragment | ||||
|     } | ||||
|  | ||||
|     private void checkSelectedDownload(final StoredDirectoryHelper mainStorage, | ||||
|                                        final Uri targetFile, final String filename, | ||||
|                                        final Uri targetFile, | ||||
|                                        final String filename, | ||||
|                                        final String mime) { | ||||
|         StoredFileHelper storage; | ||||
|  | ||||
| @@ -888,7 +988,7 @@ public class DownloadDialog extends DialogFragment | ||||
|                     break; | ||||
|             } | ||||
|  | ||||
|             askDialog.create().show(); | ||||
|             askDialog.show(); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @@ -932,7 +1032,7 @@ public class DownloadDialog extends DialogFragment | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         askDialog.create().show(); | ||||
|         askDialog.show(); | ||||
|     } | ||||
|  | ||||
|     private void continueSelectedDownload(@NonNull final StoredFileHelper storage) { | ||||
| @@ -947,7 +1047,7 @@ public class DownloadDialog extends DialogFragment | ||||
|                 storage.truncate(); | ||||
|             } | ||||
|         } catch (final IOException e) { | ||||
|             Log.e(TAG, "failed to truncate the file: " + storage.getUri().toString(), e); | ||||
|             Log.e(TAG, "Failed to truncate the file: " + storage.getUri().toString(), e); | ||||
|             showFailedDialog(R.string.overwrite_failed); | ||||
|             return; | ||||
|         } | ||||
| @@ -957,7 +1057,7 @@ public class DownloadDialog extends DialogFragment | ||||
|         final char kind; | ||||
|         int threads = dialogBinding.threads.getProgress() + 1; | ||||
|         final String[] urls; | ||||
|         final MissionRecoveryInfo[] recoveryInfo; | ||||
|         final List<MissionRecoveryInfo> recoveryInfo; | ||||
|         String psName = null; | ||||
|         String[] psArgs = null; | ||||
|         long nearLength = 0; | ||||
| @@ -991,9 +1091,8 @@ public class DownloadDialog extends DialogFragment | ||||
|                         psName = Postprocessing.ALGORITHM_WEBM_MUXER; | ||||
|                     } | ||||
|  | ||||
|                     psArgs = null; | ||||
|                     final long videoSize = wrappedVideoStreams | ||||
|                             .getSizeInBytes((VideoStream) selectedStream); | ||||
|                     final long videoSize = wrappedVideoStreams.getSizeInBytes( | ||||
|                             (VideoStream) selectedStream); | ||||
|  | ||||
|                     // set nearLength, only, if both sizes are fetched or known. This probably | ||||
|                     // does not work on slow networks but is later updated in the downloader | ||||
| @@ -1009,7 +1108,7 @@ public class DownloadDialog extends DialogFragment | ||||
|  | ||||
|                 if (selectedStream.getFormat() == MediaFormat.TTML) { | ||||
|                     psName = Postprocessing.ALGORITHM_TTML_CONVERTER; | ||||
|                     psArgs = new String[]{ | ||||
|                     psArgs = new String[] { | ||||
|                             selectedStream.getFormat().getSuffix(), | ||||
|                             "false" // ignore empty frames | ||||
|                     }; | ||||
| @@ -1020,22 +1119,27 @@ public class DownloadDialog extends DialogFragment | ||||
|         } | ||||
|  | ||||
|         if (secondaryStream == null) { | ||||
|             urls = new String[]{ | ||||
|                     selectedStream.getUrl() | ||||
|             }; | ||||
|             recoveryInfo = new MissionRecoveryInfo[]{ | ||||
|                     new MissionRecoveryInfo(selectedStream) | ||||
|             urls = new String[] { | ||||
|                     selectedStream.getContent() | ||||
|             }; | ||||
|             recoveryInfo = List.of(new MissionRecoveryInfo(selectedStream)); | ||||
|         } else { | ||||
|             urls = new String[]{ | ||||
|                     selectedStream.getUrl(), secondaryStream.getUrl() | ||||
|             if (secondaryStream.getDeliveryMethod() != PROGRESSIVE_HTTP) { | ||||
|                 throw new IllegalArgumentException("Unsupported stream delivery format" | ||||
|                         + secondaryStream.getDeliveryMethod()); | ||||
|             } | ||||
|  | ||||
|             urls = new String[] { | ||||
|                     selectedStream.getContent(), secondaryStream.getContent() | ||||
|             }; | ||||
|             recoveryInfo = new MissionRecoveryInfo[]{new MissionRecoveryInfo(selectedStream), | ||||
|                     new MissionRecoveryInfo(secondaryStream)}; | ||||
|             recoveryInfo = List.of( | ||||
|                     new MissionRecoveryInfo(selectedStream), | ||||
|                     new MissionRecoveryInfo(secondaryStream) | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         DownloadManagerService.startMission(context, urls, storage, kind, threads, | ||||
|                 currentInfo.getUrl(), psName, psArgs, nearLength, recoveryInfo); | ||||
|                 currentInfo.getUrl(), psName, psArgs, nearLength, new ArrayList<>(recoveryInfo)); | ||||
|  | ||||
|         Toast.makeText(context, getString(R.string.download_has_started), | ||||
|                 Toast.LENGTH_SHORT).show(); | ||||
|   | ||||
| @@ -0,0 +1,87 @@ | ||||
| package org.schabi.newpipe.download; | ||||
|  | ||||
| import android.os.Bundle; | ||||
| import android.util.Log; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.annotation.StringRes; | ||||
| import androidx.appcompat.widget.Toolbar; | ||||
| import androidx.fragment.app.DialogFragment; | ||||
|  | ||||
| import org.schabi.newpipe.MainActivity; | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.databinding.DownloadLoadingDialogBinding; | ||||
|  | ||||
| /** | ||||
|  * This class contains a dialog which shows a loading indicator and has a customizable title. | ||||
|  */ | ||||
| public class LoadingDialog extends DialogFragment { | ||||
|     private static final String TAG = "LoadingDialog"; | ||||
|     private static final boolean DEBUG = MainActivity.DEBUG; | ||||
|     private DownloadLoadingDialogBinding dialogLoadingBinding; | ||||
|     private final @StringRes int title; | ||||
|  | ||||
|     /** | ||||
|      * Create a new LoadingDialog. | ||||
|      * | ||||
|      * <p> | ||||
|      *     The dialog contains a loading indicator and has a customizable title. | ||||
|      *     <br/> | ||||
|      *     Use {@code show()} to display the dialog to the user. | ||||
|      * </p> | ||||
|      * | ||||
|      * @param title an informative title shown in the dialog's toolbar | ||||
|      */ | ||||
|     public LoadingDialog(final @StringRes int title) { | ||||
|         this.title = title; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onCreate(@Nullable final Bundle savedInstanceState) { | ||||
|         super.onCreate(savedInstanceState); | ||||
|         if (DEBUG) { | ||||
|             Log.d(TAG, "onCreate() called with: " | ||||
|                     + "savedInstanceState = [" + savedInstanceState + "]"); | ||||
|         } | ||||
|         this.setCancelable(false); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public View onCreateView( | ||||
|             @NonNull final LayoutInflater inflater, | ||||
|             final ViewGroup container, | ||||
|             final Bundle savedInstanceState) { | ||||
|         if (DEBUG) { | ||||
|             Log.d(TAG, "onCreateView() called with: " | ||||
|                     + "inflater = [" + inflater + "], container = [" + container + "], " | ||||
|                     + "savedInstanceState = [" + savedInstanceState + "]"); | ||||
|         } | ||||
|         return inflater.inflate(R.layout.download_loading_dialog, container); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onViewCreated(@NonNull final View view, @Nullable final Bundle savedInstanceState) { | ||||
|         super.onViewCreated(view, savedInstanceState); | ||||
|         dialogLoadingBinding = DownloadLoadingDialogBinding.bind(view); | ||||
|         initToolbar(dialogLoadingBinding.toolbarLayout.toolbar); | ||||
|     } | ||||
|  | ||||
|     private void initToolbar(final Toolbar toolbar) { | ||||
|         if (DEBUG) { | ||||
|             Log.d(TAG, "initToolbar() called with: toolbar = [" + toolbar + "]"); | ||||
|         } | ||||
|         toolbar.setTitle(requireContext().getString(title)); | ||||
|         toolbar.setNavigationOnClickListener(v -> dismiss()); | ||||
|  | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroyView() { | ||||
|         dialogLoadingBinding = null; | ||||
|         super.onDestroyView(); | ||||
|     } | ||||
| } | ||||
| @@ -17,6 +17,7 @@ import androidx.annotation.Nullable; | ||||
| import androidx.appcompat.app.ActionBar; | ||||
| import androidx.appcompat.app.AlertDialog; | ||||
| import androidx.appcompat.app.AppCompatActivity; | ||||
| import androidx.core.content.IntentCompat; | ||||
|  | ||||
| import com.grack.nanojson.JsonWriter; | ||||
|  | ||||
| @@ -31,6 +32,7 @@ import org.schabi.newpipe.util.external_communication.ShareUtils; | ||||
| import java.time.LocalDateTime; | ||||
| import java.time.format.DateTimeFormatter; | ||||
| import java.util.Arrays; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| /* | ||||
|  * Created by Christian Schabesberger on 24.10.15. | ||||
| @@ -65,11 +67,11 @@ public class ErrorActivity extends AppCompatActivity { | ||||
|     public static final String ERROR_EMAIL_ADDRESS = "crashreport@newpipe.schabi.org"; | ||||
|     public static final String ERROR_EMAIL_SUBJECT = "Exception in "; | ||||
|  | ||||
|     public static final String ERROR_GITHUB_ISSUE_URL | ||||
|             = "https://github.com/TeamNewPipe/NewPipe/issues"; | ||||
|     public static final String ERROR_GITHUB_ISSUE_URL = | ||||
|             "https://github.com/TeamNewPipe/NewPipe/issues"; | ||||
|  | ||||
|     public static final DateTimeFormatter CURRENT_TIMESTAMP_FORMATTER | ||||
|             = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm"); | ||||
|     public static final DateTimeFormatter CURRENT_TIMESTAMP_FORMATTER = | ||||
|             DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm"); | ||||
|  | ||||
|  | ||||
|     private ErrorInfo errorInfo; | ||||
| @@ -104,7 +106,7 @@ public class ErrorActivity extends AppCompatActivity { | ||||
|             actionBar.setDisplayShowTitleEnabled(true); | ||||
|         } | ||||
|  | ||||
|         errorInfo = intent.getParcelableExtra(ERROR_INFO); | ||||
|         errorInfo = IntentCompat.getParcelableExtra(intent, ERROR_INFO, ErrorInfo.class); | ||||
|  | ||||
|         // important add guru meditation | ||||
|         addGuruMeditation(); | ||||
| @@ -159,7 +161,7 @@ public class ErrorActivity extends AppCompatActivity { | ||||
|                 .setMessage(R.string.start_accept_privacy_policy) | ||||
|                 .setCancelable(false) | ||||
|                 .setNeutralButton(R.string.read_privacy_policy, (dialog, which) -> | ||||
|                         ShareUtils.openUrlInBrowser(context, | ||||
|                         ShareUtils.openUrlInApp(context, | ||||
|                                 context.getString(R.string.privacy_policy_url))) | ||||
|                 .setPositiveButton(R.string.accept, (dialog, which) -> { | ||||
|                     if (action.equals("EMAIL")) { // send on email | ||||
| @@ -170,26 +172,19 @@ public class ErrorActivity extends AppCompatActivity { | ||||
|                                         + getString(R.string.app_name) + " " | ||||
|                                         + BuildConfig.VERSION_NAME) | ||||
|                                 .putExtra(Intent.EXTRA_TEXT, buildJson()); | ||||
|                         ShareUtils.openIntentInApp(context, i, true); | ||||
|                         ShareUtils.openIntentInApp(context, i); | ||||
|                     } else if (action.equals("GITHUB")) { // open the NewPipe issue page on GitHub | ||||
|                         ShareUtils.openUrlInBrowser(this, ERROR_GITHUB_ISSUE_URL, false); | ||||
|                         ShareUtils.openUrlInApp(this, ERROR_GITHUB_ISSUE_URL); | ||||
|                     } | ||||
|                 }) | ||||
|                 .setNegativeButton(R.string.decline, (dialog, which) -> { | ||||
|                     // do nothing | ||||
|                 }) | ||||
|                 .setNegativeButton(R.string.decline, null) | ||||
|                 .show(); | ||||
|     } | ||||
|  | ||||
|     private String formErrorText(final String[] el) { | ||||
|         final StringBuilder text = new StringBuilder(); | ||||
|         if (el != null) { | ||||
|             for (final String e : el) { | ||||
|                 text.append("-------------------------------------\n").append(e); | ||||
|             } | ||||
|         } | ||||
|         text.append("-------------------------------------"); | ||||
|         return text.toString(); | ||||
|         final String separator = "-------------------------------------"; | ||||
|         return Arrays.stream(el) | ||||
|                 .collect(Collectors.joining(separator + "\n", separator + "\n", separator)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -7,15 +7,12 @@ import kotlinx.parcelize.IgnoredOnParcel | ||||
| import kotlinx.parcelize.Parcelize | ||||
| import org.schabi.newpipe.R | ||||
| import org.schabi.newpipe.extractor.Info | ||||
| import org.schabi.newpipe.extractor.NewPipe | ||||
| import org.schabi.newpipe.extractor.exceptions.AccountTerminatedException | ||||
| import org.schabi.newpipe.extractor.exceptions.ContentNotAvailableException | ||||
| import org.schabi.newpipe.extractor.exceptions.ContentNotSupportedException | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException | ||||
| import org.schabi.newpipe.extractor.services.youtube.extractors.YoutubeStreamExtractor.DeobfuscateException | ||||
| import org.schabi.newpipe.ktx.isNetworkRelated | ||||
| import java.io.PrintWriter | ||||
| import java.io.StringWriter | ||||
| import org.schabi.newpipe.util.ServiceHelper | ||||
|  | ||||
| @Parcelize | ||||
| class ErrorInfo( | ||||
| @@ -65,7 +62,7 @@ class ErrorInfo( | ||||
|     constructor(throwable: Throwable, userAction: UserAction, request: String) : | ||||
|         this(throwable, userAction, SERVICE_NONE, request) | ||||
|     constructor(throwable: Throwable, userAction: UserAction, request: String, serviceId: Int) : | ||||
|         this(throwable, userAction, NewPipe.getNameOfService(serviceId), request) | ||||
|         this(throwable, userAction, ServiceHelper.getNameOfServiceById(serviceId), request) | ||||
|     constructor(throwable: Throwable, userAction: UserAction, request: String, info: Info?) : | ||||
|         this(throwable, userAction, getInfoServiceName(info), request) | ||||
|  | ||||
| @@ -73,29 +70,20 @@ class ErrorInfo( | ||||
|     constructor(throwable: List<Throwable>, userAction: UserAction, request: String) : | ||||
|         this(throwable, userAction, SERVICE_NONE, request) | ||||
|     constructor(throwable: List<Throwable>, userAction: UserAction, request: String, serviceId: Int) : | ||||
|         this(throwable, userAction, NewPipe.getNameOfService(serviceId), request) | ||||
|         this(throwable, userAction, ServiceHelper.getNameOfServiceById(serviceId), request) | ||||
|     constructor(throwable: List<Throwable>, userAction: UserAction, request: String, info: Info?) : | ||||
|         this(throwable, userAction, getInfoServiceName(info), request) | ||||
|  | ||||
|     companion object { | ||||
|         const val SERVICE_NONE = "none" | ||||
|  | ||||
|         private fun getStackTrace(throwable: Throwable): String { | ||||
|             StringWriter().use { stringWriter -> | ||||
|                 PrintWriter(stringWriter, true).use { printWriter -> | ||||
|                     throwable.printStackTrace(printWriter) | ||||
|                     return stringWriter.buffer.toString() | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         fun throwableToStringList(throwable: Throwable) = arrayOf(throwable.stackTraceToString()) | ||||
|  | ||||
|         fun throwableToStringList(throwable: Throwable) = arrayOf(getStackTrace(throwable)) | ||||
|  | ||||
|         fun throwableListToStringList(throwable: List<Throwable>) = | ||||
|             Array(throwable.size) { i -> getStackTrace(throwable[i]) } | ||||
|         fun throwableListToStringList(throwableList: List<Throwable>) = | ||||
|             throwableList.map { it.stackTraceToString() }.toTypedArray() | ||||
|  | ||||
|         private fun getInfoServiceName(info: Info?) = | ||||
|             if (info == null) SERVICE_NONE else NewPipe.getNameOfService(info.serviceId) | ||||
|             if (info == null) SERVICE_NONE else ServiceHelper.getNameOfServiceById(info.serviceId) | ||||
|  | ||||
|         @StringRes | ||||
|         private fun getMessageStringId( | ||||
| @@ -107,7 +95,6 @@ class ErrorInfo( | ||||
|                 throwable is ContentNotAvailableException -> R.string.content_not_available | ||||
|                 throwable != null && throwable.isNetworkRelated -> R.string.network_error | ||||
|                 throwable is ContentNotSupportedException -> R.string.content_not_supported | ||||
|                 throwable is DeobfuscateException -> R.string.youtube_signature_deobfuscation_error | ||||
|                 throwable is ExtractionException -> R.string.parsing_error | ||||
|                 throwable is ExoPlaybackException -> { | ||||
|                     when (throwable.type) { | ||||
|   | ||||
| @@ -6,7 +6,6 @@ import android.util.Log | ||||
| import android.view.View | ||||
| import android.widget.Button | ||||
| import android.widget.TextView | ||||
| import androidx.annotation.Nullable | ||||
| import androidx.annotation.StringRes | ||||
| import androidx.core.view.isVisible | ||||
| import androidx.fragment.app.Fragment | ||||
| @@ -15,7 +14,6 @@ import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers | ||||
| import io.reactivex.rxjava3.disposables.Disposable | ||||
| import org.schabi.newpipe.MainActivity | ||||
| import org.schabi.newpipe.R | ||||
| import org.schabi.newpipe.extractor.NewPipe | ||||
| import org.schabi.newpipe.extractor.exceptions.AccountTerminatedException | ||||
| import org.schabi.newpipe.extractor.exceptions.AgeRestrictedContentException | ||||
| import org.schabi.newpipe.extractor.exceptions.ContentNotAvailableException | ||||
| @@ -31,6 +29,7 @@ import org.schabi.newpipe.ktx.animate | ||||
| import org.schabi.newpipe.ktx.isInterruptedCaused | ||||
| import org.schabi.newpipe.ktx.isNetworkRelated | ||||
| import org.schabi.newpipe.util.ServiceHelper | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils | ||||
| import java.util.concurrent.TimeUnit | ||||
|  | ||||
| class ErrorPanelHelper( | ||||
| @@ -53,6 +52,8 @@ class ErrorPanelHelper( | ||||
|         errorPanelRoot.findViewById(R.id.error_action_button) | ||||
|     private val errorRetryButton: Button = | ||||
|         errorPanelRoot.findViewById(R.id.error_retry_button) | ||||
|     private val errorOpenInBrowserButton: Button = | ||||
|         errorPanelRoot.findViewById(R.id.error_open_in_browser) | ||||
|  | ||||
|     private var errorDisposable: Disposable? = null | ||||
|  | ||||
| @@ -70,6 +71,7 @@ class ErrorPanelHelper( | ||||
|         errorServiceExplanationTextView.isVisible = false | ||||
|         errorActionButton.isVisible = false | ||||
|         errorRetryButton.isVisible = false | ||||
|         errorOpenInBrowserButton.isVisible = false | ||||
|     } | ||||
|  | ||||
|     fun showError(errorInfo: ErrorInfo) { | ||||
| @@ -100,13 +102,14 @@ class ErrorPanelHelper( | ||||
|             } | ||||
|  | ||||
|             errorRetryButton.isVisible = true | ||||
|             showAndSetOpenInBrowserButtonAction(errorInfo) | ||||
|         } else if (errorInfo.throwable is AccountTerminatedException) { | ||||
|             errorTextView.setText(R.string.account_terminated) | ||||
|  | ||||
|             if (!isNullOrEmpty((errorInfo.throwable as AccountTerminatedException).message)) { | ||||
|                 errorServiceInfoTextView.text = context.resources.getString( | ||||
|                     R.string.service_provides_reason, | ||||
|                     NewPipe.getNameOfService(ServiceHelper.getSelectedServiceId(context)) | ||||
|                     ServiceHelper.getSelectedService(context)?.serviceInfo?.name ?: "<unknown>" | ||||
|                 ) | ||||
|                 errorServiceInfoTextView.isVisible = true | ||||
|  | ||||
| @@ -129,6 +132,7 @@ class ErrorPanelHelper( | ||||
|                 // show retry button only for content which is not unavailable or unsupported | ||||
|                 errorRetryButton.isVisible = true | ||||
|             } | ||||
|             showAndSetOpenInBrowserButtonAction(errorInfo) | ||||
|         } | ||||
|  | ||||
|         setRootVisible() | ||||
| @@ -139,13 +143,22 @@ class ErrorPanelHelper( | ||||
|      */ | ||||
|     private fun showAndSetErrorButtonAction( | ||||
|         @StringRes resid: Int, | ||||
|         @Nullable listener: View.OnClickListener | ||||
|         listener: View.OnClickListener | ||||
|     ) { | ||||
|         errorActionButton.isVisible = true | ||||
|         errorActionButton.setText(resid) | ||||
|         errorActionButton.setOnClickListener(listener) | ||||
|     } | ||||
|  | ||||
|     fun showAndSetOpenInBrowserButtonAction( | ||||
|         errorInfo: ErrorInfo | ||||
|     ) { | ||||
|         errorOpenInBrowserButton.isVisible = true | ||||
|         errorOpenInBrowserButton.setOnClickListener { | ||||
|             ShareUtils.openUrlInBrowser(context, errorInfo.request) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun showTextError(errorString: String) { | ||||
|         ensureDefaultVisibility() | ||||
|  | ||||
|   | ||||
| @@ -5,11 +5,11 @@ import android.app.PendingIntent | ||||
| import android.content.Context | ||||
| import android.content.Intent | ||||
| import android.graphics.Color | ||||
| import android.os.Build | ||||
| import android.view.View | ||||
| import android.widget.Toast | ||||
| import androidx.core.app.NotificationCompat | ||||
| import androidx.core.app.NotificationManagerCompat | ||||
| import androidx.core.app.PendingIntentCompat | ||||
| import androidx.fragment.app.Fragment | ||||
| import com.google.android.material.snackbar.Snackbar | ||||
| import org.schabi.newpipe.R | ||||
| @@ -104,32 +104,22 @@ class ErrorUtil { | ||||
|          */ | ||||
|         @JvmStatic | ||||
|         fun createNotification(context: Context, errorInfo: ErrorInfo) { | ||||
|             var pendingIntentFlags = PendingIntent.FLAG_UPDATE_CURRENT | ||||
|             if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||
|                 pendingIntentFlags = pendingIntentFlags or PendingIntent.FLAG_IMMUTABLE | ||||
|             } | ||||
|  | ||||
|             val notificationBuilder: NotificationCompat.Builder = | ||||
|                 NotificationCompat.Builder( | ||||
|                     context, | ||||
|                     context.getString(R.string.error_report_channel_id) | ||||
|                 ) | ||||
|                     .setSmallIcon( | ||||
|                         // the vector drawable icon causes crashes on KitKat devices | ||||
|                         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) | ||||
|                             R.drawable.ic_bug_report | ||||
|                         else | ||||
|                             android.R.drawable.stat_notify_error | ||||
|                     ) | ||||
|                     .setSmallIcon(R.drawable.ic_bug_report) | ||||
|                     .setContentTitle(context.getString(R.string.error_report_notification_title)) | ||||
|                     .setContentText(context.getString(errorInfo.messageStringId)) | ||||
|                     .setAutoCancel(true) | ||||
|                     .setContentIntent( | ||||
|                         PendingIntent.getActivity( | ||||
|                         PendingIntentCompat.getActivity( | ||||
|                             context, | ||||
|                             0, | ||||
|                             getErrorActivityIntent(context, errorInfo), | ||||
|                             pendingIntentFlags | ||||
|                             PendingIntent.FLAG_UPDATE_CURRENT, | ||||
|                             false | ||||
|                         ) | ||||
|                     ) | ||||
|  | ||||
|   | ||||
| @@ -3,14 +3,15 @@ package org.schabi.newpipe.error; | ||||
| import android.annotation.SuppressLint; | ||||
| import android.content.Intent; | ||||
| import android.content.SharedPreferences; | ||||
| import android.os.Build; | ||||
| import android.os.Bundle; | ||||
| import android.util.Log; | ||||
| import android.view.Menu; | ||||
| import android.view.MenuItem; | ||||
| import android.webkit.CookieManager; | ||||
| import android.webkit.WebResourceRequest; | ||||
| import android.webkit.WebSettings; | ||||
| import android.webkit.WebView; | ||||
| import android.webkit.WebViewClient; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| @@ -18,16 +19,15 @@ import androidx.appcompat.app.ActionBar; | ||||
| import androidx.appcompat.app.AppCompatActivity; | ||||
| import androidx.core.app.NavUtils; | ||||
| import androidx.preference.PreferenceManager; | ||||
| import androidx.webkit.WebViewClientCompat; | ||||
|  | ||||
| import org.schabi.newpipe.databinding.ActivityRecaptchaBinding; | ||||
| import org.schabi.newpipe.DownloaderImpl; | ||||
| import org.schabi.newpipe.MainActivity; | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.databinding.ActivityRecaptchaBinding; | ||||
| import org.schabi.newpipe.extractor.utils.Utils; | ||||
| import org.schabi.newpipe.util.ThemeHelper; | ||||
|  | ||||
| import java.io.UnsupportedEncodingException; | ||||
| import java.net.URLDecoder; | ||||
|  | ||||
| /* | ||||
|  * Created by beneth <bmauduit@beneth.fr> on 06.12.16. | ||||
| @@ -86,14 +86,15 @@ public class ReCaptchaActivity extends AppCompatActivity { | ||||
|         webSettings.setJavaScriptEnabled(true); | ||||
|         webSettings.setUserAgentString(DownloaderImpl.USER_AGENT); | ||||
|  | ||||
|         recaptchaBinding.reCaptchaWebView.setWebViewClient(new WebViewClientCompat() { | ||||
|         recaptchaBinding.reCaptchaWebView.setWebViewClient(new WebViewClient() { | ||||
|             @Override | ||||
|             public boolean shouldOverrideUrlLoading(final WebView view, final String url) { | ||||
|             public boolean shouldOverrideUrlLoading(final WebView view, | ||||
|                                                     final WebResourceRequest request) { | ||||
|                 if (MainActivity.DEBUG) { | ||||
|                     Log.d(TAG, "shouldOverrideUrlLoading: url=" + url); | ||||
|                     Log.d(TAG, "shouldOverrideUrlLoading: url=" + request.getUrl().toString()); | ||||
|                 } | ||||
|  | ||||
|                 handleCookiesFromUrl(url); | ||||
|                 handleCookiesFromUrl(request.getUrl().toString()); | ||||
|                 return false; | ||||
|             } | ||||
|  | ||||
| @@ -107,12 +108,7 @@ public class ReCaptchaActivity extends AppCompatActivity { | ||||
|         // cleaning cache, history and cookies from webView | ||||
|         recaptchaBinding.reCaptchaWebView.clearCache(true); | ||||
|         recaptchaBinding.reCaptchaWebView.clearHistory(); | ||||
|         final CookieManager cookieManager = CookieManager.getInstance(); | ||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { | ||||
|             cookieManager.removeAllCookies(value -> { }); | ||||
|         } else { | ||||
|             cookieManager.removeAllCookie(); | ||||
|         } | ||||
|         CookieManager.getInstance().removeAllCookies(null); | ||||
|  | ||||
|         recaptchaBinding.reCaptchaWebView.loadUrl(url); | ||||
|     } | ||||
| @@ -192,7 +188,7 @@ public class ReCaptchaActivity extends AppCompatActivity { | ||||
|  | ||||
|             try { | ||||
|                 String abuseCookie = url.substring(abuseStart + 13, abuseEnd); | ||||
|                 abuseCookie = URLDecoder.decode(abuseCookie, "UTF-8"); | ||||
|                 abuseCookie = Utils.decodeUrlUtf8(abuseCookie); | ||||
|                 handleCookies(abuseCookie); | ||||
|             } catch (UnsupportedEncodingException | StringIndexOutOfBoundsException e) { | ||||
|                 if (MainActivity.DEBUG) { | ||||
|   | ||||
| @@ -19,6 +19,7 @@ public enum UserAction { | ||||
|     REQUESTED_PLAYLIST("requested playlist"), | ||||
|     REQUESTED_KIOSK("requested kiosk"), | ||||
|     REQUESTED_COMMENTS("requested comments"), | ||||
|     REQUESTED_COMMENT_REPLIES("requested comment replies"), | ||||
|     REQUESTED_FEED("requested feed"), | ||||
|     REQUESTED_BOOKMARK("bookmark"), | ||||
|     DELETE_FROM_HISTORY("delete from history"), | ||||
|   | ||||
| @@ -1,12 +1,16 @@ | ||||
| package org.schabi.newpipe.fragments; | ||||
|  | ||||
| import static org.schabi.newpipe.ktx.ViewUtils.animate; | ||||
|  | ||||
| import android.os.Bundle; | ||||
| import android.util.Log; | ||||
| import android.view.View; | ||||
| import android.widget.ProgressBar; | ||||
| import android.widget.TextView; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.annotation.StringRes; | ||||
| import androidx.fragment.app.Fragment; | ||||
|  | ||||
| import org.schabi.newpipe.BaseFragment; | ||||
| @@ -20,15 +24,15 @@ import java.util.concurrent.atomic.AtomicBoolean; | ||||
|  | ||||
| import icepick.State; | ||||
|  | ||||
| import static org.schabi.newpipe.ktx.ViewUtils.animate; | ||||
|  | ||||
| public abstract class BaseStateFragment<I> extends BaseFragment implements ViewContract<I> { | ||||
|     @State | ||||
|     protected AtomicBoolean wasLoading = new AtomicBoolean(); | ||||
|     protected AtomicBoolean isLoading = new AtomicBoolean(); | ||||
|  | ||||
|     @Nullable | ||||
|     private View emptyStateView; | ||||
|     protected View emptyStateView; | ||||
|     @Nullable | ||||
|     protected TextView emptyStateMessageView; | ||||
|     @Nullable | ||||
|     private ProgressBar loadingProgressBar; | ||||
|  | ||||
| @@ -65,6 +69,7 @@ public abstract class BaseStateFragment<I> extends BaseFragment implements ViewC | ||||
|     protected void initViews(final View rootView, final Bundle savedInstanceState) { | ||||
|         super.initViews(rootView, savedInstanceState); | ||||
|         emptyStateView = rootView.findViewById(R.id.empty_state_view); | ||||
|         emptyStateMessageView = rootView.findViewById(R.id.empty_state_message); | ||||
|         loadingProgressBar = rootView.findViewById(R.id.loading_progress_bar); | ||||
|         errorPanelHelper = new ErrorPanelHelper(this, rootView, this::onRetryButtonClicked); | ||||
|     } | ||||
| @@ -75,6 +80,8 @@ public abstract class BaseStateFragment<I> extends BaseFragment implements ViewC | ||||
|         if (errorPanelHelper != null) { | ||||
|             errorPanelHelper.dispose(); | ||||
|         } | ||||
|         emptyStateView = null; | ||||
|         emptyStateMessageView = null; | ||||
|     } | ||||
|  | ||||
|     protected void onRetryButtonClicked() { | ||||
| @@ -189,6 +196,12 @@ public abstract class BaseStateFragment<I> extends BaseFragment implements ViewC | ||||
|         errorPanelHelper.showTextError(errorString); | ||||
|     } | ||||
|  | ||||
|     protected void setEmptyStateMessage(@StringRes final int text) { | ||||
|         if (emptyStateMessageView != null) { | ||||
|             emptyStateMessageView.setText(text); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public final void hideErrorPanel() { | ||||
|         errorPanelHelper.hide(); | ||||
|         lastPanelError = null; | ||||
|   | ||||
| @@ -1,6 +1,16 @@ | ||||
| package org.schabi.newpipe.fragments; | ||||
|  | ||||
| import static android.widget.RelativeLayout.ABOVE; | ||||
| import static android.widget.RelativeLayout.ALIGN_PARENT_BOTTOM; | ||||
| import static android.widget.RelativeLayout.ALIGN_PARENT_TOP; | ||||
| import static android.widget.RelativeLayout.BELOW; | ||||
| import static com.google.android.material.tabs.TabLayout.INDICATOR_GRAVITY_BOTTOM; | ||||
| import static com.google.android.material.tabs.TabLayout.INDICATOR_GRAVITY_TOP; | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.content.SharedPreferences; | ||||
| import android.content.res.ColorStateList; | ||||
| import android.graphics.Color; | ||||
| import android.os.Bundle; | ||||
| import android.util.Log; | ||||
| import android.view.LayoutInflater; | ||||
| @@ -9,7 +19,9 @@ import android.view.MenuInflater; | ||||
| import android.view.MenuItem; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.RelativeLayout; | ||||
|  | ||||
| import androidx.annotation.ColorInt; | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.appcompat.app.ActionBar; | ||||
| @@ -17,6 +29,7 @@ import androidx.fragment.app.Fragment; | ||||
| import androidx.fragment.app.FragmentManager; | ||||
| import androidx.fragment.app.FragmentStatePagerAdapterMenuWorkaround; | ||||
| import androidx.preference.PreferenceManager; | ||||
| import androidx.viewpager.widget.ViewPager; | ||||
|  | ||||
| import com.google.android.material.tabs.TabLayout; | ||||
|  | ||||
| @@ -25,10 +38,13 @@ import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.databinding.FragmentMainBinding; | ||||
| import org.schabi.newpipe.error.ErrorUtil; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.local.playlist.LocalPlaylistFragment; | ||||
| import org.schabi.newpipe.settings.tabs.Tab; | ||||
| import org.schabi.newpipe.settings.tabs.TabsManager; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.ServiceHelper; | ||||
| import org.schabi.newpipe.util.ThemeHelper; | ||||
| import org.schabi.newpipe.views.ScrollableTabLayout; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| @@ -42,8 +58,11 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | ||||
|  | ||||
|     private boolean hasTabsChanged = false; | ||||
|  | ||||
|     private boolean previousYoutubeRestrictedModeEnabled; | ||||
|     private SharedPreferences prefs; | ||||
|     private boolean youtubeRestrictedModeEnabled; | ||||
|     private String youtubeRestrictedModeEnabledKey; | ||||
|     private boolean mainTabsPositionBottom; | ||||
|     private String mainTabsPositionKey; | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Fragment's LifeCycle | ||||
| @@ -66,10 +85,11 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         prefs = PreferenceManager.getDefaultSharedPreferences(requireContext()); | ||||
|         youtubeRestrictedModeEnabledKey = getString(R.string.youtube_restricted_mode_enabled); | ||||
|         previousYoutubeRestrictedModeEnabled = | ||||
|                 PreferenceManager.getDefaultSharedPreferences(requireContext()) | ||||
|                         .getBoolean(youtubeRestrictedModeEnabledKey, false); | ||||
|         youtubeRestrictedModeEnabled = prefs.getBoolean(youtubeRestrictedModeEnabledKey, false); | ||||
|         mainTabsPositionKey = getString(R.string.main_tabs_position_key); | ||||
|         mainTabsPositionBottom = prefs.getBoolean(mainTabsPositionKey, false); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -87,25 +107,27 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | ||||
|  | ||||
|         binding.mainTabLayout.setupWithViewPager(binding.pager); | ||||
|         binding.mainTabLayout.addOnTabSelectedListener(this); | ||||
|         binding.mainTabLayout.setTabRippleColor(binding.mainTabLayout.getTabRippleColor() | ||||
|                 .withAlpha(32)); | ||||
|  | ||||
|         setupTabs(); | ||||
|         updateTabLayoutPosition(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onResume() { | ||||
|         super.onResume(); | ||||
|  | ||||
|         final boolean youtubeRestrictedModeEnabled = | ||||
|                 PreferenceManager.getDefaultSharedPreferences(requireContext()) | ||||
|                         .getBoolean(youtubeRestrictedModeEnabledKey, false); | ||||
|         if (previousYoutubeRestrictedModeEnabled != youtubeRestrictedModeEnabled) { | ||||
|             previousYoutubeRestrictedModeEnabled = youtubeRestrictedModeEnabled; | ||||
|             setupTabs(); | ||||
|         } else if (hasTabsChanged) { | ||||
|         final boolean newYoutubeRestrictedModeEnabled = | ||||
|                 prefs.getBoolean(youtubeRestrictedModeEnabledKey, false); | ||||
|         if (youtubeRestrictedModeEnabled != newYoutubeRestrictedModeEnabled || hasTabsChanged) { | ||||
|             youtubeRestrictedModeEnabled = newYoutubeRestrictedModeEnabled; | ||||
|             setupTabs(); | ||||
|         } | ||||
|  | ||||
|         final boolean newMainTabsPosition = prefs.getBoolean(mainTabsPositionKey, false); | ||||
|         if (mainTabsPositionBottom != newMainTabsPosition) { | ||||
|             mainTabsPositionBottom = newMainTabsPosition; | ||||
|             updateTabLayoutPosition(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -118,6 +140,12 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroyView() { | ||||
|         super.onDestroyView(); | ||||
|         binding = null; | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Menu | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
| @@ -166,7 +194,6 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | ||||
|         } | ||||
|  | ||||
|         binding.pager.setAdapter(null); | ||||
|         binding.pager.setOffscreenPageLimit(tabsList.size()); | ||||
|         binding.pager.setAdapter(pagerAdapter); | ||||
|  | ||||
|         updateTabsIconAndDescription(); | ||||
| @@ -190,6 +217,44 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | ||||
|         setTitle(tabsList.get(tabPosition).getTabName(requireContext())); | ||||
|     } | ||||
|  | ||||
|     public void commitPlaylistTabs() { | ||||
|         pagerAdapter.getLocalPlaylistFragments() | ||||
|                 .stream() | ||||
|                 .forEach(LocalPlaylistFragment::saveImmediate); | ||||
|     } | ||||
|  | ||||
|     private void updateTabLayoutPosition() { | ||||
|         final ScrollableTabLayout tabLayout = binding.mainTabLayout; | ||||
|         final ViewPager viewPager = binding.pager; | ||||
|         final boolean bottom = mainTabsPositionBottom; | ||||
|  | ||||
|         // change layout params to make the tab layout appear either at the top or at the bottom | ||||
|         final var tabParams = (RelativeLayout.LayoutParams) tabLayout.getLayoutParams(); | ||||
|         final var pagerParams = (RelativeLayout.LayoutParams) viewPager.getLayoutParams(); | ||||
|  | ||||
|         tabParams.removeRule(bottom ? ALIGN_PARENT_TOP : ALIGN_PARENT_BOTTOM); | ||||
|         tabParams.addRule(bottom ? ALIGN_PARENT_BOTTOM : ALIGN_PARENT_TOP); | ||||
|         pagerParams.removeRule(bottom ? BELOW : ABOVE); | ||||
|         pagerParams.addRule(bottom ? ABOVE : BELOW, R.id.main_tab_layout); | ||||
|         tabLayout.setSelectedTabIndicatorGravity( | ||||
|                 bottom ? INDICATOR_GRAVITY_TOP : INDICATOR_GRAVITY_BOTTOM); | ||||
|  | ||||
|         tabLayout.setLayoutParams(tabParams); | ||||
|         viewPager.setLayoutParams(pagerParams); | ||||
|  | ||||
|         // change the background and icon color of the tab layout: | ||||
|         // service-colored at the top, app-background-colored at the bottom | ||||
|         tabLayout.setBackgroundColor(ThemeHelper.resolveColorFromAttr(requireContext(), | ||||
|                 bottom ? R.attr.colorSecondary : R.attr.colorPrimary)); | ||||
|  | ||||
|         @ColorInt final int iconColor = bottom | ||||
|                 ? ThemeHelper.resolveColorFromAttr(requireContext(), R.attr.colorAccent) | ||||
|                 : Color.WHITE; | ||||
|         tabLayout.setTabRippleColor(ColorStateList.valueOf(iconColor).withAlpha(32)); | ||||
|         tabLayout.setTabIconTint(ColorStateList.valueOf(iconColor)); | ||||
|         tabLayout.setSelectedTabIndicatorColor(iconColor); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onTabSelected(final TabLayout.Tab selectedTab) { | ||||
|         if (DEBUG) { | ||||
| @@ -209,10 +274,18 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | ||||
|         updateTitleForTab(tab.getPosition()); | ||||
|     } | ||||
|  | ||||
|     private static final class SelectedTabsPagerAdapter | ||||
|     public static final class SelectedTabsPagerAdapter | ||||
|             extends FragmentStatePagerAdapterMenuWorkaround { | ||||
|         private final Context context; | ||||
|         private final List<Tab> internalTabsList; | ||||
|         /** | ||||
|          * Keep reference to LocalPlaylistFragments, because their data can be modified by the user | ||||
|          * during runtime and changes are not committed immediately. However, in some cases, | ||||
|          * the changes need to be committed immediately by calling | ||||
|          * {@link LocalPlaylistFragment#saveImmediate()}. | ||||
|          * The fragments are removed when {@link LocalPlaylistFragment#onDestroy()} is called. | ||||
|          */ | ||||
|         private final List<LocalPlaylistFragment> localPlaylistFragments = new ArrayList<>(); | ||||
|  | ||||
|         private SelectedTabsPagerAdapter(final Context context, | ||||
|                                          final FragmentManager fragmentManager, | ||||
| @@ -239,9 +312,17 @@ public class MainFragment extends BaseFragment implements TabLayout.OnTabSelecte | ||||
|                 ((BaseFragment) fragment).useAsFrontPage(true); | ||||
|             } | ||||
|  | ||||
|             if (fragment instanceof LocalPlaylistFragment) { | ||||
|                 localPlaylistFragments.add((LocalPlaylistFragment) fragment); | ||||
|             } | ||||
|  | ||||
|             return fragment; | ||||
|         } | ||||
|  | ||||
|         public List<LocalPlaylistFragment> getLocalPlaylistFragments() { | ||||
|             return localPlaylistFragments; | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public int getItemPosition(@NonNull final Object object) { | ||||
|             // Causes adapter to reload all Fragments when | ||||
|   | ||||
| @@ -0,0 +1,281 @@ | ||||
| package org.schabi.newpipe.fragments.detail; | ||||
|  | ||||
| import static android.text.TextUtils.isEmpty; | ||||
| import static org.schabi.newpipe.extractor.utils.Utils.isBlank; | ||||
| import static org.schabi.newpipe.util.text.TextLinkifier.SET_LINK_MOVEMENT_METHOD; | ||||
|  | ||||
| import android.graphics.Typeface; | ||||
| import android.os.Bundle; | ||||
| import android.text.SpannableStringBuilder; | ||||
| import android.text.Spanned; | ||||
| import android.text.method.LinkMovementMethod; | ||||
| import android.text.style.ClickableSpan; | ||||
| import android.text.style.StyleSpan; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.LinearLayout; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.annotation.StringRes; | ||||
| import androidx.appcompat.widget.TooltipCompat; | ||||
| import androidx.core.text.HtmlCompat; | ||||
|  | ||||
| import com.google.android.material.chip.Chip; | ||||
|  | ||||
| import org.schabi.newpipe.BaseFragment; | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.databinding.FragmentDescriptionBinding; | ||||
| import org.schabi.newpipe.databinding.ItemMetadataBinding; | ||||
| import org.schabi.newpipe.databinding.ItemMetadataTagsBinding; | ||||
| import org.schabi.newpipe.extractor.Image; | ||||
| import org.schabi.newpipe.extractor.StreamingService; | ||||
| import org.schabi.newpipe.extractor.stream.Description; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils; | ||||
| import org.schabi.newpipe.util.image.ImageStrategy; | ||||
| import org.schabi.newpipe.util.text.TextLinkifier; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import io.reactivex.rxjava3.disposables.CompositeDisposable; | ||||
|  | ||||
| public abstract class BaseDescriptionFragment extends BaseFragment { | ||||
|     private final CompositeDisposable descriptionDisposables = new CompositeDisposable(); | ||||
|     protected FragmentDescriptionBinding binding; | ||||
|  | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, | ||||
|                              @Nullable final ViewGroup container, | ||||
|                              @Nullable final Bundle savedInstanceState) { | ||||
|         binding = FragmentDescriptionBinding.inflate(inflater, container, false); | ||||
|         setupDescription(); | ||||
|         setupMetadata(inflater, binding.detailMetadataLayout); | ||||
|         addTagsMetadataItem(inflater, binding.detailMetadataLayout); | ||||
|         return binding.getRoot(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroy() { | ||||
|         descriptionDisposables.clear(); | ||||
|         super.onDestroy(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the description to display. | ||||
|      * @return description object, if available | ||||
|      */ | ||||
|     @Nullable | ||||
|     protected abstract Description getDescription(); | ||||
|  | ||||
|     /** | ||||
|      * Get the streaming service. Used for generating description links. | ||||
|      * @return streaming service | ||||
|      */ | ||||
|     @NonNull | ||||
|     protected abstract StreamingService getService(); | ||||
|  | ||||
|     /** | ||||
|      * Get the streaming service ID. Used for tag links. | ||||
|      * @return service ID | ||||
|      */ | ||||
|     protected abstract int getServiceId(); | ||||
|  | ||||
|     /** | ||||
|      * Get the URL of the described video or audio, used to generate description links. | ||||
|      * @return stream URL | ||||
|      */ | ||||
|     @Nullable | ||||
|     protected abstract String getStreamUrl(); | ||||
|  | ||||
|     /** | ||||
|      * Get the list of tags to display below the description. | ||||
|      * @return tag list | ||||
|      */ | ||||
|     @NonNull | ||||
|     public abstract List<String> getTags(); | ||||
|  | ||||
|     /** | ||||
|      * Add additional metadata to display. | ||||
|      * @param inflater LayoutInflater | ||||
|      * @param layout detailMetadataLayout | ||||
|      */ | ||||
|     protected abstract void setupMetadata(LayoutInflater inflater, LinearLayout layout); | ||||
|  | ||||
|     private void setupDescription() { | ||||
|         final Description description = getDescription(); | ||||
|         if (description == null || isEmpty(description.getContent()) | ||||
|                 || description == Description.EMPTY_DESCRIPTION) { | ||||
|             binding.detailDescriptionView.setVisibility(View.GONE); | ||||
|             binding.detailSelectDescriptionButton.setVisibility(View.GONE); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // start with disabled state. This also loads description content (!) | ||||
|         disableDescriptionSelection(); | ||||
|  | ||||
|         binding.detailSelectDescriptionButton.setOnClickListener(v -> { | ||||
|             if (binding.detailDescriptionNoteView.getVisibility() == View.VISIBLE) { | ||||
|                 disableDescriptionSelection(); | ||||
|             } else { | ||||
|                 // enable selection only when button is clicked to prevent flickering | ||||
|                 enableDescriptionSelection(); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     private void enableDescriptionSelection() { | ||||
|         binding.detailDescriptionNoteView.setVisibility(View.VISIBLE); | ||||
|         binding.detailDescriptionView.setTextIsSelectable(true); | ||||
|  | ||||
|         final String buttonLabel = getString(R.string.description_select_disable); | ||||
|         binding.detailSelectDescriptionButton.setContentDescription(buttonLabel); | ||||
|         TooltipCompat.setTooltipText(binding.detailSelectDescriptionButton, buttonLabel); | ||||
|         binding.detailSelectDescriptionButton.setImageResource(R.drawable.ic_close); | ||||
|     } | ||||
|  | ||||
|     private void disableDescriptionSelection() { | ||||
|         // show description content again, otherwise some links are not clickable | ||||
|         final Description description = getDescription(); | ||||
|         if (description != null) { | ||||
|             TextLinkifier.fromDescription(binding.detailDescriptionView, | ||||
|                     description, HtmlCompat.FROM_HTML_MODE_LEGACY, | ||||
|                     getService(), getStreamUrl(), | ||||
|                     descriptionDisposables, SET_LINK_MOVEMENT_METHOD); | ||||
|         } | ||||
|  | ||||
|         binding.detailDescriptionNoteView.setVisibility(View.GONE); | ||||
|         binding.detailDescriptionView.setTextIsSelectable(false); | ||||
|  | ||||
|         final String buttonLabel = getString(R.string.description_select_enable); | ||||
|         binding.detailSelectDescriptionButton.setContentDescription(buttonLabel); | ||||
|         TooltipCompat.setTooltipText(binding.detailSelectDescriptionButton, buttonLabel); | ||||
|         binding.detailSelectDescriptionButton.setImageResource(R.drawable.ic_select_all); | ||||
|     } | ||||
|  | ||||
|     protected void addMetadataItem(final LayoutInflater inflater, | ||||
|                                    final LinearLayout layout, | ||||
|                                    final boolean linkifyContent, | ||||
|                                    @StringRes final int type, | ||||
|                                    @NonNull final String content) { | ||||
|         if (isBlank(content)) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         final ItemMetadataBinding itemBinding = | ||||
|                 ItemMetadataBinding.inflate(inflater, layout, false); | ||||
|  | ||||
|         itemBinding.metadataTypeView.setText(type); | ||||
|         itemBinding.metadataTypeView.setOnLongClickListener(v -> { | ||||
|             ShareUtils.copyToClipboard(requireContext(), content); | ||||
|             return true; | ||||
|         }); | ||||
|  | ||||
|         if (linkifyContent) { | ||||
|             TextLinkifier.fromPlainText(itemBinding.metadataContentView, content, null, null, | ||||
|                     descriptionDisposables, SET_LINK_MOVEMENT_METHOD); | ||||
|         } else { | ||||
|             itemBinding.metadataContentView.setText(content); | ||||
|         } | ||||
|  | ||||
|         itemBinding.metadataContentView.setClickable(true); | ||||
|  | ||||
|         layout.addView(itemBinding.getRoot()); | ||||
|     } | ||||
|  | ||||
|     private String imageSizeToText(final int heightOrWidth) { | ||||
|         if (heightOrWidth < 0) { | ||||
|             return getString(R.string.question_mark); | ||||
|         } else { | ||||
|             return String.valueOf(heightOrWidth); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     protected void addImagesMetadataItem(final LayoutInflater inflater, | ||||
|                                          final LinearLayout layout, | ||||
|                                          @StringRes final int type, | ||||
|                                          final List<Image> images) { | ||||
|         final String preferredImageUrl = ImageStrategy.choosePreferredImage(images); | ||||
|         if (preferredImageUrl == null) { | ||||
|             return; // null will be returned in case there is no image | ||||
|         } | ||||
|  | ||||
|         final ItemMetadataBinding itemBinding = | ||||
|                 ItemMetadataBinding.inflate(inflater, layout, false); | ||||
|         itemBinding.metadataTypeView.setText(type); | ||||
|  | ||||
|         final SpannableStringBuilder urls = new SpannableStringBuilder(); | ||||
|         for (final Image image : images) { | ||||
|             if (urls.length() != 0) { | ||||
|                 urls.append(", "); | ||||
|             } | ||||
|             final int entryBegin = urls.length(); | ||||
|  | ||||
|             if (image.getHeight() != Image.HEIGHT_UNKNOWN | ||||
|                     || image.getWidth() != Image.WIDTH_UNKNOWN | ||||
|                     // if even the resolution level is unknown, ?x? will be shown | ||||
|                     || image.getEstimatedResolutionLevel() == Image.ResolutionLevel.UNKNOWN) { | ||||
|                 urls.append(imageSizeToText(image.getHeight())); | ||||
|                 urls.append('x'); | ||||
|                 urls.append(imageSizeToText(image.getWidth())); | ||||
|             } else { | ||||
|                 switch (image.getEstimatedResolutionLevel()) { | ||||
|                     case LOW -> urls.append(getString(R.string.image_quality_low)); | ||||
|                     case MEDIUM -> urls.append(getString(R.string.image_quality_medium)); | ||||
|                     case HIGH -> urls.append(getString(R.string.image_quality_high)); | ||||
|                     default -> { | ||||
|                         // unreachable, Image.ResolutionLevel.UNKNOWN is already filtered out | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             urls.setSpan(new ClickableSpan() { | ||||
|                 @Override | ||||
|                 public void onClick(@NonNull final View widget) { | ||||
|                     ShareUtils.openUrlInBrowser(requireContext(), image.getUrl()); | ||||
|                 } | ||||
|             }, entryBegin, urls.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); | ||||
|  | ||||
|             if (preferredImageUrl.equals(image.getUrl())) { | ||||
|                 urls.setSpan(new StyleSpan(Typeface.BOLD), entryBegin, urls.length(), | ||||
|                         Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         itemBinding.metadataContentView.setText(urls); | ||||
|         itemBinding.metadataContentView.setMovementMethod(LinkMovementMethod.getInstance()); | ||||
|         layout.addView(itemBinding.getRoot()); | ||||
|     } | ||||
|  | ||||
|     private void addTagsMetadataItem(final LayoutInflater inflater, final LinearLayout layout) { | ||||
|         final List<String> tags = getTags(); | ||||
|  | ||||
|         if (!tags.isEmpty()) { | ||||
|             final var itemBinding = ItemMetadataTagsBinding.inflate(inflater, layout, false); | ||||
|  | ||||
|             tags.stream().sorted(String.CASE_INSENSITIVE_ORDER).forEach(tag -> { | ||||
|                 final Chip chip = (Chip) inflater.inflate(R.layout.chip, | ||||
|                         itemBinding.metadataTagsChips, false); | ||||
|                 chip.setText(tag); | ||||
|                 chip.setOnClickListener(this::onTagClick); | ||||
|                 chip.setOnLongClickListener(this::onTagLongClick); | ||||
|                 itemBinding.metadataTagsChips.addView(chip); | ||||
|             }); | ||||
|  | ||||
|             layout.addView(itemBinding.getRoot()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void onTagClick(final View chip) { | ||||
|         if (getParentFragment() != null) { | ||||
|             NavigationHelper.openSearchFragment(getParentFragment().getParentFragmentManager(), | ||||
|                     getServiceId(), ((Chip) chip).getText().toString()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private boolean onTagLongClick(final View chip) { | ||||
|         ShareUtils.copyToClipboard(requireContext(), ((Chip) chip).getText().toString()); | ||||
|         return true; | ||||
|     } | ||||
| } | ||||
| @@ -1,239 +1,108 @@ | ||||
| package org.schabi.newpipe.fragments.detail; | ||||
|  | ||||
| import android.os.Bundle; | ||||
| import static org.schabi.newpipe.extractor.stream.StreamExtractor.NO_AGE_LIMIT; | ||||
| import static org.schabi.newpipe.util.Localization.getAppLocale; | ||||
|  | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.LinearLayout; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.annotation.StringRes; | ||||
| import androidx.appcompat.widget.TooltipCompat; | ||||
| import androidx.core.text.HtmlCompat; | ||||
|  | ||||
| import com.google.android.material.chip.Chip; | ||||
|  | ||||
| import org.schabi.newpipe.BaseFragment; | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.databinding.FragmentDescriptionBinding; | ||||
| import org.schabi.newpipe.databinding.ItemMetadataBinding; | ||||
| import org.schabi.newpipe.databinding.ItemMetadataTagsBinding; | ||||
| import org.schabi.newpipe.extractor.StreamingService; | ||||
| import org.schabi.newpipe.extractor.stream.Description; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfo; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils; | ||||
| import org.schabi.newpipe.util.external_communication.TextLinkifier; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
|  | ||||
| import icepick.State; | ||||
| import io.reactivex.rxjava3.disposables.CompositeDisposable; | ||||
|  | ||||
| import static android.text.TextUtils.isEmpty; | ||||
| import static org.schabi.newpipe.extractor.stream.StreamExtractor.NO_AGE_LIMIT; | ||||
| import static org.schabi.newpipe.extractor.utils.Utils.isBlank; | ||||
|  | ||||
| public class DescriptionFragment extends BaseFragment { | ||||
| public class DescriptionFragment extends BaseDescriptionFragment { | ||||
|  | ||||
|     @State | ||||
|     StreamInfo streamInfo = null; | ||||
|     final CompositeDisposable descriptionDisposables = new CompositeDisposable(); | ||||
|     FragmentDescriptionBinding binding; | ||||
|  | ||||
|     public DescriptionFragment() { | ||||
|     } | ||||
|     StreamInfo streamInfo; | ||||
|  | ||||
|     public DescriptionFragment(final StreamInfo streamInfo) { | ||||
|         this.streamInfo = streamInfo; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, | ||||
|                              @Nullable final ViewGroup container, | ||||
|                              @Nullable final Bundle savedInstanceState) { | ||||
|         binding = FragmentDescriptionBinding.inflate(inflater, container, false); | ||||
|         if (streamInfo != null) { | ||||
|             setupUploadDate(); | ||||
|             setupDescription(); | ||||
|             setupMetadata(inflater, binding.detailMetadataLayout); | ||||
|         } | ||||
|         return binding.getRoot(); | ||||
|     protected Description getDescription() { | ||||
|         return streamInfo.getDescription(); | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     protected StreamingService getService() { | ||||
|         return streamInfo.getService(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroy() { | ||||
|         descriptionDisposables.clear(); | ||||
|         super.onDestroy(); | ||||
|     protected int getServiceId() { | ||||
|         return streamInfo.getServiceId(); | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     protected String getStreamUrl() { | ||||
|         return streamInfo.getUrl(); | ||||
|     } | ||||
|  | ||||
|     private void setupUploadDate() { | ||||
|         if (streamInfo.getUploadDate() != null) { | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public List<String> getTags() { | ||||
|         return streamInfo.getTags(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void setupMetadata(final LayoutInflater inflater, | ||||
|                                  final LinearLayout layout) { | ||||
|         if (streamInfo != null && streamInfo.getUploadDate() != null) { | ||||
|             binding.detailUploadDateView.setText(Localization | ||||
|                     .localizeUploadDate(activity, streamInfo.getUploadDate().offsetDateTime())); | ||||
|         } else { | ||||
|             binding.detailUploadDateView.setVisibility(View.GONE); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     private void setupDescription() { | ||||
|         final Description description = streamInfo.getDescription(); | ||||
|         if (description == null || isEmpty(description.getContent()) | ||||
|                 || description == Description.EMPTY_DESCRIPTION) { | ||||
|             binding.detailDescriptionView.setVisibility(View.GONE); | ||||
|             binding.detailSelectDescriptionButton.setVisibility(View.GONE); | ||||
|         if (streamInfo == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // start with disabled state. This also loads description content (!) | ||||
|         disableDescriptionSelection(); | ||||
|         addMetadataItem(inflater, layout, false, R.string.metadata_category, | ||||
|                 streamInfo.getCategory()); | ||||
|  | ||||
|         binding.detailSelectDescriptionButton.setOnClickListener(v -> { | ||||
|             if (binding.detailDescriptionNoteView.getVisibility() == View.VISIBLE) { | ||||
|                 disableDescriptionSelection(); | ||||
|             } else { | ||||
|                 // enable selection only when button is clicked to prevent flickering | ||||
|                 enableDescriptionSelection(); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     private void enableDescriptionSelection() { | ||||
|         binding.detailDescriptionNoteView.setVisibility(View.VISIBLE); | ||||
|         binding.detailDescriptionView.setTextIsSelectable(true); | ||||
|  | ||||
|         final String buttonLabel = getString(R.string.description_select_disable); | ||||
|         binding.detailSelectDescriptionButton.setContentDescription(buttonLabel); | ||||
|         TooltipCompat.setTooltipText(binding.detailSelectDescriptionButton, buttonLabel); | ||||
|         binding.detailSelectDescriptionButton.setImageResource(R.drawable.ic_close); | ||||
|     } | ||||
|  | ||||
|     private void disableDescriptionSelection() { | ||||
|         // show description content again, otherwise some links are not clickable | ||||
|         loadDescriptionContent(); | ||||
|  | ||||
|         binding.detailDescriptionNoteView.setVisibility(View.GONE); | ||||
|         binding.detailDescriptionView.setTextIsSelectable(false); | ||||
|  | ||||
|         final String buttonLabel = getString(R.string.description_select_enable); | ||||
|         binding.detailSelectDescriptionButton.setContentDescription(buttonLabel); | ||||
|         TooltipCompat.setTooltipText(binding.detailSelectDescriptionButton, buttonLabel); | ||||
|         binding.detailSelectDescriptionButton.setImageResource(R.drawable.ic_select_all); | ||||
|     } | ||||
|  | ||||
|     private void loadDescriptionContent() { | ||||
|         final Description description = streamInfo.getDescription(); | ||||
|         switch (description.getType()) { | ||||
|             case Description.HTML: | ||||
|                 TextLinkifier.createLinksFromHtmlBlock(binding.detailDescriptionView, | ||||
|                         description.getContent(), HtmlCompat.FROM_HTML_MODE_LEGACY, streamInfo, | ||||
|                         descriptionDisposables); | ||||
|                 break; | ||||
|             case Description.MARKDOWN: | ||||
|                 TextLinkifier.createLinksFromMarkdownText(binding.detailDescriptionView, | ||||
|                         description.getContent(), streamInfo, descriptionDisposables); | ||||
|                 break; | ||||
|             case Description.PLAIN_TEXT: default: | ||||
|                 TextLinkifier.createLinksFromPlainText(binding.detailDescriptionView, | ||||
|                         description.getContent(), streamInfo, descriptionDisposables); | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     private void setupMetadata(final LayoutInflater inflater, | ||||
|                                final LinearLayout layout) { | ||||
|         addMetadataItem(inflater, layout, false, | ||||
|                 R.string.metadata_category, streamInfo.getCategory()); | ||||
|  | ||||
|         addMetadataItem(inflater, layout, false, | ||||
|                 R.string.metadata_licence, streamInfo.getLicence()); | ||||
|         addMetadataItem(inflater, layout, false, R.string.metadata_licence, | ||||
|                 streamInfo.getLicence()); | ||||
|  | ||||
|         addPrivacyMetadataItem(inflater, layout); | ||||
|  | ||||
|         if (streamInfo.getAgeLimit() != NO_AGE_LIMIT) { | ||||
|             addMetadataItem(inflater, layout, false, | ||||
|                     R.string.metadata_age_limit, String.valueOf(streamInfo.getAgeLimit())); | ||||
|             addMetadataItem(inflater, layout, false, R.string.metadata_age_limit, | ||||
|                     String.valueOf(streamInfo.getAgeLimit())); | ||||
|         } | ||||
|  | ||||
|         if (streamInfo.getLanguageInfo() != null) { | ||||
|             addMetadataItem(inflater, layout, false, | ||||
|                     R.string.metadata_language, streamInfo.getLanguageInfo().getDisplayLanguage()); | ||||
|             addMetadataItem(inflater, layout, false, R.string.metadata_language, | ||||
|                     streamInfo.getLanguageInfo().getDisplayLanguage(getAppLocale(getContext()))); | ||||
|         } | ||||
|  | ||||
|         addMetadataItem(inflater, layout, true, | ||||
|                 R.string.metadata_support, streamInfo.getSupportInfo()); | ||||
|         addMetadataItem(inflater, layout, true, | ||||
|                 R.string.metadata_host, streamInfo.getHost()); | ||||
|         addMetadataItem(inflater, layout, true, | ||||
|                 R.string.metadata_thumbnail_url, streamInfo.getThumbnailUrl()); | ||||
|         addMetadataItem(inflater, layout, true, R.string.metadata_support, | ||||
|                 streamInfo.getSupportInfo()); | ||||
|         addMetadataItem(inflater, layout, true, R.string.metadata_host, | ||||
|                 streamInfo.getHost()); | ||||
|  | ||||
|         addTagsMetadataItem(inflater, layout); | ||||
|     } | ||||
|  | ||||
|     private void addMetadataItem(final LayoutInflater inflater, | ||||
|                                  final LinearLayout layout, | ||||
|                                  final boolean linkifyContent, | ||||
|                                  @StringRes final int type, | ||||
|                                  @Nullable final String content) { | ||||
|         if (isBlank(content)) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         final ItemMetadataBinding itemBinding | ||||
|                 = ItemMetadataBinding.inflate(inflater, layout, false); | ||||
|  | ||||
|         itemBinding.metadataTypeView.setText(type); | ||||
|         itemBinding.metadataTypeView.setOnLongClickListener(v -> { | ||||
|             ShareUtils.copyToClipboard(requireContext(), content); | ||||
|             return true; | ||||
|         }); | ||||
|  | ||||
|         if (linkifyContent) { | ||||
|             TextLinkifier.createLinksFromPlainText(itemBinding.metadataContentView, content, null, | ||||
|                     descriptionDisposables); | ||||
|         } else { | ||||
|             itemBinding.metadataContentView.setText(content); | ||||
|         } | ||||
|  | ||||
|         layout.addView(itemBinding.getRoot()); | ||||
|     } | ||||
|  | ||||
|     private void addTagsMetadataItem(final LayoutInflater inflater, final LinearLayout layout) { | ||||
|         if (streamInfo.getTags() != null && !streamInfo.getTags().isEmpty()) { | ||||
|             final ItemMetadataTagsBinding itemBinding | ||||
|                     = ItemMetadataTagsBinding.inflate(inflater, layout, false); | ||||
|  | ||||
|             final List<String> tags = new ArrayList<>(streamInfo.getTags()); | ||||
|             Collections.sort(tags); | ||||
|             for (final String tag : tags) { | ||||
|                 final Chip chip = (Chip) inflater.inflate(R.layout.chip, | ||||
|                         itemBinding.metadataTagsChips, false); | ||||
|                 chip.setText(tag); | ||||
|                 chip.setOnClickListener(this::onTagClick); | ||||
|                 chip.setOnLongClickListener(this::onTagLongClick); | ||||
|                 itemBinding.metadataTagsChips.addView(chip); | ||||
|             } | ||||
|  | ||||
|             layout.addView(itemBinding.getRoot()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void onTagClick(final View chip) { | ||||
|         if (getParentFragment() != null) { | ||||
|             NavigationHelper.openSearchFragment(getParentFragment().getParentFragmentManager(), | ||||
|                     streamInfo.getServiceId(), ((Chip) chip).getText().toString()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private boolean onTagLongClick(final View chip) { | ||||
|         ShareUtils.copyToClipboard(requireContext(), ((Chip) chip).getText().toString()); | ||||
|         return true; | ||||
|         addImagesMetadataItem(inflater, layout, R.string.metadata_thumbnails, | ||||
|                 streamInfo.getThumbnails()); | ||||
|         addImagesMetadataItem(inflater, layout, R.string.metadata_uploader_avatars, | ||||
|                 streamInfo.getUploaderAvatars()); | ||||
|         addImagesMetadataItem(inflater, layout, R.string.metadata_subchannel_avatars, | ||||
|                 streamInfo.getSubChannelAvatars()); | ||||
|     } | ||||
|  | ||||
|     private void addPrivacyMetadataItem(final LayoutInflater inflater, final LinearLayout layout) { | ||||
| @@ -252,14 +121,15 @@ public class DescriptionFragment extends BaseFragment { | ||||
|                 case INTERNAL: | ||||
|                     contentRes = R.string.metadata_privacy_internal; | ||||
|                     break; | ||||
|                 case OTHER: default: | ||||
|                 case OTHER: | ||||
|                 default: | ||||
|                     contentRes = 0; | ||||
|                     break; | ||||
|             } | ||||
|  | ||||
|             if (contentRes != 0) { | ||||
|                 addMetadataItem(inflater, layout, false, | ||||
|                         R.string.metadata_privacy, getString(contentRes)); | ||||
|                 addMetadataItem(inflater, layout, false, R.string.metadata_privacy, | ||||
|                         getString(contentRes)); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,7 +1,12 @@ | ||||
| package org.schabi.newpipe.fragments.detail; | ||||
|  | ||||
| import static com.google.android.exoplayer2.PlaybackException.ERROR_CODE_BEHIND_LIVE_WINDOW; | ||||
| import static com.google.android.exoplayer2.PlaybackException.ERROR_CODE_DECODING_FAILED; | ||||
| import static com.google.android.exoplayer2.PlaybackException.ERROR_CODE_UNSPECIFIED; | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.util.Log; | ||||
| import android.util.Pair; | ||||
| import android.view.ContextThemeWrapper; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.ViewGroup; | ||||
| @@ -24,15 +29,9 @@ import org.schabi.newpipe.player.Player; | ||||
| import org.schabi.newpipe.util.ThemeHelper; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.util.Collections; | ||||
| import java.util.LinkedHashMap; | ||||
| import java.util.Map; | ||||
| import java.util.List; | ||||
| import java.util.function.Supplier; | ||||
|  | ||||
| import static com.google.android.exoplayer2.PlaybackException.ERROR_CODE_BEHIND_LIVE_WINDOW; | ||||
| import static com.google.android.exoplayer2.PlaybackException.ERROR_CODE_DECODING_FAILED; | ||||
| import static com.google.android.exoplayer2.PlaybackException.ERROR_CODE_UNSPECIFIED; | ||||
|  | ||||
| /** | ||||
|  * Outsourced logic for crashing the player in the {@link VideoDetailFragment}. | ||||
|  */ | ||||
| @@ -43,50 +42,34 @@ public final class VideoDetailPlayerCrasher { | ||||
|     // https://stackoverflow.com/a/54744028 | ||||
|     private static final String TAG = "VideoDetPlayerCrasher"; | ||||
|  | ||||
|     private static final Map<String, Supplier<ExoPlaybackException>> AVAILABLE_EXCEPTION_TYPES = | ||||
|             getExceptionTypes(); | ||||
|     private static final String DEFAULT_MSG = "Dummy"; | ||||
|  | ||||
|     private static final List<Pair<String, Supplier<ExoPlaybackException>>> | ||||
|             AVAILABLE_EXCEPTION_TYPES = List.of( | ||||
|                     new Pair<>("Source", () -> ExoPlaybackException.createForSource( | ||||
|                             new IOException(DEFAULT_MSG), | ||||
|                             ERROR_CODE_BEHIND_LIVE_WINDOW | ||||
|                     )), | ||||
|                     new Pair<>("Renderer", () -> ExoPlaybackException.createForRenderer( | ||||
|                             new Exception(DEFAULT_MSG), | ||||
|                             "Dummy renderer", | ||||
|                             0, | ||||
|                             null, | ||||
|                             C.FORMAT_HANDLED, | ||||
|                             /*isRecoverable=*/false, | ||||
|                             ERROR_CODE_DECODING_FAILED | ||||
|                     )), | ||||
|                     new Pair<>("Unexpected", () -> ExoPlaybackException.createForUnexpected( | ||||
|                             new RuntimeException(DEFAULT_MSG), | ||||
|                             ERROR_CODE_UNSPECIFIED | ||||
|                     )), | ||||
|                     new Pair<>("Remote", () -> ExoPlaybackException.createForRemote(DEFAULT_MSG)) | ||||
|             ); | ||||
|  | ||||
|     private VideoDetailPlayerCrasher() { | ||||
|         // No impls | ||||
|     } | ||||
|  | ||||
|     private static Map<String, Supplier<ExoPlaybackException>> getExceptionTypes() { | ||||
|         final String defaultMsg = "Dummy"; | ||||
|         final Map<String, Supplier<ExoPlaybackException>> exceptionTypes = new LinkedHashMap<>(); | ||||
|         exceptionTypes.put( | ||||
|                 "Source", | ||||
|                 () -> ExoPlaybackException.createForSource( | ||||
|                         new IOException(defaultMsg), | ||||
|                         ERROR_CODE_BEHIND_LIVE_WINDOW | ||||
|                 ) | ||||
|         ); | ||||
|         exceptionTypes.put( | ||||
|                 "Renderer", | ||||
|                 () -> ExoPlaybackException.createForRenderer( | ||||
|                         new Exception(defaultMsg), | ||||
|                         "Dummy renderer", | ||||
|                         0, | ||||
|                         null, | ||||
|                         C.FORMAT_HANDLED, | ||||
|                         /*isRecoverable=*/false, | ||||
|                         ERROR_CODE_DECODING_FAILED | ||||
|                 ) | ||||
|         ); | ||||
|         exceptionTypes.put( | ||||
|                 "Unexpected", | ||||
|                 () -> ExoPlaybackException.createForUnexpected( | ||||
|                         new RuntimeException(defaultMsg), | ||||
|                         ERROR_CODE_UNSPECIFIED | ||||
|                 ) | ||||
|         ); | ||||
|         exceptionTypes.put( | ||||
|                 "Remote", | ||||
|                 () -> ExoPlaybackException.createForRemote(defaultMsg) | ||||
|         ); | ||||
|  | ||||
|         return Collections.unmodifiableMap(exceptionTypes); | ||||
|     } | ||||
|  | ||||
|     private static Context getThemeWrapperContext(final Context context) { | ||||
|         return new ContextThemeWrapper( | ||||
|                 context, | ||||
| @@ -97,8 +80,7 @@ public final class VideoDetailPlayerCrasher { | ||||
|  | ||||
|     public static void onCrashThePlayer( | ||||
|             @NonNull final Context context, | ||||
|             @Nullable final Player player, | ||||
|             @NonNull final LayoutInflater layoutInflater | ||||
|             @Nullable final Player player | ||||
|     ) { | ||||
|         if (player == null) { | ||||
|             Log.d(TAG, "Player is not available"); | ||||
| @@ -109,24 +91,22 @@ public final class VideoDetailPlayerCrasher { | ||||
|         } | ||||
|  | ||||
|         // -- Build the dialog/UI -- | ||||
|  | ||||
|         final Context themeWrapperContext = getThemeWrapperContext(context); | ||||
|  | ||||
|         final LayoutInflater inflater = LayoutInflater.from(themeWrapperContext); | ||||
|         final RadioGroup radioGroup = SingleChoiceDialogViewBinding.inflate(layoutInflater) | ||||
|                 .list; | ||||
|  | ||||
|         final AlertDialog alertDialog = new AlertDialog.Builder(getThemeWrapperContext(context)) | ||||
|         final SingleChoiceDialogViewBinding binding = | ||||
|                 SingleChoiceDialogViewBinding.inflate(inflater); | ||||
|  | ||||
|         final AlertDialog alertDialog = new AlertDialog.Builder(themeWrapperContext) | ||||
|                 .setTitle("Choose an exception") | ||||
|                 .setView(radioGroup) | ||||
|                 .setView(binding.getRoot()) | ||||
|                 .setCancelable(true) | ||||
|                 .setNegativeButton(R.string.cancel, null) | ||||
|                 .create(); | ||||
|  | ||||
|         for (final Map.Entry<String, Supplier<ExoPlaybackException>> entry | ||||
|                 : AVAILABLE_EXCEPTION_TYPES.entrySet()) { | ||||
|         for (final Pair<String, Supplier<ExoPlaybackException>> entry : AVAILABLE_EXCEPTION_TYPES) { | ||||
|             final RadioButton radioButton = ListRadioIconItemBinding.inflate(inflater).getRoot(); | ||||
|             radioButton.setText(entry.getKey()); | ||||
|             radioButton.setText(entry.first); | ||||
|             radioButton.setChecked(false); | ||||
|             radioButton.setLayoutParams( | ||||
|                     new RadioGroup.LayoutParams( | ||||
| @@ -135,12 +115,10 @@ public final class VideoDetailPlayerCrasher { | ||||
|                     ) | ||||
|             ); | ||||
|             radioButton.setOnClickListener(v -> { | ||||
|                 tryCrashPlayerWith(player, entry.getValue().get()); | ||||
|                 if (alertDialog != null) { | ||||
|                     alertDialog.cancel(); | ||||
|                 } | ||||
|                 tryCrashPlayerWith(player, entry.second.get()); | ||||
|                 alertDialog.cancel(); | ||||
|             }); | ||||
|             radioGroup.addView(radioButton); | ||||
|             binding.list.addView(radioButton); | ||||
|         } | ||||
|  | ||||
|         alertDialog.show(); | ||||
|   | ||||
| @@ -5,7 +5,6 @@ import static org.schabi.newpipe.ktx.ViewUtils.animateHideRecyclerViewAllowingSc | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.content.SharedPreferences; | ||||
| import android.content.res.Configuration; | ||||
| import android.content.res.Resources; | ||||
| import android.os.Bundle; | ||||
| import android.util.Log; | ||||
| @@ -23,17 +22,16 @@ import androidx.recyclerview.widget.RecyclerView; | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.error.ErrorUtil; | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfoItem; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.extractor.playlist.PlaylistInfoItem; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.fragments.BaseStateFragment; | ||||
| import org.schabi.newpipe.fragments.OnScrollBelowItemsListener; | ||||
| import org.schabi.newpipe.info_list.dialog.InfoItemDialog; | ||||
| import org.schabi.newpipe.info_list.InfoListAdapter; | ||||
| import org.schabi.newpipe.info_list.ItemViewMode; | ||||
| import org.schabi.newpipe.info_list.dialog.InfoItemDialog; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.OnClickGesture; | ||||
| import org.schabi.newpipe.util.StateSaver; | ||||
| import org.schabi.newpipe.util.ThemeHelper; | ||||
| import org.schabi.newpipe.views.SuperScrollLayoutManager; | ||||
|  | ||||
| import java.util.List; | ||||
| @@ -94,11 +92,7 @@ public abstract class BaseListFragment<I, N> extends BaseStateFragment<I> | ||||
|  | ||||
|         if (updateFlags != 0) { | ||||
|             if ((updateFlags & LIST_MODE_UPDATE_FLAG) != 0) { | ||||
|                 final boolean useGrid = isGridLayout(); | ||||
|                 itemsList.setLayoutManager(useGrid | ||||
|                         ? getGridLayoutManager() : getListLayoutManager()); | ||||
|                 infoListAdapter.setUseGridVariant(useGrid); | ||||
|                 infoListAdapter.notifyDataSetChanged(); | ||||
|                 refreshItemViewMode(); | ||||
|             } | ||||
|             updateFlags = 0; | ||||
|         } | ||||
| @@ -218,22 +212,29 @@ public abstract class BaseListFragment<I, N> extends BaseStateFragment<I> | ||||
|         final Resources resources = activity.getResources(); | ||||
|         int width = resources.getDimensionPixelSize(R.dimen.video_item_grid_thumbnail_image_width); | ||||
|         width += (24 * resources.getDisplayMetrics().density); | ||||
|         final int spanCount = (int) Math.floor(resources.getDisplayMetrics().widthPixels | ||||
|                 / (double) width); | ||||
|         final int spanCount = Math.floorDiv(resources.getDisplayMetrics().widthPixels, width); | ||||
|         final GridLayoutManager lm = new GridLayoutManager(activity, spanCount); | ||||
|         lm.setSpanSizeLookup(infoListAdapter.getSpanSizeLookup(spanCount)); | ||||
|         return lm; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Updates the item view mode based on user preference. | ||||
|      */ | ||||
|     private void refreshItemViewMode() { | ||||
|         final ItemViewMode itemViewMode = getItemViewMode(); | ||||
|         itemsList.setLayoutManager((itemViewMode == ItemViewMode.GRID) | ||||
|                 ? getGridLayoutManager() : getListLayoutManager()); | ||||
|         infoListAdapter.setItemViewMode(itemViewMode); | ||||
|         infoListAdapter.notifyDataSetChanged(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void initViews(final View rootView, final Bundle savedInstanceState) { | ||||
|         super.initViews(rootView, savedInstanceState); | ||||
|  | ||||
|         final boolean useGrid = isGridLayout(); | ||||
|         itemsList = rootView.findViewById(R.id.items_list); | ||||
|         itemsList.setLayoutManager(useGrid ? getGridLayoutManager() : getListLayoutManager()); | ||||
|  | ||||
|         infoListAdapter.setUseGridVariant(useGrid); | ||||
|         refreshItemViewMode(); | ||||
|  | ||||
|         final Supplier<View> listHeaderSupplier = getListHeaderSupplier(); | ||||
|         if (listHeaderSupplier != null) { | ||||
| @@ -264,45 +265,28 @@ public abstract class BaseListFragment<I, N> extends BaseStateFragment<I> | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         infoListAdapter.setOnChannelSelectedListener(new OnClickGesture<>() { | ||||
|             @Override | ||||
|             public void selected(final ChannelInfoItem selectedItem) { | ||||
|                 try { | ||||
|                     onItemSelected(selectedItem); | ||||
|                     NavigationHelper.openChannelFragment(getFM(), | ||||
|                             selectedItem.getServiceId(), | ||||
|                             selectedItem.getUrl(), | ||||
|                             selectedItem.getName()); | ||||
|                 } catch (final Exception e) { | ||||
|                     ErrorUtil.showUiErrorSnackbar( | ||||
|                             BaseListFragment.this, "Opening channel fragment", e); | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         infoListAdapter.setOnPlaylistSelectedListener(new OnClickGesture<>() { | ||||
|             @Override | ||||
|             public void selected(final PlaylistInfoItem selectedItem) { | ||||
|                 try { | ||||
|                     onItemSelected(selectedItem); | ||||
|                     NavigationHelper.openPlaylistFragment(getFM(), | ||||
|                             selectedItem.getServiceId(), | ||||
|                             selectedItem.getUrl(), | ||||
|                             selectedItem.getName()); | ||||
|                 } catch (final Exception e) { | ||||
|                     ErrorUtil.showUiErrorSnackbar(BaseListFragment.this, | ||||
|                             "Opening playlist fragment", e); | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         infoListAdapter.setOnCommentsSelectedListener(new OnClickGesture<>() { | ||||
|             @Override | ||||
|             public void selected(final CommentsInfoItem selectedItem) { | ||||
|         infoListAdapter.setOnChannelSelectedListener(selectedItem -> { | ||||
|             try { | ||||
|                 onItemSelected(selectedItem); | ||||
|                 NavigationHelper.openChannelFragment(getFM(), selectedItem.getServiceId(), | ||||
|                         selectedItem.getUrl(), selectedItem.getName()); | ||||
|             } catch (final Exception e) { | ||||
|                 ErrorUtil.showUiErrorSnackbar(this, "Opening channel fragment", e); | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         infoListAdapter.setOnPlaylistSelectedListener(selectedItem -> { | ||||
|             try { | ||||
|                 onItemSelected(selectedItem); | ||||
|                 NavigationHelper.openPlaylistFragment(getFM(), selectedItem.getServiceId(), | ||||
|                         selectedItem.getUrl(), selectedItem.getName()); | ||||
|             } catch (final Exception e) { | ||||
|                 ErrorUtil.showUiErrorSnackbar(this, "Opening playlist fragment", e); | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         infoListAdapter.setOnCommentsSelectedListener(this::onItemSelected); | ||||
|  | ||||
|         // Ensure that there is always a scroll listener (e.g. when rotating the device) | ||||
|         useNormalItemListScrollListener(); | ||||
|     } | ||||
| @@ -490,21 +474,16 @@ public abstract class BaseListFragment<I, N> extends BaseStateFragment<I> | ||||
|     @Override | ||||
|     public void onSharedPreferenceChanged(final SharedPreferences sharedPreferences, | ||||
|                                           final String key) { | ||||
|         if (key.equals(getString(R.string.list_view_mode_key))) { | ||||
|         if (getString(R.string.list_view_mode_key).equals(key)) { | ||||
|             updateFlags |= LIST_MODE_UPDATE_FLAG; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     protected boolean isGridLayout() { | ||||
|         final String listMode = PreferenceManager.getDefaultSharedPreferences(activity) | ||||
|                 .getString(getString(R.string.list_view_mode_key), | ||||
|                         getString(R.string.list_view_mode_value)); | ||||
|         if ("auto".equals(listMode)) { | ||||
|             final Configuration configuration = getResources().getConfiguration(); | ||||
|             return configuration.orientation == Configuration.ORIENTATION_LANDSCAPE | ||||
|                     && configuration.isLayoutSizeAtLeast(Configuration.SCREENLAYOUT_SIZE_LARGE); | ||||
|         } else { | ||||
|             return "grid".equals(listMode); | ||||
|         } | ||||
|     /** | ||||
|      * Returns preferred item view mode. | ||||
|      * @return ItemViewMode | ||||
|      */ | ||||
|     protected ItemViewMode getItemViewMode() { | ||||
|         return ThemeHelper.getItemViewMode(requireContext()); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,7 @@ | ||||
| package org.schabi.newpipe.fragments.list; | ||||
|  | ||||
| import static org.schabi.newpipe.extractor.ServiceList.SoundCloud; | ||||
|  | ||||
| import android.os.Bundle; | ||||
| import android.text.TextUtils; | ||||
| import android.util.Log; | ||||
| @@ -7,13 +9,13 @@ import android.view.View; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.error.ErrorInfo; | ||||
| import org.schabi.newpipe.error.UserAction; | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.ListExtractor; | ||||
| import org.schabi.newpipe.extractor.ListInfo; | ||||
| import org.schabi.newpipe.extractor.Page; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfo; | ||||
| import org.schabi.newpipe.extractor.exceptions.ContentNotSupportedException; | ||||
| import org.schabi.newpipe.util.Constants; | ||||
| import org.schabi.newpipe.views.NewPipeRecyclerView; | ||||
| @@ -229,13 +231,11 @@ public abstract class BaseListInfoFragment<I extends InfoItem, L extends ListInf | ||||
|             if (!result.getRelatedItems().isEmpty()) { | ||||
|                 infoListAdapter.addInfoItemList(result.getRelatedItems()); | ||||
|                 showListFooter(hasMoreItems()); | ||||
|             } else if (hasMoreItems()) { | ||||
|                 loadMoreItems(); | ||||
|             } else { | ||||
|                 infoListAdapter.clearStreamItemList(); | ||||
|                 // showEmptyState should be called only if there is no item as | ||||
|                 // well as no header in infoListAdapter | ||||
|                 if (!(result instanceof ChannelInfo && infoListAdapter.getItemCount() == 1)) { | ||||
|                     showEmptyState(); | ||||
|                 } | ||||
|                 showEmptyState(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -252,6 +252,20 @@ public abstract class BaseListInfoFragment<I extends InfoItem, L extends ListInf | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void showEmptyState() { | ||||
|         // show "no streams" for SoundCloud; otherwise "no videos" | ||||
|         // showing "no live streams" is handled in KioskFragment | ||||
|         if (emptyStateView != null) { | ||||
|             if (currentInfo.getService() == SoundCloud) { | ||||
|                 setEmptyStateMessage(R.string.no_streams); | ||||
|             } else { | ||||
|                 setEmptyStateMessage(R.string.no_videos); | ||||
|             } | ||||
|         } | ||||
|         super.showEmptyState(); | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Utils | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|   | ||||
| @@ -0,0 +1,91 @@ | ||||
| package org.schabi.newpipe.fragments.list.channel; | ||||
|  | ||||
| import static org.schabi.newpipe.extractor.stream.StreamExtractor.UNKNOWN_SUBSCRIBER_COUNT; | ||||
|  | ||||
| import android.os.Bundle; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.widget.LinearLayout; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.extractor.StreamingService; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfo; | ||||
| import org.schabi.newpipe.extractor.stream.Description; | ||||
| import org.schabi.newpipe.fragments.detail.BaseDescriptionFragment; | ||||
| import org.schabi.newpipe.util.DeviceUtils; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| import icepick.State; | ||||
|  | ||||
| public class ChannelAboutFragment extends BaseDescriptionFragment { | ||||
|     @State | ||||
|     protected ChannelInfo channelInfo; | ||||
|  | ||||
|     ChannelAboutFragment(@NonNull final ChannelInfo channelInfo) { | ||||
|         this.channelInfo = channelInfo; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     @Override | ||||
|     protected void initViews(final View rootView, final Bundle savedInstanceState) { | ||||
|         super.initViews(rootView, savedInstanceState); | ||||
|         binding.constraintLayout.setPadding(0, DeviceUtils.dpToPx(8, requireContext()), 0, 0); | ||||
|     } | ||||
|  | ||||
|     @Nullable | ||||
|     @Override | ||||
|     protected Description getDescription() { | ||||
|         return new Description(channelInfo.getDescription(), Description.PLAIN_TEXT); | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     protected StreamingService getService() { | ||||
|         return channelInfo.getService(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected int getServiceId() { | ||||
|         return channelInfo.getServiceId(); | ||||
|     } | ||||
|  | ||||
|     @Nullable | ||||
|     @Override | ||||
|     protected String getStreamUrl() { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     @NonNull | ||||
|     @Override | ||||
|     public List<String> getTags() { | ||||
|         return channelInfo.getTags(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void setupMetadata(final LayoutInflater inflater, | ||||
|                                  final LinearLayout layout) { | ||||
|         // There is no upload date available for channels, so hide the relevant UI element | ||||
|         binding.detailUploadDateView.setVisibility(View.GONE); | ||||
|  | ||||
|         if (channelInfo == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (channelInfo.getSubscriberCount() != UNKNOWN_SUBSCRIBER_COUNT) { | ||||
|             addMetadataItem(inflater, layout, false, R.string.metadata_subscribers, | ||||
|                     Localization.localizeNumber( | ||||
|                             requireContext(), | ||||
|                             channelInfo.getSubscriberCount())); | ||||
|         } | ||||
|  | ||||
|         addImagesMetadataItem(inflater, layout, R.string.metadata_avatars, | ||||
|                 channelInfo.getAvatars()); | ||||
|         addImagesMetadataItem(inflater, layout, R.string.metadata_banners, | ||||
|                 channelInfo.getBanners()); | ||||
|     } | ||||
| } | ||||
| @@ -5,6 +5,7 @@ import static org.schabi.newpipe.ktx.ViewUtils.animate; | ||||
| import static org.schabi.newpipe.ktx.ViewUtils.animateBackgroundColor; | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.content.SharedPreferences; | ||||
| import android.graphics.Color; | ||||
| import android.os.Bundle; | ||||
| import android.text.TextUtils; | ||||
| @@ -16,51 +17,50 @@ import android.view.MenuInflater; | ||||
| import android.view.MenuItem; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.Button; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.appcompat.app.ActionBar; | ||||
| import androidx.core.content.ContextCompat; | ||||
| import androidx.core.graphics.ColorUtils; | ||||
| import androidx.preference.PreferenceManager; | ||||
|  | ||||
| import com.google.android.material.snackbar.Snackbar; | ||||
| import com.google.android.material.tabs.TabLayout; | ||||
| import com.jakewharton.rxbinding4.view.RxView; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.database.subscription.NotificationMode; | ||||
| import org.schabi.newpipe.database.subscription.SubscriptionEntity; | ||||
| import org.schabi.newpipe.databinding.ChannelHeaderBinding; | ||||
| import org.schabi.newpipe.databinding.FragmentChannelBinding; | ||||
| import org.schabi.newpipe.databinding.PlaylistControlBinding; | ||||
| import org.schabi.newpipe.error.ErrorInfo; | ||||
| import org.schabi.newpipe.error.ErrorUtil; | ||||
| import org.schabi.newpipe.error.UserAction; | ||||
| import org.schabi.newpipe.extractor.ListExtractor; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfo; | ||||
| import org.schabi.newpipe.extractor.exceptions.ContentNotSupportedException; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.fragments.list.BaseListInfoFragment; | ||||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandler; | ||||
| import org.schabi.newpipe.fragments.BaseStateFragment; | ||||
| import org.schabi.newpipe.fragments.detail.TabAdapter; | ||||
| import org.schabi.newpipe.ktx.AnimationType; | ||||
| import org.schabi.newpipe.local.subscription.SubscriptionManager; | ||||
| import org.schabi.newpipe.local.feed.notifications.NotificationHelper; | ||||
| import org.schabi.newpipe.player.MainPlayer.PlayerType; | ||||
| import org.schabi.newpipe.player.playqueue.ChannelPlayQueue; | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueue; | ||||
| import org.schabi.newpipe.local.subscription.SubscriptionManager; | ||||
| import org.schabi.newpipe.util.ChannelTabHelper; | ||||
| import org.schabi.newpipe.util.Constants; | ||||
| import org.schabi.newpipe.util.ExtractorHelper; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.PicassoHelper; | ||||
| import org.schabi.newpipe.util.StateSaver; | ||||
| import org.schabi.newpipe.util.image.ImageStrategy; | ||||
| import org.schabi.newpipe.util.image.PicassoHelper; | ||||
| import org.schabi.newpipe.util.ThemeHelper; | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Queue; | ||||
| import java.util.concurrent.TimeUnit; | ||||
| import java.util.function.Supplier; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| import icepick.State; | ||||
| import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers; | ||||
| import io.reactivex.rxjava3.core.Observable; | ||||
| import io.reactivex.rxjava3.core.Single; | ||||
| import io.reactivex.rxjava3.disposables.CompositeDisposable; | ||||
| import io.reactivex.rxjava3.disposables.Disposable; | ||||
| import io.reactivex.rxjava3.functions.Action; | ||||
| @@ -68,27 +68,37 @@ import io.reactivex.rxjava3.functions.Consumer; | ||||
| import io.reactivex.rxjava3.functions.Function; | ||||
| import io.reactivex.rxjava3.schedulers.Schedulers; | ||||
|  | ||||
| public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, ChannelInfo> | ||||
|         implements View.OnClickListener { | ||||
| public class ChannelFragment extends BaseStateFragment<ChannelInfo> | ||||
|         implements StateSaver.WriteRead { | ||||
|  | ||||
|     private static final int BUTTON_DEBOUNCE_INTERVAL = 100; | ||||
|     private static final String PICASSO_CHANNEL_TAG = "PICASSO_CHANNEL_TAG"; | ||||
|  | ||||
|     @State | ||||
|     protected int serviceId = Constants.NO_SERVICE_ID; | ||||
|     @State | ||||
|     protected String name; | ||||
|     @State | ||||
|     protected String url; | ||||
|  | ||||
|     private ChannelInfo currentInfo; | ||||
|     private Disposable currentWorker; | ||||
|     private final CompositeDisposable disposables = new CompositeDisposable(); | ||||
|     private Disposable subscribeButtonMonitor; | ||||
|     private SubscriptionManager subscriptionManager; | ||||
|     private int lastTab; | ||||
|     private boolean channelContentNotSupported = false; | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Views | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     private SubscriptionManager subscriptionManager; | ||||
|  | ||||
|     private FragmentChannelBinding channelBinding; | ||||
|     private ChannelHeaderBinding headerBinding; | ||||
|     private PlaylistControlBinding playlistControlBinding; | ||||
|     private FragmentChannelBinding binding; | ||||
|     private TabAdapter tabAdapter; | ||||
|  | ||||
|     private MenuItem menuRssButton; | ||||
|     private MenuItem menuNotifyButton; | ||||
|     private SubscriptionEntity channelSubscription; | ||||
|  | ||||
|     public static ChannelFragment getInstance(final int serviceId, final String url, | ||||
|                                               final String name) { | ||||
| @@ -97,22 +107,23 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|         return instance; | ||||
|     } | ||||
|  | ||||
|     public ChannelFragment() { | ||||
|         super(UserAction.REQUESTED_CHANNEL); | ||||
|     private void setInitialData(final int sid, final String u, final String title) { | ||||
|         this.serviceId = sid; | ||||
|         this.url = u; | ||||
|         this.name = !TextUtils.isEmpty(title) ? title : ""; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onResume() { | ||||
|         super.onResume(); | ||||
|         if (activity != null && useAsFrontPage) { | ||||
|             setTitle(currentInfo != null ? currentInfo.getName() : name); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // LifeCycle | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     public void onCreate(final Bundle savedInstanceState) { | ||||
|         super.onCreate(savedInstanceState); | ||||
|         setHasOptionsMenu(true); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onAttach(@NonNull final Context context) { | ||||
|         super.onAttach(context); | ||||
| @@ -123,48 +134,58 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, | ||||
|                              @Nullable final ViewGroup container, | ||||
|                              @Nullable final Bundle savedInstanceState) { | ||||
|         return inflater.inflate(R.layout.fragment_channel, container, false); | ||||
|         binding = FragmentChannelBinding.inflate(inflater, container, false); | ||||
|         return binding.getRoot(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onViewCreated(@NonNull final View rootView, final Bundle savedInstanceState) { | ||||
|         super.onViewCreated(rootView, savedInstanceState); | ||||
|         channelBinding = FragmentChannelBinding.bind(rootView); | ||||
|     } | ||||
|     @Override // called from onViewCreated in BaseFragment.onViewCreated | ||||
|     protected void initViews(final View rootView, final Bundle savedInstanceState) { | ||||
|         super.initViews(rootView, savedInstanceState); | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroy() { | ||||
|         super.onDestroy(); | ||||
|         disposables.clear(); | ||||
|         if (subscribeButtonMonitor != null) { | ||||
|             subscribeButtonMonitor.dispose(); | ||||
|         tabAdapter = new TabAdapter(getChildFragmentManager()); | ||||
|         binding.viewPager.setAdapter(tabAdapter); | ||||
|         binding.tabLayout.setupWithViewPager(binding.viewPager); | ||||
|  | ||||
|         setTitle(name); | ||||
|         binding.channelTitleView.setText(name); | ||||
|         if (!ImageStrategy.shouldLoadImages()) { | ||||
|             // do not waste space for the banner if it is not going to be loaded | ||||
|             binding.channelBannerImage.setImageDrawable(null); | ||||
|         } | ||||
|         channelBinding = null; | ||||
|         headerBinding = null; | ||||
|         playlistControlBinding = null; | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Init | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     protected Supplier<View> getListHeaderSupplier() { | ||||
|         headerBinding = ChannelHeaderBinding | ||||
|                 .inflate(activity.getLayoutInflater(), itemsList, false); | ||||
|         playlistControlBinding = headerBinding.playlistControl; | ||||
|  | ||||
|         return headerBinding::getRoot; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void initListeners() { | ||||
|         super.initListeners(); | ||||
|  | ||||
|         headerBinding.subChannelTitleView.setOnClickListener(this); | ||||
|         headerBinding.subChannelAvatarView.setOnClickListener(this); | ||||
|         final View.OnClickListener openSubChannel = v -> { | ||||
|             if (!TextUtils.isEmpty(currentInfo.getParentChannelUrl())) { | ||||
|                 try { | ||||
|                     NavigationHelper.openChannelFragment(getFM(), currentInfo.getServiceId(), | ||||
|                             currentInfo.getParentChannelUrl(), | ||||
|                             currentInfo.getParentChannelName()); | ||||
|                 } catch (final Exception e) { | ||||
|                     ErrorUtil.showUiErrorSnackbar(this, "Opening channel fragment", e); | ||||
|                 } | ||||
|             } else if (DEBUG) { | ||||
|                 Log.i(TAG, "Can't open parent channel because we got no channel URL"); | ||||
|             } | ||||
|         }; | ||||
|         binding.subChannelAvatarView.setOnClickListener(openSubChannel); | ||||
|         binding.subChannelTitleView.setOnClickListener(openSubChannel); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroy() { | ||||
|         super.onDestroy(); | ||||
|         if (currentWorker != null) { | ||||
|             currentWorker.dispose(); | ||||
|         } | ||||
|         disposables.clear(); | ||||
|         binding = null; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Menu | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
| @@ -173,36 +194,36 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|     public void onCreateOptionsMenu(@NonNull final Menu menu, | ||||
|                                     @NonNull final MenuInflater inflater) { | ||||
|         super.onCreateOptionsMenu(menu, inflater); | ||||
|         final ActionBar supportActionBar = activity.getSupportActionBar(); | ||||
|         if (useAsFrontPage && supportActionBar != null) { | ||||
|             supportActionBar.setDisplayHomeAsUpEnabled(false); | ||||
|         } else { | ||||
|             inflater.inflate(R.menu.menu_channel, menu); | ||||
|         inflater.inflate(R.menu.menu_channel, menu); | ||||
|  | ||||
|             if (DEBUG) { | ||||
|                 Log.d(TAG, "onCreateOptionsMenu() called with: " | ||||
|                         + "menu = [" + menu + "], inflater = [" + inflater + "]"); | ||||
|             } | ||||
|             menuRssButton = menu.findItem(R.id.menu_item_rss); | ||||
|             menuNotifyButton = menu.findItem(R.id.menu_item_notify); | ||||
|         if (DEBUG) { | ||||
|             Log.d(TAG, "onCreateOptionsMenu() called with: " | ||||
|                     + "menu = [" + menu + "], inflater = [" + inflater + "]"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean onOptionsItemSelected(final MenuItem item) { | ||||
|     public void onPrepareOptionsMenu(@NonNull final Menu menu) { | ||||
|         super.onPrepareOptionsMenu(menu); | ||||
|         menuRssButton = menu.findItem(R.id.menu_item_rss); | ||||
|         menuNotifyButton = menu.findItem(R.id.menu_item_notify); | ||||
|         updateNotifyButton(channelSubscription); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public boolean onOptionsItemSelected(@NonNull final MenuItem item) { | ||||
|         switch (item.getItemId()) { | ||||
|             case R.id.action_settings: | ||||
|                 NavigationHelper.openSettings(requireContext()); | ||||
|                 break; | ||||
|             case R.id.menu_item_notify: | ||||
|                 final boolean value = !item.isChecked(); | ||||
|                 item.setEnabled(false); | ||||
|                 setNotify(value); | ||||
|                 break; | ||||
|             case R.id.action_settings: | ||||
|                 NavigationHelper.openSettings(requireContext()); | ||||
|                 break; | ||||
|             case R.id.menu_item_rss: | ||||
|                 if (currentInfo != null) { | ||||
|                     ShareUtils.openUrlInBrowser( | ||||
|                             requireContext(), currentInfo.getFeedUrl(), false); | ||||
|                     ShareUtils.openUrlInApp(requireContext(), currentInfo.getFeedUrl()); | ||||
|                 } | ||||
|                 break; | ||||
|             case R.id.menu_item_openInBrowser: | ||||
| @@ -213,7 +234,7 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|             case R.id.menu_item_share: | ||||
|                 if (currentInfo != null) { | ||||
|                     ShareUtils.shareText(requireContext(), name, currentInfo.getOriginalUrl(), | ||||
|                             currentInfo.getAvatarUrl()); | ||||
|                             currentInfo.getAvatars()); | ||||
|                 } | ||||
|                 break; | ||||
|             default: | ||||
| @@ -222,13 +243,14 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Channel Subscription | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     private void monitorSubscription(final ChannelInfo info) { | ||||
|         final Consumer<Throwable> onError = (Throwable throwable) -> { | ||||
|             animate(headerBinding.channelSubscribeButton, false, 100); | ||||
|             animate(binding.channelSubscribeButton, false, 100); | ||||
|             showSnackBarError(new ErrorInfo(throwable, UserAction.SUBSCRIPTION_GET, | ||||
|                     "Get subscription status", currentInfo)); | ||||
|         }; | ||||
| @@ -261,10 +283,9 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|                 }, onError)); | ||||
|     } | ||||
|  | ||||
|     private Function<Object, Object> mapOnSubscribe(final SubscriptionEntity subscription, | ||||
|                                                     final ChannelInfo info) { | ||||
|     private Function<Object, Object> mapOnSubscribe(final SubscriptionEntity subscription) { | ||||
|         return (@NonNull Object o) -> { | ||||
|             subscriptionManager.insertSubscription(subscription, info); | ||||
|             subscriptionManager.insertSubscription(subscription); | ||||
|             return o; | ||||
|         }; | ||||
|     } | ||||
| @@ -296,8 +317,7 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|                 .subscribe(onComplete, onError)); | ||||
|     } | ||||
|  | ||||
|     private Disposable monitorSubscribeButton(final Button subscribeButton, | ||||
|                                               final Function<Object, Object> action) { | ||||
|     private Disposable monitorSubscribeButton(final Function<Object, Object> action) { | ||||
|         final Consumer<Object> onNext = (@NonNull Object o) -> { | ||||
|             if (DEBUG) { | ||||
|                 Log.d(TAG, "Changed subscription status to this channel!"); | ||||
| @@ -309,7 +329,7 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|                         "Changing subscription for " + currentInfo.getUrl(), currentInfo)); | ||||
|  | ||||
|         /* Emit clicks from main thread unto io thread */ | ||||
|         return RxView.clicks(subscribeButton) | ||||
|         return RxView.clicks(binding.channelSubscribeButton) | ||||
|                 .subscribeOn(AndroidSchedulers.mainThread()) | ||||
|                 .observeOn(Schedulers.io()) | ||||
|                 .debounce(BUTTON_DEBOUNCE_INTERVAL, TimeUnit.MILLISECONDS) // Ignore rapid clicks | ||||
| @@ -335,20 +355,20 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|                 channel.setServiceId(info.getServiceId()); | ||||
|                 channel.setUrl(info.getUrl()); | ||||
|                 channel.setData(info.getName(), | ||||
|                         info.getAvatarUrl(), | ||||
|                         ImageStrategy.imageListToDbUrl(info.getAvatars()), | ||||
|                         info.getDescription(), | ||||
|                         info.getSubscriberCount()); | ||||
|                 channelSubscription = null; | ||||
|                 updateNotifyButton(null); | ||||
|                 subscribeButtonMonitor = monitorSubscribeButton( | ||||
|                         headerBinding.channelSubscribeButton, mapOnSubscribe(channel, info)); | ||||
|                 subscribeButtonMonitor = monitorSubscribeButton(mapOnSubscribe(channel)); | ||||
|             } else { | ||||
|                 if (DEBUG) { | ||||
|                     Log.d(TAG, "Found subscription to this channel!"); | ||||
|                 } | ||||
|                 final SubscriptionEntity subscription = subscriptionEntities.get(0); | ||||
|                 updateNotifyButton(subscription); | ||||
|                 subscribeButtonMonitor = monitorSubscribeButton( | ||||
|                         headerBinding.channelSubscribeButton, mapOnUnsubscribe(subscription)); | ||||
|                 channelSubscription = subscriptionEntities.get(0); | ||||
|                 updateNotifyButton(channelSubscription); | ||||
|                 subscribeButtonMonitor = | ||||
|                         monitorSubscribeButton(mapOnUnsubscribe(channelSubscription)); | ||||
|             } | ||||
|         }; | ||||
|     } | ||||
| @@ -359,34 +379,33 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|                     + "isSubscribed = [" + isSubscribed + "]"); | ||||
|         } | ||||
|  | ||||
|         final boolean isButtonVisible = headerBinding.channelSubscribeButton.getVisibility() | ||||
|         final boolean isButtonVisible = binding.channelSubscribeButton.getVisibility() | ||||
|                 == View.VISIBLE; | ||||
|         final int backgroundDuration = isButtonVisible ? 300 : 0; | ||||
|         final int textDuration = isButtonVisible ? 200 : 0; | ||||
|  | ||||
|         final int subscribeBackground = ThemeHelper | ||||
|                 .resolveColorFromAttr(activity, R.attr.colorPrimary); | ||||
|         final int subscribeText = ContextCompat.getColor(activity, R.color.subscribe_text_color); | ||||
|         final int subscribedBackground = ContextCompat | ||||
|                 .getColor(activity, R.color.subscribed_background_color); | ||||
|         final int subscribedText = ContextCompat.getColor(activity, R.color.subscribed_text_color); | ||||
|         final int subscribeBackground = ColorUtils.blendARGB(ThemeHelper | ||||
|                 .resolveColorFromAttr(activity, R.attr.colorPrimary), subscribedBackground, 0.35f); | ||||
|         final int subscribeText = ContextCompat.getColor(activity, R.color.subscribe_text_color); | ||||
|  | ||||
|         if (!isSubscribed) { | ||||
|             headerBinding.channelSubscribeButton.setText(R.string.subscribe_button_title); | ||||
|             animateBackgroundColor(headerBinding.channelSubscribeButton, backgroundDuration, | ||||
|                     subscribedBackground, subscribeBackground); | ||||
|             animateTextColor(headerBinding.channelSubscribeButton, textDuration, subscribedText, | ||||
|                     subscribeText); | ||||
|         } else { | ||||
|             headerBinding.channelSubscribeButton.setText(R.string.subscribed_button_title); | ||||
|             animateBackgroundColor(headerBinding.channelSubscribeButton, backgroundDuration, | ||||
|         if (isSubscribed) { | ||||
|             binding.channelSubscribeButton.setText(R.string.subscribed_button_title); | ||||
|             animateBackgroundColor(binding.channelSubscribeButton, backgroundDuration, | ||||
|                     subscribeBackground, subscribedBackground); | ||||
|             animateTextColor(headerBinding.channelSubscribeButton, textDuration, subscribeText, | ||||
|             animateTextColor(binding.channelSubscribeButton, textDuration, subscribeText, | ||||
|                     subscribedText); | ||||
|         } else { | ||||
|             binding.channelSubscribeButton.setText(R.string.subscribe_button_title); | ||||
|             animateBackgroundColor(binding.channelSubscribeButton, backgroundDuration, | ||||
|                     subscribedBackground, subscribeBackground); | ||||
|             animateTextColor(binding.channelSubscribeButton, textDuration, subscribedText, | ||||
|                     subscribeText); | ||||
|         } | ||||
|  | ||||
|         animate(headerBinding.channelSubscribeButton, true, 100, | ||||
|                 AnimationType.LIGHT_SCALE_AND_ALPHA); | ||||
|         animate(binding.channelSubscribeButton, true, 100, AnimationType.LIGHT_SCALE_AND_ALPHA); | ||||
|     } | ||||
|  | ||||
|     private void updateNotifyButton(@Nullable final SubscriptionEntity subscription) { | ||||
| @@ -422,111 +441,185 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|      * Show a snackbar with the option to enable notifications on new streams for this channel. | ||||
|      */ | ||||
|     private void showNotifySnackbar() { | ||||
|         Snackbar.make(itemsList, R.string.you_successfully_subscribed, Snackbar.LENGTH_LONG) | ||||
|         Snackbar.make(binding.getRoot(), R.string.you_successfully_subscribed, Snackbar.LENGTH_LONG) | ||||
|                 .setAction(R.string.get_notified, v -> setNotify(true)) | ||||
|                 .setActionTextColor(Color.YELLOW) | ||||
|                 .show(); | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Load and handle | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     protected Single<ListExtractor.InfoItemsPage<StreamInfoItem>> loadMoreItemsLogic() { | ||||
|         return ExtractorHelper.getMoreChannelItems(serviceId, url, currentNextPage); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Single<ChannelInfo> loadResult(final boolean forceLoad) { | ||||
|         return ExtractorHelper.getChannelInfo(serviceId, url, forceLoad); | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // OnClick | ||||
|     // Init | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     public void onClick(final View v) { | ||||
|         if (isLoading.get() || currentInfo == null) { | ||||
|             return; | ||||
|         } | ||||
|     private void updateTabs() { | ||||
|         tabAdapter.clearAllItems(); | ||||
|  | ||||
|         switch (v.getId()) { | ||||
|             case R.id.sub_channel_avatar_view: | ||||
|             case R.id.sub_channel_title_view: | ||||
|                 if (!TextUtils.isEmpty(currentInfo.getParentChannelUrl())) { | ||||
|                     try { | ||||
|                         NavigationHelper.openChannelFragment(getFM(), currentInfo.getServiceId(), | ||||
|                                 currentInfo.getParentChannelUrl(), | ||||
|                                 currentInfo.getParentChannelName()); | ||||
|                     } catch (final Exception e) { | ||||
|                         ErrorUtil.showUiErrorSnackbar(this, "Opening channel fragment", e); | ||||
|                     } | ||||
|                 } else if (DEBUG) { | ||||
|                     Log.i(TAG, "Can't open parent channel because we got no channel URL"); | ||||
|         if (currentInfo != null && !channelContentNotSupported) { | ||||
|             final Context context = requireContext(); | ||||
|             final SharedPreferences preferences = PreferenceManager | ||||
|                     .getDefaultSharedPreferences(context); | ||||
|  | ||||
|             for (final ListLinkHandler linkHandler : currentInfo.getTabs()) { | ||||
|                 final String tab = linkHandler.getContentFilters().get(0); | ||||
|                 if (ChannelTabHelper.showChannelTab(context, preferences, tab)) { | ||||
|                     final ChannelTabFragment channelTabFragment = | ||||
|                             ChannelTabFragment.getInstance(serviceId, linkHandler, name); | ||||
|                     channelTabFragment.useAsFrontPage(useAsFrontPage); | ||||
|                     tabAdapter.addFragment(channelTabFragment, | ||||
|                             context.getString(ChannelTabHelper.getTranslationKey(tab))); | ||||
|                 } | ||||
|                 break; | ||||
|             } | ||||
|  | ||||
|             if (ChannelTabHelper.showChannelTab( | ||||
|                     context, preferences, R.string.show_channel_tabs_about)) { | ||||
|                 tabAdapter.addFragment( | ||||
|                         new ChannelAboutFragment(currentInfo), | ||||
|                         context.getString(R.string.channel_tab_about)); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         tabAdapter.notifyDataSetUpdate(); | ||||
|  | ||||
|         for (int i = 0; i < tabAdapter.getCount(); i++) { | ||||
|             binding.tabLayout.getTabAt(i).setText(tabAdapter.getItemTitle(i)); | ||||
|         } | ||||
|  | ||||
|         // Restore previously selected tab | ||||
|         final TabLayout.Tab ltab = binding.tabLayout.getTabAt(lastTab); | ||||
|         if (ltab != null) { | ||||
|             binding.tabLayout.selectTab(ltab); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // State Saving | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     public String generateSuffix() { | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void writeTo(final Queue<Object> objectsToSave) { | ||||
|         objectsToSave.add(currentInfo); | ||||
|         objectsToSave.add(binding == null ? 0 : binding.tabLayout.getSelectedTabPosition()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void readFrom(@NonNull final Queue<Object> savedObjects) { | ||||
|         currentInfo = (ChannelInfo) savedObjects.poll(); | ||||
|         lastTab = (Integer) savedObjects.poll(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onSaveInstanceState(final @NonNull Bundle outState) { | ||||
|         super.onSaveInstanceState(outState); | ||||
|         if (binding != null) { | ||||
|             outState.putInt("LastTab", binding.tabLayout.getSelectedTabPosition()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void onRestoreInstanceState(@NonNull final Bundle savedInstanceState) { | ||||
|         super.onRestoreInstanceState(savedInstanceState); | ||||
|         lastTab = savedInstanceState.getInt("LastTab", 0); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Contract | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     protected void doInitialLoadLogic() { | ||||
|         if (currentInfo == null) { | ||||
|             startLoading(false); | ||||
|         } else { | ||||
|             handleResult(currentInfo); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void startLoading(final boolean forceLoad) { | ||||
|         super.startLoading(forceLoad); | ||||
|  | ||||
|         currentInfo = null; | ||||
|         updateTabs(); | ||||
|         if (currentWorker != null) { | ||||
|             currentWorker.dispose(); | ||||
|         } | ||||
|  | ||||
|         runWorker(forceLoad); | ||||
|     } | ||||
|  | ||||
|     private void runWorker(final boolean forceLoad) { | ||||
|         currentWorker = ExtractorHelper.getChannelInfo(serviceId, url, forceLoad) | ||||
|                 .subscribeOn(Schedulers.io()) | ||||
|                 .observeOn(AndroidSchedulers.mainThread()) | ||||
|                 .subscribe(result -> { | ||||
|                     isLoading.set(false); | ||||
|                     handleResult(result); | ||||
|                 }, throwable -> showError(new ErrorInfo(throwable, UserAction.REQUESTED_CHANNEL, | ||||
|                         url == null ? "No URL" : url, serviceId))); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void showLoading() { | ||||
|         super.showLoading(); | ||||
|         PicassoHelper.cancelTag(PICASSO_CHANNEL_TAG); | ||||
|         animate(headerBinding.channelSubscribeButton, false, 100); | ||||
|         animate(binding.channelSubscribeButton, false, 100); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void handleResult(@NonNull final ChannelInfo result) { | ||||
|         super.handleResult(result); | ||||
|         currentInfo = result; | ||||
|         setInitialData(result.getServiceId(), result.getOriginalUrl(), result.getName()); | ||||
|  | ||||
|         headerBinding.getRoot().setVisibility(View.VISIBLE); | ||||
|         PicassoHelper.loadBanner(result.getBannerUrl()).tag(PICASSO_CHANNEL_TAG) | ||||
|                 .into(headerBinding.channelBannerImage); | ||||
|         PicassoHelper.loadAvatar(result.getAvatarUrl()).tag(PICASSO_CHANNEL_TAG) | ||||
|                 .into(headerBinding.channelAvatarView); | ||||
|         PicassoHelper.loadAvatar(result.getParentChannelAvatarUrl()).tag(PICASSO_CHANNEL_TAG) | ||||
|                 .into(headerBinding.subChannelAvatarView); | ||||
|         if (ImageStrategy.shouldLoadImages() && !result.getBanners().isEmpty()) { | ||||
|             PicassoHelper.loadBanner(result.getBanners()).tag(PICASSO_CHANNEL_TAG) | ||||
|                     .into(binding.channelBannerImage); | ||||
|         } else { | ||||
|             // do not waste space for the banner, if the user disabled images or there is not one | ||||
|             binding.channelBannerImage.setImageDrawable(null); | ||||
|         } | ||||
|  | ||||
|         headerBinding.channelSubscriberView.setVisibility(View.VISIBLE); | ||||
|         PicassoHelper.loadAvatar(result.getAvatars()).tag(PICASSO_CHANNEL_TAG) | ||||
|                 .into(binding.channelAvatarView); | ||||
|         PicassoHelper.loadAvatar(result.getParentChannelAvatars()).tag(PICASSO_CHANNEL_TAG) | ||||
|                 .into(binding.subChannelAvatarView); | ||||
|  | ||||
|         binding.channelTitleView.setText(result.getName()); | ||||
|         binding.channelSubscriberView.setVisibility(View.VISIBLE); | ||||
|         if (result.getSubscriberCount() >= 0) { | ||||
|             headerBinding.channelSubscriberView.setText(Localization | ||||
|             binding.channelSubscriberView.setText(Localization | ||||
|                     .shortSubscriberCount(activity, result.getSubscriberCount())); | ||||
|         } else { | ||||
|             headerBinding.channelSubscriberView.setText(R.string.subscribers_count_not_available); | ||||
|             binding.channelSubscriberView.setText(R.string.subscribers_count_not_available); | ||||
|         } | ||||
|  | ||||
|         if (!TextUtils.isEmpty(currentInfo.getParentChannelName())) { | ||||
|             headerBinding.subChannelTitleView.setText(String.format( | ||||
|             binding.subChannelTitleView.setText(String.format( | ||||
|                     getString(R.string.channel_created_by), | ||||
|                     currentInfo.getParentChannelName()) | ||||
|             ); | ||||
|             headerBinding.subChannelTitleView.setVisibility(View.VISIBLE); | ||||
|             headerBinding.subChannelAvatarView.setVisibility(View.VISIBLE); | ||||
|         } else { | ||||
|             headerBinding.subChannelTitleView.setVisibility(View.GONE); | ||||
|             binding.subChannelTitleView.setVisibility(View.VISIBLE); | ||||
|             binding.subChannelAvatarView.setVisibility(View.VISIBLE); | ||||
|         } | ||||
|  | ||||
|         if (menuRssButton != null) { | ||||
|             menuRssButton.setVisible(!TextUtils.isEmpty(result.getFeedUrl())); | ||||
|         } | ||||
|  | ||||
|         // PlaylistControls should be visible only if there is some item in | ||||
|         // infoListAdapter other than header | ||||
|         if (infoListAdapter.getItemCount() != 1) { | ||||
|             playlistControlBinding.getRoot().setVisibility(View.VISIBLE); | ||||
|         } else { | ||||
|             playlistControlBinding.getRoot().setVisibility(View.GONE); | ||||
|         } | ||||
|  | ||||
|         channelContentNotSupported = false; | ||||
|         for (final Throwable throwable : result.getErrors()) { | ||||
|             if (throwable instanceof ContentNotSupportedException) { | ||||
|                 showContentNotSupported(); | ||||
|                 channelContentNotSupported = true; | ||||
|                 showContentNotSupportedIfNeeded(); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
| @@ -534,60 +627,21 @@ public class ChannelFragment extends BaseListInfoFragment<StreamInfoItem, Channe | ||||
|         if (subscribeButtonMonitor != null) { | ||||
|             subscribeButtonMonitor.dispose(); | ||||
|         } | ||||
|  | ||||
|         updateTabs(); | ||||
|         updateSubscription(result); | ||||
|         monitorSubscription(result); | ||||
|  | ||||
|         playlistControlBinding.playlistCtrlPlayAllButton | ||||
|                 .setOnClickListener(view -> NavigationHelper | ||||
|                         .playOnMainPlayer(activity, getPlayQueue())); | ||||
|         playlistControlBinding.playlistCtrlPlayPopupButton | ||||
|                 .setOnClickListener(view -> NavigationHelper | ||||
|                         .playOnPopupPlayer(activity, getPlayQueue(), false)); | ||||
|         playlistControlBinding.playlistCtrlPlayBgButton | ||||
|                 .setOnClickListener(view -> NavigationHelper | ||||
|                         .playOnBackgroundPlayer(activity, getPlayQueue(), false)); | ||||
|  | ||||
|         playlistControlBinding.playlistCtrlPlayPopupButton.setOnLongClickListener(view -> { | ||||
|             NavigationHelper.enqueueOnPlayer(activity, getPlayQueue(), PlayerType.POPUP); | ||||
|             return true; | ||||
|         }); | ||||
|  | ||||
|         playlistControlBinding.playlistCtrlPlayBgButton.setOnLongClickListener(view -> { | ||||
|             NavigationHelper.enqueueOnPlayer(activity, getPlayQueue(), PlayerType.AUDIO); | ||||
|             return true; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     private void showContentNotSupported() { | ||||
|         channelBinding.errorContentNotSupported.setVisibility(View.VISIBLE); | ||||
|         channelBinding.channelKaomoji.setText("(︶︹︺)"); | ||||
|         channelBinding.channelKaomoji.setTextSize(TypedValue.COMPLEX_UNIT_SP, 45f); | ||||
|         channelBinding.channelNoVideos.setVisibility(View.GONE); | ||||
|     } | ||||
|  | ||||
|     private PlayQueue getPlayQueue() { | ||||
|         return getPlayQueue(0); | ||||
|     } | ||||
|  | ||||
|     private PlayQueue getPlayQueue(final int index) { | ||||
|         final List<StreamInfoItem> streamItems = infoListAdapter.getItemsList().stream() | ||||
|                 .filter(StreamInfoItem.class::isInstance) | ||||
|                 .map(StreamInfoItem.class::cast) | ||||
|                 .collect(Collectors.toList()); | ||||
|  | ||||
|         return new ChannelPlayQueue(currentInfo.getServiceId(), currentInfo.getUrl(), | ||||
|                 currentInfo.getNextPage(), streamItems, index); | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Utils | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     public void setTitle(final String title) { | ||||
|         super.setTitle(title); | ||||
|         if (!useAsFrontPage) { | ||||
|             headerBinding.channelTitleView.setText(title); | ||||
|     private void showContentNotSupportedIfNeeded() { | ||||
|         // channelBinding might not be initialized when handleResult() is called | ||||
|         // (e.g. after rotating the screen, #6696) | ||||
|         if (!channelContentNotSupported || binding == null) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         binding.errorContentNotSupported.setVisibility(View.VISIBLE); | ||||
|         binding.channelKaomoji.setText("(︶︹︺)"); | ||||
|         binding.channelKaomoji.setTextSize(TypedValue.COMPLEX_UNIT_SP, 45f); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,168 @@ | ||||
| package org.schabi.newpipe.fragments.list.channel; | ||||
|  | ||||
| import android.os.Bundle; | ||||
| import android.util.Log; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.databinding.PlaylistControlBinding; | ||||
| import org.schabi.newpipe.error.UserAction; | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.ListExtractor; | ||||
| import org.schabi.newpipe.extractor.channel.tabs.ChannelTabInfo; | ||||
| import org.schabi.newpipe.extractor.exceptions.ParsingException; | ||||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandler; | ||||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandlerFactory; | ||||
| import org.schabi.newpipe.extractor.linkhandler.ReadyChannelTabListLinkHandler; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.fragments.list.BaseListInfoFragment; | ||||
| import org.schabi.newpipe.fragments.list.playlist.PlaylistControlViewHolder; | ||||
| import org.schabi.newpipe.player.playqueue.ChannelTabPlayQueue; | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueue; | ||||
| import org.schabi.newpipe.util.ChannelTabHelper; | ||||
| import org.schabi.newpipe.util.ExtractorHelper; | ||||
| import org.schabi.newpipe.util.PlayButtonHelper; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.function.Supplier; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| import icepick.State; | ||||
| import io.reactivex.rxjava3.core.Single; | ||||
|  | ||||
| public class ChannelTabFragment extends BaseListInfoFragment<InfoItem, ChannelTabInfo> | ||||
|         implements PlaylistControlViewHolder { | ||||
|  | ||||
|     // states must be protected and not private for IcePick being able to access them | ||||
|     @State | ||||
|     protected ListLinkHandler tabHandler; | ||||
|     @State | ||||
|     protected String channelName; | ||||
|  | ||||
|     private PlaylistControlBinding playlistControlBinding; | ||||
|  | ||||
|     @NonNull | ||||
|     public static ChannelTabFragment getInstance(final int serviceId, | ||||
|                                                  final ListLinkHandler tabHandler, | ||||
|                                                  final String channelName) { | ||||
|         final ChannelTabFragment instance = new ChannelTabFragment(); | ||||
|         instance.serviceId = serviceId; | ||||
|         instance.tabHandler = tabHandler; | ||||
|         instance.channelName = channelName; | ||||
|         return instance; | ||||
|     } | ||||
|  | ||||
|     public ChannelTabFragment() { | ||||
|         super(UserAction.REQUESTED_CHANNEL); | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // LifeCycle | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     public void onCreate(final Bundle savedInstanceState) { | ||||
|         super.onCreate(savedInstanceState); | ||||
|         setHasOptionsMenu(false); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, | ||||
|                              @Nullable final ViewGroup container, | ||||
|                              @Nullable final Bundle savedInstanceState) { | ||||
|         return inflater.inflate(R.layout.fragment_channel_tab, container, false); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroyView() { | ||||
|         super.onDestroyView(); | ||||
|         playlistControlBinding = null; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Supplier<View> getListHeaderSupplier() { | ||||
|         if (ChannelTabHelper.isStreamsTab(tabHandler)) { | ||||
|             playlistControlBinding = PlaylistControlBinding | ||||
|                     .inflate(activity.getLayoutInflater(), itemsList, false); | ||||
|             return playlistControlBinding::getRoot; | ||||
|         } | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Single<ChannelTabInfo> loadResult(final boolean forceLoad) { | ||||
|         return ExtractorHelper.getChannelTab(serviceId, tabHandler, forceLoad); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Single<ListExtractor.InfoItemsPage<InfoItem>> loadMoreItemsLogic() { | ||||
|         return ExtractorHelper.getMoreChannelTabItems(serviceId, tabHandler, currentNextPage); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void setTitle(final String title) { | ||||
|         // The channel name is displayed as title in the toolbar. | ||||
|         // The title is always a description of the content of the tab fragment. | ||||
|         // It should be unique for each channel because multiple channel tabs | ||||
|         // can be added to the main page. Therefore, the channel name is used. | ||||
|         // Using the title variable would cause the title to be the same for all channel tabs. | ||||
|         super.setTitle(channelName); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void handleResult(@NonNull final ChannelTabInfo result) { | ||||
|         super.handleResult(result); | ||||
|  | ||||
|         // FIXME this is a really hacky workaround, to avoid storing useless data in the fragment | ||||
|         //  state. The problem is, `ReadyChannelTabListLinkHandler` might contain raw JSON data that | ||||
|         //  uses a lot of memory (e.g. ~800KB for YouTube). While 800KB doesn't seem much, if | ||||
|         //  you combine just a couple of channel tab fragments you easily go over the 1MB | ||||
|         //  save&restore transaction limit, and get `TransactionTooLargeException`s. A proper | ||||
|         //  solution would require rethinking about `ReadyChannelTabListLinkHandler`s. | ||||
|         if (tabHandler instanceof ReadyChannelTabListLinkHandler) { | ||||
|             try { | ||||
|                 // once `handleResult` is called, the parsed data was already saved to cache, so | ||||
|                 // we can discard any raw data in ReadyChannelTabListLinkHandler and create a | ||||
|                 // link handler with identical properties, but without any raw data | ||||
|                 final ListLinkHandlerFactory channelTabLHFactory = result.getService() | ||||
|                         .getChannelTabLHFactory(); | ||||
|                 if (channelTabLHFactory != null) { | ||||
|                     // some services do not not have a ChannelTabLHFactory | ||||
|                     tabHandler = channelTabLHFactory.fromQuery(tabHandler.getId(), | ||||
|                             tabHandler.getContentFilters(), tabHandler.getSortFilter()); | ||||
|                 } | ||||
|             } catch (final ParsingException e) { | ||||
|                 // silently ignore the error, as the app can continue to function normally | ||||
|                 Log.w(TAG, "Could not recreate channel tab handler", e); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (playlistControlBinding != null) { | ||||
|             // PlaylistControls should be visible only if there is some item in | ||||
|             // infoListAdapter other than header | ||||
|             if (infoListAdapter.getItemCount() > 1) { | ||||
|                 playlistControlBinding.getRoot().setVisibility(View.VISIBLE); | ||||
|             } else { | ||||
|                 playlistControlBinding.getRoot().setVisibility(View.GONE); | ||||
|             } | ||||
|  | ||||
|             PlayButtonHelper.initPlaylistControlClickListener( | ||||
|                     activity, playlistControlBinding, this); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public PlayQueue getPlayQueue() { | ||||
|         final List<StreamInfoItem> streamItems = infoListAdapter.getItemsList().stream() | ||||
|                 .filter(StreamInfoItem.class::isInstance) | ||||
|                 .map(StreamInfoItem.class::cast) | ||||
|                 .collect(Collectors.toList()); | ||||
|  | ||||
|         return new ChannelTabPlayQueue(currentInfo.getServiceId(), tabHandler, | ||||
|                 currentInfo.getNextPage(), streamItems, 0); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,168 @@ | ||||
| package org.schabi.newpipe.fragments.list.comments; | ||||
|  | ||||
| import static org.schabi.newpipe.util.ServiceHelper.getServiceById; | ||||
|  | ||||
| import android.os.Bundle; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.constraintlayout.widget.ConstraintLayout; | ||||
| import androidx.core.text.HtmlCompat; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.databinding.CommentRepliesHeaderBinding; | ||||
| import org.schabi.newpipe.error.UserAction; | ||||
| import org.schabi.newpipe.extractor.ListExtractor; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.fragments.list.BaseListInfoFragment; | ||||
| import org.schabi.newpipe.info_list.ItemViewMode; | ||||
| import org.schabi.newpipe.util.DeviceUtils; | ||||
| import org.schabi.newpipe.util.ExtractorHelper; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.image.ImageStrategy; | ||||
| import org.schabi.newpipe.util.image.PicassoHelper; | ||||
| import org.schabi.newpipe.util.text.TextLinkifier; | ||||
|  | ||||
| import java.util.Queue; | ||||
| import java.util.function.Supplier; | ||||
|  | ||||
| import io.reactivex.rxjava3.core.Single; | ||||
| import io.reactivex.rxjava3.disposables.CompositeDisposable; | ||||
|  | ||||
| public final class CommentRepliesFragment | ||||
|         extends BaseListInfoFragment<CommentsInfoItem, CommentRepliesInfo> { | ||||
|  | ||||
|     public static final String TAG = CommentRepliesFragment.class.getSimpleName(); | ||||
|  | ||||
|     private CommentsInfoItem commentsInfoItem; // the comment to show replies of | ||||
|     private final CompositeDisposable disposables = new CompositeDisposable(); | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Constructors and lifecycle | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     // only called by the Android framework, after which readFrom is called and restores all data | ||||
|     public CommentRepliesFragment() { | ||||
|         super(UserAction.REQUESTED_COMMENT_REPLIES); | ||||
|     } | ||||
|  | ||||
|     public CommentRepliesFragment(@NonNull final CommentsInfoItem commentsInfoItem) { | ||||
|         this(); | ||||
|         this.commentsInfoItem = commentsInfoItem; | ||||
|         // setting "" as title since the title will be properly set right after | ||||
|         setInitialData(commentsInfoItem.getServiceId(), commentsInfoItem.getUrl(), ""); | ||||
|     } | ||||
|  | ||||
|     @Nullable | ||||
|     @Override | ||||
|     public View onCreateView(@NonNull final LayoutInflater inflater, | ||||
|                              @Nullable final ViewGroup container, | ||||
|                              @Nullable final Bundle savedInstanceState) { | ||||
|         return inflater.inflate(R.layout.fragment_comments, container, false); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDestroyView() { | ||||
|         disposables.clear(); | ||||
|         super.onDestroyView(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Supplier<View> getListHeaderSupplier() { | ||||
|         return () -> { | ||||
|             final CommentRepliesHeaderBinding binding = CommentRepliesHeaderBinding | ||||
|                     .inflate(activity.getLayoutInflater(), itemsList, false); | ||||
|             final CommentsInfoItem item = commentsInfoItem; | ||||
|  | ||||
|             // load the author avatar | ||||
|             PicassoHelper.loadAvatar(item.getUploaderAvatars()).into(binding.authorAvatar); | ||||
|             binding.authorAvatar.setVisibility(ImageStrategy.shouldLoadImages() | ||||
|                     ? View.VISIBLE : View.GONE); | ||||
|  | ||||
|             // setup author name and comment date | ||||
|             binding.authorName.setText(item.getUploaderName()); | ||||
|             binding.uploadDate.setText(Localization.relativeTimeOrTextual( | ||||
|                     getContext(), item.getUploadDate(), item.getTextualUploadDate())); | ||||
|             binding.authorTouchArea.setOnClickListener( | ||||
|                     v -> NavigationHelper.openCommentAuthorIfPresent(requireActivity(), item)); | ||||
|  | ||||
|             // setup like count, hearted and pinned | ||||
|             binding.thumbsUpCount.setText( | ||||
|                     Localization.likeCount(requireContext(), item.getLikeCount())); | ||||
|             // for heartImage goneMarginEnd was used, but there is no way to tell ConstraintLayout | ||||
|             // not to use a different margin only when both the next two views are gone | ||||
|             ((ConstraintLayout.LayoutParams) binding.thumbsUpCount.getLayoutParams()) | ||||
|                     .setMarginEnd(DeviceUtils.dpToPx( | ||||
|                             (item.isHeartedByUploader() || item.isPinned() ? 8 : 16), | ||||
|                             requireContext())); | ||||
|             binding.heartImage.setVisibility(item.isHeartedByUploader() ? View.VISIBLE : View.GONE); | ||||
|             binding.pinnedImage.setVisibility(item.isPinned() ? View.VISIBLE : View.GONE); | ||||
|  | ||||
|             // setup comment content | ||||
|             TextLinkifier.fromDescription(binding.commentContent, item.getCommentText(), | ||||
|                     HtmlCompat.FROM_HTML_MODE_LEGACY, getServiceById(item.getServiceId()), | ||||
|                     item.getUrl(), disposables, null); | ||||
|  | ||||
|             return binding.getRoot(); | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // State saving | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     public void writeTo(final Queue<Object> objectsToSave) { | ||||
|         super.writeTo(objectsToSave); | ||||
|         objectsToSave.add(commentsInfoItem); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void readFrom(@NonNull final Queue<Object> savedObjects) throws Exception { | ||||
|         super.readFrom(savedObjects); | ||||
|         commentsInfoItem = (CommentsInfoItem) savedObjects.poll(); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Data loading | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     protected Single<CommentRepliesInfo> loadResult(final boolean forceLoad) { | ||||
|         return Single.fromCallable(() -> new CommentRepliesInfo(commentsInfoItem, | ||||
|                 // the reply count string will be shown as the activity title | ||||
|                 Localization.replyCount(requireContext(), commentsInfoItem.getReplyCount()))); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Single<ListExtractor.InfoItemsPage<CommentsInfoItem>> loadMoreItemsLogic() { | ||||
|         // commentsInfoItem.getUrl() should contain the url of the original | ||||
|         // ListInfo<CommentsInfoItem>, which should be the stream url | ||||
|         return ExtractorHelper.getMoreCommentItems( | ||||
|                 serviceId, commentsInfoItem.getUrl(), currentNextPage); | ||||
|     } | ||||
|  | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Utils | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     protected ItemViewMode getItemViewMode() { | ||||
|         return ItemViewMode.LIST; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @return the comment to which the replies are shown | ||||
|      */ | ||||
|     public CommentsInfoItem getCommentsInfoItem() { | ||||
|         return commentsInfoItem; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,22 @@ | ||||
| package org.schabi.newpipe.fragments.list.comments; | ||||
|  | ||||
| import org.schabi.newpipe.extractor.ListInfo; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandler; | ||||
|  | ||||
| import java.util.Collections; | ||||
|  | ||||
| public final class CommentRepliesInfo extends ListInfo<CommentsInfoItem> { | ||||
|     /** | ||||
|      * This class is used to wrap the comment replies page into a ListInfo object. | ||||
|      * | ||||
|      * @param comment the comment from which to get replies | ||||
|      * @param name will be shown as the fragment title | ||||
|      */ | ||||
|     public CommentRepliesInfo(final CommentsInfoItem comment, final String name) { | ||||
|         super(comment.getServiceId(), | ||||
|                 new ListLinkHandler("", "", "", Collections.emptyList(), null), name); | ||||
|         setNextPage(comment.getReplies()); | ||||
|         setRelatedItems(Collections.emptyList()); // since it must be non-null | ||||
|     } | ||||
| } | ||||
| @@ -17,6 +17,7 @@ import org.schabi.newpipe.extractor.ListExtractor; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfo; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.fragments.list.BaseListInfoFragment; | ||||
| import org.schabi.newpipe.info_list.ItemViewMode; | ||||
| import org.schabi.newpipe.ktx.ViewUtils; | ||||
| import org.schabi.newpipe.util.ExtractorHelper; | ||||
|  | ||||
| @@ -106,7 +107,17 @@ public class CommentsFragment extends BaseListInfoFragment<CommentsInfoItem, Com | ||||
|                                     @NonNull final MenuInflater inflater) { } | ||||
|  | ||||
|     @Override | ||||
|     protected boolean isGridLayout() { | ||||
|         return false; | ||||
|     protected ItemViewMode getItemViewMode() { | ||||
|         return ItemViewMode.LIST; | ||||
|     } | ||||
|  | ||||
|     public boolean scrollToComment(final CommentsInfoItem comment) { | ||||
|         final int position = infoListAdapter.getItemsList().indexOf(comment); | ||||
|         if (position < 0) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         itemsList.scrollToPosition(position); | ||||
|         return true; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -16,11 +16,13 @@ import org.schabi.newpipe.error.ErrorInfo; | ||||
| import org.schabi.newpipe.error.UserAction; | ||||
| import org.schabi.newpipe.extractor.ListExtractor; | ||||
| import org.schabi.newpipe.extractor.NewPipe; | ||||
| import org.schabi.newpipe.extractor.ServiceList; | ||||
| import org.schabi.newpipe.extractor.StreamingService; | ||||
| import org.schabi.newpipe.extractor.exceptions.ExtractionException; | ||||
| import org.schabi.newpipe.extractor.kiosk.KioskInfo; | ||||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandlerFactory; | ||||
| import org.schabi.newpipe.extractor.localization.ContentCountry; | ||||
| import org.schabi.newpipe.extractor.services.media_ccc.extractors.MediaCCCLiveStreamKiosk; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.fragments.list.BaseListInfoFragment; | ||||
| import org.schabi.newpipe.util.ExtractorHelper; | ||||
| @@ -161,4 +163,14 @@ public class KioskFragment extends BaseListInfoFragment<StreamInfoItem, KioskInf | ||||
|         name = kioskTranslatedName; | ||||
|         setTitle(kioskTranslatedName); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void showEmptyState() { | ||||
|         // show "no live streams" for live stream kiosk | ||||
|         super.showEmptyState(); | ||||
|         if (MediaCCCLiveStreamKiosk.KIOSK_ID.equals(currentInfo.getId()) | ||||
|                 && ServiceList.MediaCCC.getServiceId() == currentInfo.getServiceId()) { | ||||
|             setEmptyStateMessage(R.string.no_live_streams); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,11 @@ | ||||
| package org.schabi.newpipe.fragments.list.playlist; | ||||
|  | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueue; | ||||
|  | ||||
| /** | ||||
|  * Interface for {@code R.layout.playlist_control} view holders | ||||
|  * to give access to the play queue. | ||||
|  */ | ||||
| public interface PlaylistControlViewHolder { | ||||
|     PlayQueue getPlayQueue(); | ||||
| } | ||||
| @@ -1,10 +1,11 @@ | ||||
| package org.schabi.newpipe.fragments.list.playlist; | ||||
|  | ||||
| import static org.schabi.newpipe.extractor.utils.Utils.isBlank; | ||||
| import static org.schabi.newpipe.ktx.ViewUtils.animate; | ||||
| import static org.schabi.newpipe.ktx.ViewUtils.animateHideRecyclerViewAllowingScrolling; | ||||
| import static org.schabi.newpipe.util.ServiceHelper.getServiceById; | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.content.res.ColorStateList; | ||||
| import android.os.Bundle; | ||||
| import android.text.TextUtils; | ||||
| import android.util.Log; | ||||
| @@ -18,7 +19,6 @@ import android.view.ViewGroup; | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.annotation.Nullable; | ||||
| import androidx.appcompat.content.res.AppCompatResources; | ||||
| import androidx.core.content.ContextCompat; | ||||
|  | ||||
| import com.google.android.material.shape.CornerFamily; | ||||
| import com.google.android.material.shape.ShapeAppearanceModel; | ||||
| @@ -28,6 +28,7 @@ import org.reactivestreams.Subscription; | ||||
| import org.schabi.newpipe.NewPipeDatabase; | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.database.playlist.model.PlaylistRemoteEntity; | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity; | ||||
| import org.schabi.newpipe.databinding.PlaylistControlBinding; | ||||
| import org.schabi.newpipe.databinding.PlaylistHeaderBinding; | ||||
| import org.schabi.newpipe.error.ErrorInfo; | ||||
| @@ -38,24 +39,28 @@ import org.schabi.newpipe.extractor.ListExtractor; | ||||
| import org.schabi.newpipe.extractor.ServiceList; | ||||
| import org.schabi.newpipe.extractor.playlist.PlaylistInfo; | ||||
| import org.schabi.newpipe.extractor.services.youtube.YoutubeParsingHelper; | ||||
| import org.schabi.newpipe.extractor.stream.Description; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.fragments.list.BaseListInfoFragment; | ||||
| import org.schabi.newpipe.info_list.dialog.InfoItemDialog; | ||||
| import org.schabi.newpipe.info_list.dialog.StreamDialogDefaultEntry; | ||||
| import org.schabi.newpipe.local.dialog.PlaylistDialog; | ||||
| import org.schabi.newpipe.local.playlist.RemotePlaylistManager; | ||||
| import org.schabi.newpipe.player.MainPlayer.PlayerType; | ||||
| import org.schabi.newpipe.player.playqueue.PlayQueue; | ||||
| import org.schabi.newpipe.player.playqueue.PlaylistPlayQueue; | ||||
| import org.schabi.newpipe.util.ExtractorHelper; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.PicassoHelper; | ||||
| import org.schabi.newpipe.info_list.dialog.StreamDialogDefaultEntry; | ||||
| import org.schabi.newpipe.util.PlayButtonHelper; | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils; | ||||
| import org.schabi.newpipe.util.image.PicassoHelper; | ||||
| import org.schabi.newpipe.util.text.TextEllipsizer; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.concurrent.atomic.AtomicBoolean; | ||||
| import java.util.function.Supplier; | ||||
| import java.util.stream.Collectors; | ||||
|  | ||||
| import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers; | ||||
| import io.reactivex.rxjava3.core.Flowable; | ||||
| @@ -63,7 +68,8 @@ import io.reactivex.rxjava3.core.Single; | ||||
| import io.reactivex.rxjava3.disposables.CompositeDisposable; | ||||
| import io.reactivex.rxjava3.disposables.Disposable; | ||||
|  | ||||
| public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, PlaylistInfo> { | ||||
| public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, PlaylistInfo> | ||||
|         implements PlaylistControlViewHolder { | ||||
|  | ||||
|     private static final String PICASSO_PLAYLIST_TAG = "PICASSO_PLAYLIST_TAG"; | ||||
|  | ||||
| @@ -83,6 +89,9 @@ public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, Playl | ||||
|  | ||||
|     private MenuItem playlistBookmarkButton; | ||||
|  | ||||
|     private long streamCount; | ||||
|     private long playlistOverallDurationSeconds; | ||||
|  | ||||
|     public static PlaylistFragment getInstance(final int serviceId, final String url, | ||||
|                                                final String name) { | ||||
|         final PlaylistFragment instance = new PlaylistFragment(); | ||||
| @@ -131,6 +140,8 @@ public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, Playl | ||||
|     protected void initViews(final View rootView, final Bundle savedInstanceState) { | ||||
|         super.initViews(rootView, savedInstanceState); | ||||
|  | ||||
|         // Is mini variant still relevant? | ||||
|         // Only the remote playlist screen uses it now | ||||
|         infoListAdapter.setUseMiniVariant(true); | ||||
|     } | ||||
|  | ||||
| @@ -229,14 +240,25 @@ public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, Playl | ||||
|                 ShareUtils.openUrlInBrowser(requireContext(), url); | ||||
|                 break; | ||||
|             case R.id.menu_item_share: | ||||
|                 if (currentInfo != null) { | ||||
|                     ShareUtils.shareText(requireContext(), name, url, | ||||
|                             currentInfo.getThumbnailUrl()); | ||||
|                 } | ||||
|                 ShareUtils.shareText(requireContext(), name, url, | ||||
|                         currentInfo == null ? List.of() : currentInfo.getThumbnails()); | ||||
|                 break; | ||||
|             case R.id.menu_item_bookmark: | ||||
|                 onBookmarkClicked(); | ||||
|                 break; | ||||
|             case R.id.menu_item_append_playlist: | ||||
|                 if (currentInfo != null) { | ||||
|                     disposables.add(PlaylistDialog.createCorrespondingDialog( | ||||
|                             getContext(), | ||||
|                             getPlayQueue() | ||||
|                                     .getStreams() | ||||
|                                     .stream() | ||||
|                                     .map(StreamEntity::new) | ||||
|                                     .collect(Collectors.toList()), | ||||
|                             dialog -> dialog.show(getFM(), TAG) | ||||
|                     )); | ||||
|                 } | ||||
|                 break; | ||||
|             default: | ||||
|                 return super.onOptionsItemSelected(item); | ||||
|         } | ||||
| @@ -258,6 +280,12 @@ public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, Playl | ||||
|         animate(headerBinding.uploaderLayout, false, 200); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void handleNextItems(final ListExtractor.InfoItemsPage result) { | ||||
|         super.handleNextItems(result); | ||||
|         setStreamCountAndOverallDuration(result.getItems(), !result.hasNextPage()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void handleResult(@NonNull final PlaylistInfo result) { | ||||
|         super.handleResult(result); | ||||
| @@ -284,7 +312,6 @@ public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, Playl | ||||
|  | ||||
|         playlistControlBinding.getRoot().setVisibility(View.VISIBLE); | ||||
|  | ||||
|         final String avatarUrl = result.getUploaderAvatarUrl(); | ||||
|         if (result.getServiceId() == ServiceList.YouTube.getServiceId() | ||||
|                 && (YoutubeParsingHelper.isYoutubeMixId(result.getId()) | ||||
|                 || YoutubeParsingHelper.isYoutubeMusicMixId(result.getId()))) { | ||||
| @@ -293,21 +320,42 @@ public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, Playl | ||||
|                     .setAllCorners(CornerFamily.ROUNDED, 0f) | ||||
|                     .build(); // this turns the image back into a square | ||||
|             headerBinding.uploaderAvatarView.setShapeAppearanceModel(model); | ||||
|             headerBinding.uploaderAvatarView.setStrokeColor( | ||||
|                     ColorStateList.valueOf(ContextCompat.getColor( | ||||
|                             requireContext(), R.color.transparent_background_color)) | ||||
|             ); | ||||
|             headerBinding.uploaderAvatarView.setStrokeColor(AppCompatResources | ||||
|                     .getColorStateList(requireContext(), R.color.transparent_background_color)); | ||||
|             headerBinding.uploaderAvatarView.setImageDrawable( | ||||
|                     AppCompatResources.getDrawable(requireContext(), | ||||
|                     R.drawable.ic_radio) | ||||
|             ); | ||||
|         } else { | ||||
|             PicassoHelper.loadAvatar(avatarUrl).tag(PICASSO_PLAYLIST_TAG) | ||||
|             PicassoHelper.loadAvatar(result.getUploaderAvatars()).tag(PICASSO_PLAYLIST_TAG) | ||||
|                     .into(headerBinding.uploaderAvatarView); | ||||
|         } | ||||
|  | ||||
|         headerBinding.playlistStreamCount.setText(Localization | ||||
|                 .localizeStreamCount(getContext(), result.getStreamCount())); | ||||
|         streamCount = result.getStreamCount(); | ||||
|         setStreamCountAndOverallDuration(result.getRelatedItems(), !result.hasNextPage()); | ||||
|  | ||||
|         final Description description = result.getDescription(); | ||||
|         if (description != null && description != Description.EMPTY_DESCRIPTION | ||||
|                 && !isBlank(description.getContent())) { | ||||
|             final TextEllipsizer ellipsizer = new TextEllipsizer( | ||||
|                     headerBinding.playlistDescription, 5, getServiceById(result.getServiceId())); | ||||
|             ellipsizer.setStateChangeListener(isEllipsized -> | ||||
|                 headerBinding.playlistDescriptionReadMore.setText( | ||||
|                         Boolean.TRUE.equals(isEllipsized) ? R.string.show_more : R.string.show_less | ||||
|                 )); | ||||
|             ellipsizer.setOnContentChanged(canBeEllipsized -> { | ||||
|                 headerBinding.playlistDescriptionReadMore.setVisibility( | ||||
|                         Boolean.TRUE.equals(canBeEllipsized) ? View.VISIBLE : View.GONE); | ||||
|                 if (Boolean.TRUE.equals(canBeEllipsized)) { | ||||
|                     ellipsizer.ellipsize(); | ||||
|                 } | ||||
|             }); | ||||
|             ellipsizer.setContent(description); | ||||
|             headerBinding.playlistDescriptionReadMore.setOnClickListener(v -> ellipsizer.toggle()); | ||||
|         } else { | ||||
|             headerBinding.playlistDescription.setVisibility(View.GONE); | ||||
|             headerBinding.playlistDescriptionReadMore.setVisibility(View.GONE); | ||||
|         } | ||||
|  | ||||
|         if (!result.getErrors().isEmpty()) { | ||||
|             showSnackBarError(new ErrorInfo(result.getErrors(), UserAction.REQUESTED_PLAYLIST, | ||||
| @@ -320,25 +368,10 @@ public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, Playl | ||||
|                 .observeOn(AndroidSchedulers.mainThread()) | ||||
|                 .subscribe(getPlaylistBookmarkSubscriber()); | ||||
|  | ||||
|         playlistControlBinding.playlistCtrlPlayAllButton.setOnClickListener(view -> | ||||
|                 NavigationHelper.playOnMainPlayer(activity, getPlayQueue())); | ||||
|         playlistControlBinding.playlistCtrlPlayPopupButton.setOnClickListener(view -> | ||||
|                 NavigationHelper.playOnPopupPlayer(activity, getPlayQueue(), false)); | ||||
|         playlistControlBinding.playlistCtrlPlayBgButton.setOnClickListener(view -> | ||||
|                 NavigationHelper.playOnBackgroundPlayer(activity, getPlayQueue(), false)); | ||||
|  | ||||
|         playlistControlBinding.playlistCtrlPlayPopupButton.setOnLongClickListener(view -> { | ||||
|             NavigationHelper.enqueueOnPlayer(activity, getPlayQueue(), PlayerType.POPUP); | ||||
|             return true; | ||||
|         }); | ||||
|  | ||||
|         playlistControlBinding.playlistCtrlPlayBgButton.setOnLongClickListener(view -> { | ||||
|             NavigationHelper.enqueueOnPlayer(activity, getPlayQueue(), PlayerType.AUDIO); | ||||
|             return true; | ||||
|         }); | ||||
|         PlayButtonHelper.initPlaylistControlClickListener(activity, playlistControlBinding, this); | ||||
|     } | ||||
|  | ||||
|     private PlayQueue getPlayQueue() { | ||||
|     public PlayQueue getPlayQueue() { | ||||
|         return getPlayQueue(0); | ||||
|     } | ||||
|  | ||||
| @@ -462,4 +495,20 @@ public class PlaylistFragment extends BaseListInfoFragment<StreamInfoItem, Playl | ||||
|         playlistBookmarkButton.setIcon(drawable); | ||||
|         playlistBookmarkButton.setTitle(titleRes); | ||||
|     } | ||||
|  | ||||
|     private void setStreamCountAndOverallDuration(final List<StreamInfoItem> list, | ||||
|                                                   final boolean isDurationComplete) { | ||||
|         if (activity != null && headerBinding != null) { | ||||
|             playlistOverallDurationSeconds += list.stream() | ||||
|                     .mapToLong(x -> x.getDuration()) | ||||
|                     .sum(); | ||||
|             headerBinding.playlistStreamCount.setText( | ||||
|                 Localization.concatenateStrings( | ||||
|                     Localization.localizeStreamCount(activity, streamCount), | ||||
|                     Localization.getDurationString(playlistOverallDurationSeconds, | ||||
|                             isDurationComplete)) | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package org.schabi.newpipe.fragments.list.search; | ||||
|  | ||||
| import static androidx.recyclerview.widget.ItemTouchHelper.Callback.makeMovementFlags; | ||||
| import static org.schabi.newpipe.extractor.utils.Utils.isBlank; | ||||
| import static org.schabi.newpipe.ktx.ViewUtils.animate; | ||||
| import static org.schabi.newpipe.util.ExtractorHelper.showMetaInfoInTextView; | ||||
| import static java.util.Arrays.asList; | ||||
| @@ -33,6 +34,7 @@ import androidx.annotation.Nullable; | ||||
| import androidx.appcompat.app.ActionBar; | ||||
| import androidx.appcompat.app.AlertDialog; | ||||
| import androidx.appcompat.widget.TooltipCompat; | ||||
| import androidx.collection.SparseArrayCompat; | ||||
| import androidx.core.text.HtmlCompat; | ||||
| import androidx.preference.PreferenceManager; | ||||
| import androidx.recyclerview.widget.ItemTouchHelper; | ||||
| @@ -70,9 +72,7 @@ import org.schabi.newpipe.util.ServiceHelper; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.Collections; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Queue; | ||||
| import java.util.concurrent.TimeUnit; | ||||
| import java.util.stream.Collectors; | ||||
| @@ -141,7 +141,7 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|     @State | ||||
|     boolean wasSearchFocused = false; | ||||
|  | ||||
|     @Nullable private Map<Integer, String> menuItemToFilterName = null; | ||||
|     private final SparseArrayCompat<String> menuItemToFilterName = new SparseArrayCompat<>(); | ||||
|     private StreamingService service; | ||||
|     private Page nextPage; | ||||
|     private boolean showLocalSuggestions = true; | ||||
| @@ -168,6 +168,10 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     /** | ||||
|      * TextWatcher to remove rich-text formatting on the search EditText when pasting content | ||||
|      * from the clipboard. | ||||
|      */ | ||||
|     private TextWatcher textWatcher; | ||||
|  | ||||
|     public static SearchFragment getInstance(final int serviceId, final String searchString) { | ||||
| @@ -200,7 +204,7 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|         showLocalSuggestions = NewPipeSettings.showLocalSearchSuggestions(activity, prefs); | ||||
|         showRemoteSuggestions = NewPipeSettings.showRemoteSearchSuggestions(activity, prefs); | ||||
|  | ||||
|         suggestionListAdapter = new SuggestionListAdapter(activity); | ||||
|         suggestionListAdapter = new SuggestionListAdapter(); | ||||
|         historyRecordManager = new HistoryRecordManager(context); | ||||
|     } | ||||
|  | ||||
| @@ -340,6 +344,8 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|         super.initViews(rootView, savedInstanceState); | ||||
|  | ||||
|         searchBinding.suggestionsList.setAdapter(suggestionListAdapter); | ||||
|         // animations are just strange and useless, since the suggestions keep changing too much | ||||
|         searchBinding.suggestionsList.setItemAnimator(null); | ||||
|         new ItemTouchHelper(new ItemTouchHelper.Callback() { | ||||
|             @Override | ||||
|             public int getMovementFlags(@NonNull final RecyclerView recyclerView, | ||||
| @@ -384,7 +390,7 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|     @Override | ||||
|     public void onSaveInstanceState(@NonNull final Bundle bundle) { | ||||
|         searchString = searchEditText != null | ||||
|                 ? searchEditText.getText().toString() | ||||
|                 ? getSearchEditString().trim() | ||||
|                 : searchString; | ||||
|         super.onSaveInstanceState(bundle); | ||||
|     } | ||||
| @@ -395,11 +401,11 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|  | ||||
|     @Override | ||||
|     public void reloadContent() { | ||||
|         if (!TextUtils.isEmpty(searchString) | ||||
|                 || (searchEditText != null && !TextUtils.isEmpty(searchEditText.getText()))) { | ||||
|         if (!TextUtils.isEmpty(searchString) || (searchEditText != null | ||||
|                 && !isSearchEditBlank())) { | ||||
|             search(!TextUtils.isEmpty(searchString) | ||||
|                     ? searchString | ||||
|                     : searchEditText.getText().toString(), this.contentFilter, ""); | ||||
|                     : getSearchEditString(), this.contentFilter, ""); | ||||
|         } else { | ||||
|             if (searchEditText != null) { | ||||
|                 searchEditText.setText(""); | ||||
| @@ -424,8 +430,6 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|             supportActionBar.setDisplayHomeAsUpEnabled(true); | ||||
|         } | ||||
|  | ||||
|         menuItemToFilterName = new HashMap<>(); | ||||
|  | ||||
|         int itemId = 0; | ||||
|         boolean isFirstItem = true; | ||||
|         final Context c = getContext(); | ||||
| @@ -466,11 +470,8 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|  | ||||
|     @Override | ||||
|     public boolean onOptionsItemSelected(@NonNull final MenuItem item) { | ||||
|         if (menuItemToFilterName != null) { | ||||
|             final List<String> cf = new ArrayList<>(1); | ||||
|             cf.add(menuItemToFilterName.get(item.getItemId())); | ||||
|             changeContentFilter(item, cf); | ||||
|         } | ||||
|         final var filter = Collections.singletonList(menuItemToFilterName.get(item.getItemId())); | ||||
|         changeContentFilter(item, filter); | ||||
|         return true; | ||||
|     } | ||||
|  | ||||
| @@ -497,11 +498,9 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|                     + lastSearchedString); | ||||
|         } | ||||
|         searchEditText.setText(searchString); | ||||
|         if (android.os.Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.LOLLIPOP) { | ||||
|             searchEditText.setHintTextColor(searchEditText.getTextColors().withAlpha(128)); | ||||
|         } | ||||
|  | ||||
|         if (TextUtils.isEmpty(searchString) || TextUtils.isEmpty(searchEditText.getText())) { | ||||
|         if (TextUtils.isEmpty(searchString) | ||||
|                 || isSearchEditBlank()) { | ||||
|             searchToolbarContainer.setTranslationX(100); | ||||
|             searchToolbarContainer.setAlpha(0.0f); | ||||
|             searchToolbarContainer.setVisibility(View.VISIBLE); | ||||
| @@ -525,7 +524,7 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|             if (DEBUG) { | ||||
|                 Log.d(TAG, "onClick() called with: v = [" + v + "]"); | ||||
|             } | ||||
|             if (TextUtils.isEmpty(searchEditText.getText())) { | ||||
|             if (isSearchEditBlank()) { | ||||
|                 NavigationHelper.gotoMainFragment(getFM()); | ||||
|                 return; | ||||
|             } | ||||
| @@ -533,7 +532,7 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|             searchBinding.correctSuggestion.setVisibility(View.GONE); | ||||
|  | ||||
|             searchEditText.setText(""); | ||||
|             suggestionListAdapter.setItems(new ArrayList<>()); | ||||
|             suggestionListAdapter.submitList(null); | ||||
|             showKeyboardSearch(); | ||||
|         }); | ||||
|  | ||||
| @@ -590,11 +589,13 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|             @Override | ||||
|             public void beforeTextChanged(final CharSequence s, final int start, | ||||
|                                           final int count, final int after) { | ||||
|                 // Do nothing, old text is already clean | ||||
|             } | ||||
|  | ||||
|             @Override | ||||
|             public void onTextChanged(final CharSequence s, final int start, | ||||
|                                       final int before, final int count) { | ||||
|                 // Changes are handled in afterTextChanged; CharSequence cannot be changed here. | ||||
|             } | ||||
|  | ||||
|             @Override | ||||
| @@ -604,7 +605,7 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|                     s.removeSpan(span); | ||||
|                 } | ||||
|  | ||||
|                 final String newText = searchEditText.getText().toString(); | ||||
|                 final String newText = getSearchEditString().trim(); | ||||
|                 suggestionPublisher.onNext(newText); | ||||
|             } | ||||
|         }; | ||||
| @@ -620,7 +621,8 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|                     } else if (event != null | ||||
|                             && (event.getKeyCode() == KeyEvent.KEYCODE_ENTER | ||||
|                             || event.getAction() == EditorInfo.IME_ACTION_SEARCH)) { | ||||
|                         search(searchEditText.getText().toString(), new String[0], ""); | ||||
|                         searchEditText.setText(getSearchEditString().trim()); | ||||
|                         search(getSearchEditString(), new String[0], ""); | ||||
|                         return true; | ||||
|                     } | ||||
|                     return false; | ||||
| @@ -695,7 +697,7 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|                             .observeOn(AndroidSchedulers.mainThread()) | ||||
|                             .subscribe( | ||||
|                                     howManyDeleted -> suggestionPublisher | ||||
|                                             .onNext(searchEditText.getText().toString()), | ||||
|                                             .onNext(getSearchEditString()), | ||||
|                                     throwable -> showSnackBarError(new ErrorInfo(throwable, | ||||
|                                             UserAction.DELETE_FROM_HISTORY, | ||||
|                                             "Deleting item failed"))); | ||||
| @@ -724,9 +726,9 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|                 .getRelatedSearches(query, similarQueryLimit, 25) | ||||
|                 .toObservable() | ||||
|                 .map(searchHistoryEntries -> | ||||
|                     searchHistoryEntries.stream() | ||||
|                             .map(entry -> new SuggestionItem(true, entry)) | ||||
|                             .collect(Collectors.toList())); | ||||
|                         searchHistoryEntries.stream() | ||||
|                                 .map(entry -> new SuggestionItem(true, entry)) | ||||
|                                 .collect(Collectors.toList())); | ||||
|     } | ||||
|  | ||||
|     private Observable<List<SuggestionItem>> getRemoteSuggestionsObservable(final String query) { | ||||
| @@ -793,12 +795,12 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|                             } else if (listNotification.isOnError() | ||||
|                                     && listNotification.getError() != null | ||||
|                                     && !ExceptionUtils.isInterruptedCaused( | ||||
|                                             listNotification.getError())) { | ||||
|                                     listNotification.getError())) { | ||||
|                                 showSnackBarError(new ErrorInfo(listNotification.getError(), | ||||
|                                         UserAction.GET_SUGGESTIONS, searchString, serviceId)); | ||||
|                             } | ||||
|                         }, throwable -> showSnackBarError(new ErrorInfo( | ||||
|                             throwable, UserAction.GET_SUGGESTIONS, searchString, serviceId))); | ||||
|                                 throwable, UserAction.GET_SUGGESTIONS, searchString, serviceId))); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -806,7 +808,13 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|         // no-op | ||||
|     } | ||||
|  | ||||
|     private void search(final String theSearchString, | ||||
|     /** | ||||
|      * Perform a search. | ||||
|      * @param theSearchString the trimmed search string | ||||
|      * @param theContentFilter the content filter to use. FIXME: unused param | ||||
|      * @param theSortFilter FIXME: unused param | ||||
|      */ | ||||
|     private void search(@NonNull final String theSearchString, | ||||
|                         final String[] theContentFilter, | ||||
|                         final String theSortFilter) { | ||||
|         if (DEBUG) { | ||||
| @@ -816,25 +824,26 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Check if theSearchString is a URL which can be opened by NewPipe directly | ||||
|         // and open it if possible. | ||||
|         try { | ||||
|             final StreamingService streamingService = NewPipe.getServiceByUrl(theSearchString); | ||||
|             if (streamingService != null) { | ||||
|                 showLoading(); | ||||
|                 disposables.add(Observable | ||||
|                         .fromCallable(() -> NavigationHelper.getIntentByLink(activity, | ||||
|                                 streamingService, theSearchString)) | ||||
|                         .subscribeOn(Schedulers.io()) | ||||
|                         .observeOn(AndroidSchedulers.mainThread()) | ||||
|                         .subscribe(intent -> { | ||||
|                             getFM().popBackStackImmediate(); | ||||
|                             activity.startActivity(intent); | ||||
|                         }, throwable -> showTextError(getString(R.string.unsupported_url)))); | ||||
|                 return; | ||||
|             } | ||||
|             showLoading(); | ||||
|             disposables.add(Observable | ||||
|                     .fromCallable(() -> NavigationHelper.getIntentByLink(activity, | ||||
|                             streamingService, theSearchString)) | ||||
|                     .subscribeOn(Schedulers.io()) | ||||
|                     .observeOn(AndroidSchedulers.mainThread()) | ||||
|                     .subscribe(intent -> { | ||||
|                         getFM().popBackStackImmediate(); | ||||
|                         activity.startActivity(intent); | ||||
|                     }, throwable -> showTextError(getString(R.string.unsupported_url)))); | ||||
|             return; | ||||
|         } catch (final Exception ignored) { | ||||
|             // Exception occurred, it's not a url | ||||
|         } | ||||
|  | ||||
|         // prepare search | ||||
|         lastSearchedString = this.searchString; | ||||
|         this.searchString = theSearchString; | ||||
|         infoListAdapter.clearStreamItemList(); | ||||
| @@ -843,13 +852,17 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|                 searchBinding.searchMetaInfoSeparator, disposables); | ||||
|         hideKeyboardSearch(); | ||||
|  | ||||
|         // store search query if search history is enabled | ||||
|         disposables.add(historyRecordManager.onSearched(serviceId, theSearchString) | ||||
|                 .observeOn(AndroidSchedulers.mainThread()) | ||||
|                 .subscribe( | ||||
|                         ignored -> { }, | ||||
|                         ignored -> { | ||||
|                         }, | ||||
|                         throwable -> showSnackBarError(new ErrorInfo(throwable, UserAction.SEARCHED, | ||||
|                                 theSearchString, serviceId)) | ||||
|                 )); | ||||
|  | ||||
|         // load search results | ||||
|         suggestionPublisher.onNext(theSearchString); | ||||
|         startLoading(false); | ||||
|     } | ||||
| @@ -922,7 +935,7 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|         filterItemCheckedId = item.getItemId(); | ||||
|         item.setChecked(true); | ||||
|  | ||||
|         contentFilter = new String[]{theContentFilter.get(0)}; | ||||
|         contentFilter = theContentFilter.toArray(new String[0]); | ||||
|  | ||||
|         if (!TextUtils.isEmpty(searchString)) { | ||||
|             search(searchString, contentFilter, sortFilter); | ||||
| @@ -939,6 +952,14 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|         sortFilter = theSortFilter; | ||||
|     } | ||||
|  | ||||
|     private String getSearchEditString() { | ||||
|         return searchEditText.getText().toString(); | ||||
|     } | ||||
|  | ||||
|     private boolean isSearchEditBlank() { | ||||
|         return isBlank(getSearchEditString()); | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Suggestion Results | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
| @@ -947,8 +968,8 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|         if (DEBUG) { | ||||
|             Log.d(TAG, "handleSuggestions() called with: suggestions = [" + suggestions + "]"); | ||||
|         } | ||||
|         searchBinding.suggestionsList.smoothScrollToPosition(0); | ||||
|         searchBinding.suggestionsList.post(() -> suggestionListAdapter.setItems(suggestions)); | ||||
|         suggestionListAdapter.submitList(suggestions, | ||||
|                 () -> searchBinding.suggestionsList.scrollToPosition(0)); | ||||
|  | ||||
|         if (suggestionsPanelVisible && isErrorPanelVisible()) { | ||||
|             hideLoading(); | ||||
| @@ -980,11 +1001,13 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|         } | ||||
|  | ||||
|         searchSuggestion = result.getSearchSuggestion(); | ||||
|         if (searchSuggestion != null) { | ||||
|             searchSuggestion = searchSuggestion.trim(); | ||||
|         } | ||||
|         isCorrectedSearch = result.isCorrectedSearch(); | ||||
|  | ||||
|         // List<MetaInfo> cannot be bundled without creating some containers | ||||
|         metaInfo = new MetaInfo[result.getMetaInfo().size()]; | ||||
|         metaInfo = result.getMetaInfo().toArray(metaInfo); | ||||
|         metaInfo = result.getMetaInfo().toArray(new MetaInfo[0]); | ||||
|         showMetaInfoInTextView(result.getMetaInfo(), searchBinding.searchMetaInfoTextView, | ||||
|                 searchBinding.searchMetaInfoSeparator, disposables); | ||||
|  | ||||
| @@ -1070,19 +1093,19 @@ public class SearchFragment extends BaseListFragment<SearchInfo, ListExtractor.I | ||||
|             return 0; | ||||
|         } | ||||
|  | ||||
|         final SuggestionItem item = suggestionListAdapter.getItem(position); | ||||
|         final SuggestionItem item = suggestionListAdapter.getCurrentList().get(position); | ||||
|         return item.fromHistory ? makeMovementFlags(0, | ||||
|                 ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT) : 0; | ||||
|     } | ||||
|  | ||||
|     public void onSuggestionItemSwiped(@NonNull final RecyclerView.ViewHolder viewHolder) { | ||||
|         final int position = viewHolder.getBindingAdapterPosition(); | ||||
|         final String query = suggestionListAdapter.getItem(position).query; | ||||
|         final String query = suggestionListAdapter.getCurrentList().get(position).query; | ||||
|         final Disposable onDelete = historyRecordManager.deleteSearchHistory(query) | ||||
|                 .observeOn(AndroidSchedulers.mainThread()) | ||||
|                 .subscribe( | ||||
|                         howManyDeleted -> suggestionPublisher | ||||
|                                 .onNext(searchEditText.getText().toString()), | ||||
|                                 .onNext(getSearchEditString()), | ||||
|                         throwable -> showSnackBarError(new ErrorInfo(throwable, | ||||
|                                 UserAction.DELETE_FROM_HISTORY, "Deleting item failed"))); | ||||
|         disposables.add(onDelete); | ||||
|   | ||||
| @@ -1,34 +1,22 @@ | ||||
| package org.schabi.newpipe.fragments.list.search; | ||||
|  | ||||
| import android.content.Context; | ||||
| import android.view.LayoutInflater; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageView; | ||||
| import android.widget.TextView; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.recyclerview.widget.DiffUtil; | ||||
| import androidx.recyclerview.widget.ListAdapter; | ||||
| import androidx.recyclerview.widget.RecyclerView; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import org.schabi.newpipe.databinding.ItemSearchSuggestionBinding; | ||||
|  | ||||
| public class SuggestionListAdapter | ||||
|         extends RecyclerView.Adapter<SuggestionListAdapter.SuggestionItemHolder> { | ||||
|     private final ArrayList<SuggestionItem> items = new ArrayList<>(); | ||||
|     private final Context context; | ||||
|         extends ListAdapter<SuggestionItem, SuggestionListAdapter.SuggestionItemHolder> { | ||||
|     private OnSuggestionItemSelected listener; | ||||
|  | ||||
|     public SuggestionListAdapter(final Context context) { | ||||
|         this.context = context; | ||||
|     } | ||||
|  | ||||
|     public void setItems(final List<SuggestionItem> items) { | ||||
|         this.items.clear(); | ||||
|         this.items.addAll(items); | ||||
|         notifyDataSetChanged(); | ||||
|     public SuggestionListAdapter() { | ||||
|         super(new SuggestionItemCallback()); | ||||
|     } | ||||
|  | ||||
|     public void setListener(final OnSuggestionItemSelected listener) { | ||||
| @@ -39,45 +27,32 @@ public class SuggestionListAdapter | ||||
|     @Override | ||||
|     public SuggestionItemHolder onCreateViewHolder(@NonNull final ViewGroup parent, | ||||
|                                                    final int viewType) { | ||||
|         return new SuggestionItemHolder(LayoutInflater.from(context) | ||||
|                 .inflate(R.layout.item_search_suggestion, parent, false)); | ||||
|         return new SuggestionItemHolder(ItemSearchSuggestionBinding | ||||
|                 .inflate(LayoutInflater.from(parent.getContext()), parent, false)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onBindViewHolder(final SuggestionItemHolder holder, final int position) { | ||||
|         final SuggestionItem currentItem = getItem(position); | ||||
|         holder.updateFrom(currentItem); | ||||
|         holder.queryView.setOnClickListener(v -> { | ||||
|         holder.itemBinding.suggestionSearch.setOnClickListener(v -> { | ||||
|             if (listener != null) { | ||||
|                 listener.onSuggestionItemSelected(currentItem); | ||||
|             } | ||||
|         }); | ||||
|         holder.queryView.setOnLongClickListener(v -> { | ||||
|         holder.itemBinding.suggestionSearch.setOnLongClickListener(v -> { | ||||
|             if (listener != null) { | ||||
|                 listener.onSuggestionItemLongClick(currentItem); | ||||
|             } | ||||
|             return true; | ||||
|         }); | ||||
|         holder.insertView.setOnClickListener(v -> { | ||||
|         holder.itemBinding.suggestionInsert.setOnClickListener(v -> { | ||||
|             if (listener != null) { | ||||
|                 listener.onSuggestionItemInserted(currentItem); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     SuggestionItem getItem(final int position) { | ||||
|         return items.get(position); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public int getItemCount() { | ||||
|         return items.size(); | ||||
|     } | ||||
|  | ||||
|     public boolean isEmpty() { | ||||
|         return getItemCount() == 0; | ||||
|     } | ||||
|  | ||||
|     public interface OnSuggestionItemSelected { | ||||
|         void onSuggestionItemSelected(SuggestionItem item); | ||||
|  | ||||
| @@ -87,30 +62,32 @@ public class SuggestionListAdapter | ||||
|     } | ||||
|  | ||||
|     public static final class SuggestionItemHolder extends RecyclerView.ViewHolder { | ||||
|         private final TextView itemSuggestionQuery; | ||||
|         private final ImageView suggestionIcon; | ||||
|         private final View queryView; | ||||
|         private final View insertView; | ||||
|         private final ItemSearchSuggestionBinding itemBinding; | ||||
|  | ||||
|         // Cache some ids, as they can potentially be constantly updated/recycled | ||||
|         private final int historyResId; | ||||
|         private final int searchResId; | ||||
|  | ||||
|         private SuggestionItemHolder(final View rootView) { | ||||
|             super(rootView); | ||||
|             suggestionIcon = rootView.findViewById(R.id.item_suggestion_icon); | ||||
|             itemSuggestionQuery = rootView.findViewById(R.id.item_suggestion_query); | ||||
|  | ||||
|             queryView = rootView.findViewById(R.id.suggestion_search); | ||||
|             insertView = rootView.findViewById(R.id.suggestion_insert); | ||||
|  | ||||
|             historyResId = R.drawable.ic_history; | ||||
|             searchResId = R.drawable.ic_search; | ||||
|         private SuggestionItemHolder(final ItemSearchSuggestionBinding binding) { | ||||
|             super(binding.getRoot()); | ||||
|             this.itemBinding = binding; | ||||
|         } | ||||
|  | ||||
|         private void updateFrom(final SuggestionItem item) { | ||||
|             suggestionIcon.setImageResource(item.fromHistory ? historyResId : searchResId); | ||||
|             itemSuggestionQuery.setText(item.query); | ||||
|             itemBinding.itemSuggestionIcon.setImageResource(item.fromHistory ? R.drawable.ic_history | ||||
|                     : R.drawable.ic_search); | ||||
|             itemBinding.itemSuggestionQuery.setText(item.query); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private static class SuggestionItemCallback extends DiffUtil.ItemCallback<SuggestionItem> { | ||||
|         @Override | ||||
|         public boolean areItemsTheSame(@NonNull final SuggestionItem oldItem, | ||||
|                                        @NonNull final SuggestionItem newItem) { | ||||
|             return oldItem.fromHistory == newItem.fromHistory | ||||
|                     && oldItem.query.equals(newItem.query); | ||||
|         } | ||||
|  | ||||
|         @Override | ||||
|         public boolean areContentsTheSame(@NonNull final SuggestionItem oldItem, | ||||
|                                           @NonNull final SuggestionItem newItem) { | ||||
|             return true; // items' contents never change; the list of items themselves does | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -19,19 +19,19 @@ import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.ListExtractor; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfo; | ||||
| import org.schabi.newpipe.fragments.list.BaseListInfoFragment; | ||||
| import org.schabi.newpipe.info_list.ItemViewMode; | ||||
| import org.schabi.newpipe.ktx.ViewUtils; | ||||
| import org.schabi.newpipe.util.RelatedItemInfo; | ||||
|  | ||||
| import java.io.Serializable; | ||||
| import java.util.function.Supplier; | ||||
|  | ||||
| import io.reactivex.rxjava3.core.Single; | ||||
|  | ||||
| public class RelatedItemsFragment extends BaseListInfoFragment<InfoItem, RelatedItemInfo> | ||||
| public class RelatedItemsFragment extends BaseListInfoFragment<InfoItem, RelatedItemsInfo> | ||||
|         implements SharedPreferences.OnSharedPreferenceChangeListener { | ||||
|     private static final String INFO_KEY = "related_info_key"; | ||||
|  | ||||
|     private RelatedItemInfo relatedItemInfo; | ||||
|     private RelatedItemsInfo relatedItemsInfo; | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Views | ||||
| @@ -68,7 +68,7 @@ public class RelatedItemsFragment extends BaseListInfoFragment<InfoItem, Related | ||||
|  | ||||
|     @Override | ||||
|     protected Supplier<View> getListHeaderSupplier() { | ||||
|         if (relatedItemInfo == null || relatedItemInfo.getRelatedItems() == null) { | ||||
|         if (relatedItemsInfo == null || relatedItemsInfo.getRelatedItems() == null) { | ||||
|             return null; | ||||
|         } | ||||
|  | ||||
| @@ -96,8 +96,8 @@ public class RelatedItemsFragment extends BaseListInfoFragment<InfoItem, Related | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
|  | ||||
|     @Override | ||||
|     protected Single<RelatedItemInfo> loadResult(final boolean forceLoad) { | ||||
|         return Single.fromCallable(() -> relatedItemInfo); | ||||
|     protected Single<RelatedItemsInfo> loadResult(final boolean forceLoad) { | ||||
|         return Single.fromCallable(() -> relatedItemsInfo); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
| @@ -109,7 +109,7 @@ public class RelatedItemsFragment extends BaseListInfoFragment<InfoItem, Related | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void handleResult(@NonNull final RelatedItemInfo result) { | ||||
|     public void handleResult(@NonNull final RelatedItemsInfo result) { | ||||
|         super.handleResult(result); | ||||
|  | ||||
|         if (headerBinding != null) { | ||||
| @@ -136,38 +136,41 @@ public class RelatedItemsFragment extends BaseListInfoFragment<InfoItem, Related | ||||
|  | ||||
|     private void setInitialData(final StreamInfo info) { | ||||
|         super.setInitialData(info.getServiceId(), info.getUrl(), info.getName()); | ||||
|         if (this.relatedItemInfo == null) { | ||||
|             this.relatedItemInfo = RelatedItemInfo.getInfo(info); | ||||
|         if (this.relatedItemsInfo == null) { | ||||
|             this.relatedItemsInfo = new RelatedItemsInfo(info); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onSaveInstanceState(@NonNull final Bundle outState) { | ||||
|         super.onSaveInstanceState(outState); | ||||
|         outState.putSerializable(INFO_KEY, relatedItemInfo); | ||||
|         outState.putSerializable(INFO_KEY, relatedItemsInfo); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void onRestoreInstanceState(@NonNull final Bundle savedState) { | ||||
|         super.onRestoreInstanceState(savedState); | ||||
|         final Serializable serializable = savedState.getSerializable(INFO_KEY); | ||||
|         if (serializable instanceof RelatedItemInfo) { | ||||
|             this.relatedItemInfo = (RelatedItemInfo) serializable; | ||||
|         if (serializable instanceof RelatedItemsInfo) { | ||||
|             this.relatedItemsInfo = (RelatedItemsInfo) serializable; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onSharedPreferenceChanged(final SharedPreferences sharedPreferences, | ||||
|                                           final String s) { | ||||
|         if (headerBinding != null) { | ||||
|             headerBinding.autoplaySwitch.setChecked( | ||||
|                     sharedPreferences.getBoolean( | ||||
|                             getString(R.string.auto_queue_key), false)); | ||||
|                                           final String key) { | ||||
|         if (headerBinding != null && getString(R.string.auto_queue_key).equals(key)) { | ||||
|             headerBinding.autoplaySwitch.setChecked(sharedPreferences.getBoolean(key, false)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected boolean isGridLayout() { | ||||
|         return false; | ||||
|     protected ItemViewMode getItemViewMode() { | ||||
|         ItemViewMode mode = super.getItemViewMode(); | ||||
|         // Only list mode is supported. Either List or card will be used. | ||||
|         if (mode != ItemViewMode.LIST && mode != ItemViewMode.CARD) { | ||||
|             mode = ItemViewMode.LIST; | ||||
|         } | ||||
|         return mode; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,22 @@ | ||||
| package org.schabi.newpipe.fragments.list.videos; | ||||
|  | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.ListInfo; | ||||
| import org.schabi.newpipe.extractor.linkhandler.ListLinkHandler; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfo; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.Collections; | ||||
|  | ||||
| public final class RelatedItemsInfo extends ListInfo<InfoItem> { | ||||
|     /** | ||||
|      * This class is used to wrap the related items of a StreamInfo into a ListInfo object. | ||||
|      * | ||||
|      * @param info the stream info from which to get related items | ||||
|      */ | ||||
|     public RelatedItemsInfo(final StreamInfo info) { | ||||
|         super(info.getServiceId(), new ListLinkHandler(info.getOriginalUrl(), info.getUrl(), | ||||
|                 info.getId(), Collections.emptyList(), null), info.getName()); | ||||
|         setRelatedItems(new ArrayList<>(info.getRelatedItems())); | ||||
|     } | ||||
| } | ||||
| @@ -13,8 +13,7 @@ import org.schabi.newpipe.extractor.playlist.PlaylistInfoItem; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.info_list.holder.ChannelInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.ChannelMiniInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.CommentsInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.CommentsMiniInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.CommentInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.InfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.PlaylistInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.PlaylistMiniInfoItemHolder; | ||||
| @@ -67,8 +66,8 @@ public class InfoItemBuilder { | ||||
|     public View buildView(@NonNull final ViewGroup parent, @NonNull final InfoItem infoItem, | ||||
|                           final HistoryRecordManager historyRecordManager, | ||||
|                           final boolean useMiniVariant) { | ||||
|         final InfoItemHolder holder | ||||
|                 = holderFromInfoType(parent, infoItem.getInfoType(), useMiniVariant); | ||||
|         final InfoItemHolder holder = | ||||
|                 holderFromInfoType(parent, infoItem.getInfoType(), useMiniVariant); | ||||
|         holder.updateFromItem(infoItem, historyRecordManager); | ||||
|         return holder.itemView; | ||||
|     } | ||||
| @@ -87,8 +86,7 @@ public class InfoItemBuilder { | ||||
|                 return useMiniVariant ? new PlaylistMiniInfoItemHolder(this, parent) | ||||
|                         : new PlaylistInfoItemHolder(this, parent); | ||||
|             case COMMENT: | ||||
|                 return useMiniVariant ? new CommentsMiniInfoItemHolder(this, parent) | ||||
|                         : new CommentsInfoItemHolder(this, parent); | ||||
|                 return new CommentInfoItemHolder(this, parent); | ||||
|             default: | ||||
|                 throw new RuntimeException("InfoType not expected = " + infoType.name()); | ||||
|         } | ||||
|   | ||||
| @@ -17,15 +17,17 @@ import org.schabi.newpipe.extractor.channel.ChannelInfoItem; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.extractor.playlist.PlaylistInfoItem; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.info_list.holder.ChannelCardInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.ChannelGridInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.ChannelInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.ChannelMiniInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.CommentsInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.CommentsMiniInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.CommentInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.InfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.PlaylistCardInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.PlaylistGridInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.PlaylistInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.PlaylistMiniInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.StreamCardInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.StreamGridInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.StreamInfoItemHolder; | ||||
| import org.schabi.newpipe.info_list.holder.StreamMiniInfoItemHolder; | ||||
| @@ -67,14 +69,16 @@ public class InfoListAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolde | ||||
|     private static final int MINI_STREAM_HOLDER_TYPE = 0x100; | ||||
|     private static final int STREAM_HOLDER_TYPE = 0x101; | ||||
|     private static final int GRID_STREAM_HOLDER_TYPE = 0x102; | ||||
|     private static final int CARD_STREAM_HOLDER_TYPE = 0x103; | ||||
|     private static final int MINI_CHANNEL_HOLDER_TYPE = 0x200; | ||||
|     private static final int CHANNEL_HOLDER_TYPE = 0x201; | ||||
|     private static final int GRID_CHANNEL_HOLDER_TYPE = 0x202; | ||||
|     private static final int CARD_CHANNEL_HOLDER_TYPE = 0x203; | ||||
|     private static final int MINI_PLAYLIST_HOLDER_TYPE = 0x300; | ||||
|     private static final int PLAYLIST_HOLDER_TYPE = 0x301; | ||||
|     private static final int GRID_PLAYLIST_HOLDER_TYPE = 0x302; | ||||
|     private static final int MINI_COMMENT_HOLDER_TYPE = 0x400; | ||||
|     private static final int COMMENT_HOLDER_TYPE = 0x401; | ||||
|     private static final int CARD_PLAYLIST_HOLDER_TYPE = 0x303; | ||||
|     private static final int COMMENT_HOLDER_TYPE = 0x400; | ||||
|  | ||||
|     private final LayoutInflater layoutInflater; | ||||
|     private final InfoItemBuilder infoItemBuilder; | ||||
| @@ -82,9 +86,10 @@ public class InfoListAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolde | ||||
|     private final HistoryRecordManager recordManager; | ||||
|  | ||||
|     private boolean useMiniVariant = false; | ||||
|     private boolean useGridVariant = false; | ||||
|     private boolean showFooter = false; | ||||
|  | ||||
|     private ItemViewMode itemMode = ItemViewMode.LIST; | ||||
|  | ||||
|     private Supplier<View> headerSupplier = null; | ||||
|  | ||||
|     public InfoListAdapter(final Context context) { | ||||
| @@ -114,8 +119,8 @@ public class InfoListAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolde | ||||
|         this.useMiniVariant = useMiniVariant; | ||||
|     } | ||||
|  | ||||
|     public void setUseGridVariant(final boolean useGridVariant) { | ||||
|         this.useGridVariant = useGridVariant; | ||||
|     public void setItemViewMode(final ItemViewMode itemViewMode) { | ||||
|         this.itemMode = itemViewMode; | ||||
|     } | ||||
|  | ||||
|     public void addInfoItemList(@Nullable final List<? extends InfoItem> data) { | ||||
| @@ -234,16 +239,37 @@ public class InfoListAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolde | ||||
|         final InfoItem item = infoItemList.get(position); | ||||
|         switch (item.getInfoType()) { | ||||
|             case STREAM: | ||||
|                 return useGridVariant ? GRID_STREAM_HOLDER_TYPE : useMiniVariant | ||||
|                         ? MINI_STREAM_HOLDER_TYPE : STREAM_HOLDER_TYPE; | ||||
|                 if (itemMode == ItemViewMode.CARD) { | ||||
|                     return CARD_STREAM_HOLDER_TYPE; | ||||
|                 } else if (itemMode == ItemViewMode.GRID) { | ||||
|                     return GRID_STREAM_HOLDER_TYPE; | ||||
|                 } else if (useMiniVariant) { | ||||
|                     return MINI_STREAM_HOLDER_TYPE; | ||||
|                 } else { | ||||
|                     return STREAM_HOLDER_TYPE; | ||||
|                 } | ||||
|             case CHANNEL: | ||||
|                 return useGridVariant ? GRID_CHANNEL_HOLDER_TYPE : useMiniVariant | ||||
|                         ? MINI_CHANNEL_HOLDER_TYPE : CHANNEL_HOLDER_TYPE; | ||||
|                 if (itemMode == ItemViewMode.CARD) { | ||||
|                     return CARD_CHANNEL_HOLDER_TYPE; | ||||
|                 } else if (itemMode == ItemViewMode.GRID) { | ||||
|                     return GRID_CHANNEL_HOLDER_TYPE; | ||||
|                 } else if (useMiniVariant) { | ||||
|                     return MINI_CHANNEL_HOLDER_TYPE; | ||||
|                 } else { | ||||
|                     return CHANNEL_HOLDER_TYPE; | ||||
|                 } | ||||
|             case PLAYLIST: | ||||
|                 return useGridVariant ? GRID_PLAYLIST_HOLDER_TYPE : useMiniVariant | ||||
|                         ? MINI_PLAYLIST_HOLDER_TYPE : PLAYLIST_HOLDER_TYPE; | ||||
|                 if (itemMode == ItemViewMode.CARD) { | ||||
|                     return CARD_PLAYLIST_HOLDER_TYPE; | ||||
|                 } else if (itemMode == ItemViewMode.GRID) { | ||||
|                     return GRID_PLAYLIST_HOLDER_TYPE; | ||||
|                 } else if (useMiniVariant) { | ||||
|                     return MINI_PLAYLIST_HOLDER_TYPE; | ||||
|                 } else { | ||||
|                     return PLAYLIST_HOLDER_TYPE; | ||||
|                 } | ||||
|             case COMMENT: | ||||
|                 return useMiniVariant ? MINI_COMMENT_HOLDER_TYPE : COMMENT_HOLDER_TYPE; | ||||
|                 return COMMENT_HOLDER_TYPE; | ||||
|             default: | ||||
|                 return -1; | ||||
|         } | ||||
| @@ -274,10 +300,14 @@ public class InfoListAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolde | ||||
|                 return new StreamInfoItemHolder(infoItemBuilder, parent); | ||||
|             case GRID_STREAM_HOLDER_TYPE: | ||||
|                 return new StreamGridInfoItemHolder(infoItemBuilder, parent); | ||||
|             case CARD_STREAM_HOLDER_TYPE: | ||||
|                 return new StreamCardInfoItemHolder(infoItemBuilder, parent); | ||||
|             case MINI_CHANNEL_HOLDER_TYPE: | ||||
|                 return new ChannelMiniInfoItemHolder(infoItemBuilder, parent); | ||||
|             case CHANNEL_HOLDER_TYPE: | ||||
|                 return new ChannelInfoItemHolder(infoItemBuilder, parent); | ||||
|             case CARD_CHANNEL_HOLDER_TYPE: | ||||
|                 return new ChannelCardInfoItemHolder(infoItemBuilder, parent); | ||||
|             case GRID_CHANNEL_HOLDER_TYPE: | ||||
|                 return new ChannelGridInfoItemHolder(infoItemBuilder, parent); | ||||
|             case MINI_PLAYLIST_HOLDER_TYPE: | ||||
| @@ -286,10 +316,10 @@ public class InfoListAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolde | ||||
|                 return new PlaylistInfoItemHolder(infoItemBuilder, parent); | ||||
|             case GRID_PLAYLIST_HOLDER_TYPE: | ||||
|                 return new PlaylistGridInfoItemHolder(infoItemBuilder, parent); | ||||
|             case MINI_COMMENT_HOLDER_TYPE: | ||||
|                 return new CommentsMiniInfoItemHolder(infoItemBuilder, parent); | ||||
|             case CARD_PLAYLIST_HOLDER_TYPE: | ||||
|                 return new PlaylistCardInfoItemHolder(infoItemBuilder, parent); | ||||
|             case COMMENT_HOLDER_TYPE: | ||||
|                 return new CommentsInfoItemHolder(infoItemBuilder, parent); | ||||
|                 return new CommentInfoItemHolder(infoItemBuilder, parent); | ||||
|             default: | ||||
|                 return new FallbackViewHolder(new View(parent.getContext())); | ||||
|         } | ||||
|   | ||||
| @@ -0,0 +1,23 @@ | ||||
| package org.schabi.newpipe.info_list; | ||||
|  | ||||
| /** | ||||
|  * Item view mode for streams & playlist listing screens. | ||||
|  */ | ||||
| public enum ItemViewMode { | ||||
|     /** | ||||
|      * Default mode. | ||||
|      */ | ||||
|     AUTO, | ||||
|     /** | ||||
|      * Full width list item with thumb on the left and two line title & uploader in right. | ||||
|      */ | ||||
|     LIST, | ||||
|     /** | ||||
|      * Grid mode places two cards per row. | ||||
|      */ | ||||
|     GRID, | ||||
|     /** | ||||
|      * A full width card in phone - portrait. | ||||
|      */ | ||||
|     CARD | ||||
| } | ||||
| @@ -61,5 +61,6 @@ class StreamSegmentAdapter( | ||||
|  | ||||
|     interface StreamSegmentListener { | ||||
|         fun onItemClick(item: StreamSegmentItem, seconds: Int) | ||||
|         fun onItemLongClick(item: StreamSegmentItem, seconds: Int) | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import com.xwray.groupie.Item | ||||
| import org.schabi.newpipe.R | ||||
| import org.schabi.newpipe.extractor.stream.StreamSegment | ||||
| import org.schabi.newpipe.util.Localization | ||||
| import org.schabi.newpipe.util.PicassoHelper | ||||
| import org.schabi.newpipe.util.image.PicassoHelper | ||||
|  | ||||
| class StreamSegmentItem( | ||||
|     private val item: StreamSegment, | ||||
| @@ -41,6 +41,7 @@ class StreamSegmentItem( | ||||
|         viewHolder.root.findViewById<TextView>(R.id.textViewStartSeconds).text = | ||||
|             Localization.getDurationString(item.startTimeSeconds.toLong()) | ||||
|         viewHolder.root.setOnClickListener { onClick.onItemClick(this, item.startTimeSeconds) } | ||||
|         viewHolder.root.setOnLongClickListener { onClick.onItemLongClick(this, item.startTimeSeconds); true } | ||||
|         viewHolder.root.isSelected = isSelected | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -24,6 +24,7 @@ import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.extractor.stream.StreamType; | ||||
| import org.schabi.newpipe.player.helper.PlayerHolder; | ||||
| import org.schabi.newpipe.util.StreamTypeUtil; | ||||
| import org.schabi.newpipe.util.external_communication.KoreUtils; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| @@ -251,10 +252,11 @@ public final class InfoItemDialog { | ||||
|          * @return the current {@link Builder} instance | ||||
|          */ | ||||
|         public Builder addEnqueueEntriesIfNeeded() { | ||||
|             if (PlayerHolder.getInstance().isPlayQueueReady()) { | ||||
|             final PlayerHolder holder = PlayerHolder.getInstance(); | ||||
|             if (holder.isPlayQueueReady()) { | ||||
|                 addEntry(StreamDialogDefaultEntry.ENQUEUE); | ||||
|  | ||||
|                 if (PlayerHolder.getInstance().getQueueSize() > 1) { | ||||
|                 if (holder.getQueuePosition() < holder.getQueueSize() - 1) { | ||||
|                     addEntry(StreamDialogDefaultEntry.ENQUEUE_NEXT); | ||||
|                 } | ||||
|             } | ||||
| @@ -269,8 +271,7 @@ public final class InfoItemDialog { | ||||
|          */ | ||||
|         public Builder addStartHereEntries() { | ||||
|             addEntry(StreamDialogDefaultEntry.START_HERE_ON_BACKGROUND); | ||||
|             if (infoItem.getStreamType() != StreamType.AUDIO_STREAM | ||||
|                     && infoItem.getStreamType() != StreamType.AUDIO_LIVE_STREAM) { | ||||
|             if (!StreamTypeUtil.isAudio(infoItem.getStreamType())) { | ||||
|                 addEntry(StreamDialogDefaultEntry.START_HERE_ON_POPUP); | ||||
|             } | ||||
|             return this; | ||||
| @@ -285,9 +286,7 @@ public final class InfoItemDialog { | ||||
|             final boolean isWatchHistoryEnabled = PreferenceManager | ||||
|                     .getDefaultSharedPreferences(context) | ||||
|                     .getBoolean(context.getString(R.string.enable_watch_history_key), false); | ||||
|             if (isWatchHistoryEnabled | ||||
|                     && infoItem.getStreamType() != StreamType.LIVE_STREAM | ||||
|                     && infoItem.getStreamType() != StreamType.AUDIO_LIVE_STREAM) { | ||||
|             if (isWatchHistoryEnabled && !StreamTypeUtil.isLiveStream(infoItem.getStreamType())) { | ||||
|                 addEntry(StreamDialogDefaultEntry.MARK_AS_WATCHED); | ||||
|             } | ||||
|             return this; | ||||
| @@ -323,6 +322,7 @@ public final class InfoItemDialog { | ||||
|          */ | ||||
|         public Builder addDefaultEndEntries() { | ||||
|             addAllEntries( | ||||
|                     StreamDialogDefaultEntry.DOWNLOAD, | ||||
|                     StreamDialogDefaultEntry.APPEND_PLAYLIST, | ||||
|                     StreamDialogDefaultEntry.SHARE, | ||||
|                     StreamDialogDefaultEntry.OPEN_IN_BROWSER | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package org.schabi.newpipe.info_list.dialog; | ||||
|  | ||||
| import static org.schabi.newpipe.util.NavigationHelper.openChannelFragment; | ||||
| import static org.schabi.newpipe.util.SparseItemUtil.fetchItemInfoIfSparse; | ||||
| import static org.schabi.newpipe.util.SparseItemUtil.fetchStreamInfoAndSaveToDatabase; | ||||
| import static org.schabi.newpipe.util.SparseItemUtil.fetchUploaderUrlIfSparse; | ||||
|  | ||||
| import android.net.Uri; | ||||
| @@ -11,6 +12,7 @@ import androidx.annotation.StringRes; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.database.stream.model.StreamEntity; | ||||
| import org.schabi.newpipe.download.DownloadDialog; | ||||
| import org.schabi.newpipe.local.dialog.PlaylistAppendDialog; | ||||
| import org.schabi.newpipe.local.dialog.PlaylistDialog; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| @@ -18,7 +20,7 @@ import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.external_communication.KoreUtils; | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils; | ||||
|  | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
|  | ||||
| import io.reactivex.rxjava3.android.schedulers.AndroidSchedulers; | ||||
|  | ||||
| @@ -87,7 +89,7 @@ public enum StreamDialogDefaultEntry { | ||||
|     APPEND_PLAYLIST(R.string.add_to_playlist, (fragment, item) -> | ||||
|         PlaylistDialog.createCorrespondingDialog( | ||||
|                 fragment.getContext(), | ||||
|                 Collections.singletonList(new StreamEntity(item)), | ||||
|                 List.of(new StreamEntity(item)), | ||||
|                 dialog -> dialog.show( | ||||
|                         fragment.getParentFragmentManager(), | ||||
|                         "StreamDialogEntry@" | ||||
| @@ -97,18 +99,28 @@ public enum StreamDialogDefaultEntry { | ||||
|         ) | ||||
|     ), | ||||
|  | ||||
|     PLAY_WITH_KODI(R.string.play_with_kodi_title, (fragment, item) -> { | ||||
|         final Uri videoUrl = Uri.parse(item.getUrl()); | ||||
|         try { | ||||
|             NavigationHelper.playWithKore(fragment.requireContext(), videoUrl); | ||||
|         } catch (final Exception e) { | ||||
|             KoreUtils.showInstallKoreDialog(fragment.requireActivity()); | ||||
|         } | ||||
|     }), | ||||
|     PLAY_WITH_KODI(R.string.play_with_kodi_title, (fragment, item) -> | ||||
|             KoreUtils.playWithKore(fragment.requireContext(), Uri.parse(item.getUrl()))), | ||||
|  | ||||
|     SHARE(R.string.share, (fragment, item) -> | ||||
|             ShareUtils.shareText(fragment.requireContext(), item.getName(), item.getUrl(), | ||||
|                     item.getThumbnailUrl())), | ||||
|                     item.getThumbnails())), | ||||
|  | ||||
|     /** | ||||
|      * Opens a {@link DownloadDialog} after fetching some stream info. | ||||
|      * If the user quits the current fragment, it will not open a DownloadDialog. | ||||
|      */ | ||||
|     DOWNLOAD(R.string.download, (fragment, item) -> | ||||
|             fetchStreamInfoAndSaveToDatabase(fragment.requireContext(), item.getServiceId(), | ||||
|                     item.getUrl(), info -> { | ||||
|                         if (fragment.getContext() != null) { | ||||
|                             final DownloadDialog downloadDialog = | ||||
|                                     new DownloadDialog(fragment.requireContext(), info); | ||||
|                             downloadDialog.show(fragment.getChildFragmentManager(), | ||||
|                                     "downloadDialog"); | ||||
|                         } | ||||
|                     }) | ||||
|     ), | ||||
|  | ||||
|     OPEN_IN_BROWSER(R.string.open_in_browser, (fragment, item) -> | ||||
|             ShareUtils.openUrlInBrowser(fragment.requireContext(), item.getUrl())), | ||||
|   | ||||
| @@ -0,0 +1,22 @@ | ||||
| package org.schabi.newpipe.info_list.holder; | ||||
|  | ||||
| import android.view.ViewGroup; | ||||
|  | ||||
| import androidx.annotation.Nullable; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
|  | ||||
| public class ChannelCardInfoItemHolder extends ChannelMiniInfoItemHolder { | ||||
|     public ChannelCardInfoItemHolder(final InfoItemBuilder infoItemBuilder, | ||||
|                                      final ViewGroup parent) { | ||||
|         super(infoItemBuilder, R.layout.list_channel_card_item, parent); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected int getDescriptionMaxLineCount(@Nullable final String content) { | ||||
|         // Based on `list_channel_card_item` left side content (thumbnail 100dp | ||||
|         // + additional details), Right side description can grow up to 8 lines. | ||||
|         return 8; | ||||
|     } | ||||
| } | ||||
| @@ -1,14 +1,9 @@ | ||||
| package org.schabi.newpipe.info_list.holder; | ||||
|  | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.TextView; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfoItem; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
|  | ||||
| /* | ||||
|  * Created by Christian Schabesberger on 12.02.17. | ||||
| @@ -31,40 +26,7 @@ import org.schabi.newpipe.util.Localization; | ||||
|  */ | ||||
|  | ||||
| public class ChannelInfoItemHolder extends ChannelMiniInfoItemHolder { | ||||
|     private final TextView itemChannelDescriptionView; | ||||
|  | ||||
|     public ChannelInfoItemHolder(final InfoItemBuilder infoItemBuilder, final ViewGroup parent) { | ||||
|         super(infoItemBuilder, R.layout.list_channel_item, parent); | ||||
|         itemChannelDescriptionView = itemView.findViewById(R.id.itemChannelDescriptionView); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void updateFromItem(final InfoItem infoItem, | ||||
|                                final HistoryRecordManager historyRecordManager) { | ||||
|         super.updateFromItem(infoItem, historyRecordManager); | ||||
|  | ||||
|         if (!(infoItem instanceof ChannelInfoItem)) { | ||||
|             return; | ||||
|         } | ||||
|         final ChannelInfoItem item = (ChannelInfoItem) infoItem; | ||||
|  | ||||
|         itemChannelDescriptionView.setText(item.getDescription()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected String getDetailLine(final ChannelInfoItem item) { | ||||
|         String details = super.getDetailLine(item); | ||||
|  | ||||
|         if (item.getStreamCount() >= 0) { | ||||
|             final String formattedVideoAmount = Localization.localizeStreamCount( | ||||
|                     itemBuilder.getContext(), item.getStreamCount()); | ||||
|  | ||||
|             if (!details.isEmpty()) { | ||||
|                 details += " • " + formattedVideoAmount; | ||||
|             } else { | ||||
|                 details = formattedVideoAmount; | ||||
|             } | ||||
|         } | ||||
|         return details; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,21 +1,26 @@ | ||||
| package org.schabi.newpipe.info_list.holder; | ||||
|  | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageView; | ||||
| import android.widget.TextView; | ||||
|  | ||||
| import androidx.annotation.Nullable; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.channel.ChannelInfoItem; | ||||
| import org.schabi.newpipe.extractor.utils.Utils; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| import org.schabi.newpipe.util.PicassoHelper; | ||||
| import org.schabi.newpipe.util.image.PicassoHelper; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
|  | ||||
| public class ChannelMiniInfoItemHolder extends InfoItemHolder { | ||||
|     public final ImageView itemThumbnailView; | ||||
|     public final TextView itemTitleView; | ||||
|     private final ImageView itemThumbnailView; | ||||
|     private final TextView itemTitleView; | ||||
|     private final TextView itemAdditionalDetailView; | ||||
|     private final TextView itemChannelDescriptionView; | ||||
|  | ||||
|     ChannelMiniInfoItemHolder(final InfoItemBuilder infoItemBuilder, final int layoutId, | ||||
|                               final ViewGroup parent) { | ||||
| @@ -24,6 +29,7 @@ public class ChannelMiniInfoItemHolder extends InfoItemHolder { | ||||
|         itemThumbnailView = itemView.findViewById(R.id.itemThumbnailView); | ||||
|         itemTitleView = itemView.findViewById(R.id.itemTitleView); | ||||
|         itemAdditionalDetailView = itemView.findViewById(R.id.itemAdditionalDetails); | ||||
|         itemChannelDescriptionView = itemView.findViewById(R.id.itemChannelDescriptionView); | ||||
|     } | ||||
|  | ||||
|     public ChannelMiniInfoItemHolder(final InfoItemBuilder infoItemBuilder, | ||||
| @@ -40,9 +46,17 @@ public class ChannelMiniInfoItemHolder extends InfoItemHolder { | ||||
|         final ChannelInfoItem item = (ChannelInfoItem) infoItem; | ||||
|  | ||||
|         itemTitleView.setText(item.getName()); | ||||
|         itemAdditionalDetailView.setText(getDetailLine(item)); | ||||
|         itemTitleView.setSelected(true); | ||||
|  | ||||
|         PicassoHelper.loadThumbnail(item.getThumbnailUrl()).into(itemThumbnailView); | ||||
|         final String detailLine = getDetailLine(item); | ||||
|         if (detailLine == null) { | ||||
|             itemAdditionalDetailView.setVisibility(View.GONE); | ||||
|         } else { | ||||
|             itemAdditionalDetailView.setVisibility(View.VISIBLE); | ||||
|             itemAdditionalDetailView.setText(getDetailLine(item)); | ||||
|         } | ||||
|  | ||||
|         PicassoHelper.loadAvatar(item.getThumbnails()).into(itemThumbnailView); | ||||
|  | ||||
|         itemView.setOnClickListener(view -> { | ||||
|             if (itemBuilder.getOnChannelSelectedListener() != null) { | ||||
| @@ -56,14 +70,48 @@ public class ChannelMiniInfoItemHolder extends InfoItemHolder { | ||||
|             } | ||||
|             return true; | ||||
|         }); | ||||
|  | ||||
|         if (itemChannelDescriptionView != null) { | ||||
|             // itemChannelDescriptionView will be null in the mini variant | ||||
|             if (Utils.isBlank(item.getDescription())) { | ||||
|                 itemChannelDescriptionView.setVisibility(View.GONE); | ||||
|             } else { | ||||
|                 itemChannelDescriptionView.setVisibility(View.VISIBLE); | ||||
|                 itemChannelDescriptionView.setText(item.getDescription()); | ||||
|                 // setMaxLines utilize the line space for description if the additional details | ||||
|                 // (sub / video count) are not present. | ||||
|                 // Case1: 2 lines of description + 1 line additional details | ||||
|                 // Case2: 3 lines of description (additionalDetails is GONE) | ||||
|                 itemChannelDescriptionView.setMaxLines(getDescriptionMaxLineCount(detailLine)); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     protected String getDetailLine(final ChannelInfoItem item) { | ||||
|         String details = ""; | ||||
|         if (item.getSubscriberCount() >= 0) { | ||||
|             details += Localization.shortSubscriberCount(itemBuilder.getContext(), | ||||
|     /** | ||||
|      * Returns max number of allowed lines for the description field. | ||||
|      * @param content additional detail content (video / sub count) | ||||
|      * @return max line count | ||||
|      */ | ||||
|     protected int getDescriptionMaxLineCount(@Nullable final String content) { | ||||
|         return content == null ? 3 : 2; | ||||
|     } | ||||
|  | ||||
|     @Nullable | ||||
|     private String getDetailLine(final ChannelInfoItem item) { | ||||
|         if (item.getStreamCount() >= 0 && item.getSubscriberCount() >= 0) { | ||||
|             return Localization.concatenateStrings( | ||||
|                     Localization.shortSubscriberCount(itemBuilder.getContext(), | ||||
|                             item.getSubscriberCount()), | ||||
|                     Localization.localizeStreamCount(itemBuilder.getContext(), | ||||
|                             item.getStreamCount())); | ||||
|         } else if (item.getStreamCount() >= 0) { | ||||
|             return Localization.localizeStreamCount(itemBuilder.getContext(), | ||||
|                     item.getStreamCount()); | ||||
|         } else if (item.getSubscriberCount() >= 0) { | ||||
|             return Localization.shortSubscriberCount(itemBuilder.getContext(), | ||||
|                     item.getSubscriberCount()); | ||||
|         } else { | ||||
|             return null; | ||||
|         } | ||||
|         return details; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -0,0 +1,188 @@ | ||||
| package org.schabi.newpipe.info_list.holder; | ||||
|  | ||||
| import static org.schabi.newpipe.util.ServiceHelper.getServiceById; | ||||
|  | ||||
| import android.text.method.LinkMovementMethod; | ||||
| import android.text.style.URLSpan; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.Button; | ||||
| import android.widget.ImageView; | ||||
| import android.widget.RelativeLayout; | ||||
| import android.widget.TextView; | ||||
|  | ||||
| import androidx.annotation.NonNull; | ||||
| import androidx.fragment.app.FragmentActivity; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| import org.schabi.newpipe.util.DeviceUtils; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils; | ||||
| import org.schabi.newpipe.util.image.ImageStrategy; | ||||
| import org.schabi.newpipe.util.image.PicassoHelper; | ||||
| import org.schabi.newpipe.util.text.CommentTextOnTouchListener; | ||||
| import org.schabi.newpipe.util.text.TextEllipsizer; | ||||
|  | ||||
| public class CommentInfoItemHolder extends InfoItemHolder { | ||||
|  | ||||
|     private static final int COMMENT_DEFAULT_LINES = 2; | ||||
|     private final int commentHorizontalPadding; | ||||
|     private final int commentVerticalPadding; | ||||
|  | ||||
|     private final RelativeLayout itemRoot; | ||||
|     private final ImageView itemThumbnailView; | ||||
|     private final TextView itemContentView; | ||||
|     private final ImageView itemThumbsUpView; | ||||
|     private final TextView itemLikesCountView; | ||||
|     private final TextView itemTitleView; | ||||
|     private final ImageView itemHeartView; | ||||
|     private final ImageView itemPinnedView; | ||||
|     private final Button repliesButton; | ||||
|  | ||||
|     @NonNull | ||||
|     private final TextEllipsizer textEllipsizer; | ||||
|  | ||||
|     public CommentInfoItemHolder(final InfoItemBuilder infoItemBuilder, | ||||
|                                  final ViewGroup parent) { | ||||
|         super(infoItemBuilder, R.layout.list_comment_item, parent); | ||||
|  | ||||
|         itemRoot = itemView.findViewById(R.id.itemRoot); | ||||
|         itemThumbnailView = itemView.findViewById(R.id.itemThumbnailView); | ||||
|         itemContentView = itemView.findViewById(R.id.itemCommentContentView); | ||||
|         itemThumbsUpView = itemView.findViewById(R.id.detail_thumbs_up_img_view); | ||||
|         itemLikesCountView = itemView.findViewById(R.id.detail_thumbs_up_count_view); | ||||
|         itemTitleView = itemView.findViewById(R.id.itemTitleView); | ||||
|         itemHeartView = itemView.findViewById(R.id.detail_heart_image_view); | ||||
|         itemPinnedView = itemView.findViewById(R.id.detail_pinned_view); | ||||
|         repliesButton = itemView.findViewById(R.id.replies_button); | ||||
|  | ||||
|         commentHorizontalPadding = (int) infoItemBuilder.getContext() | ||||
|                 .getResources().getDimension(R.dimen.comments_horizontal_padding); | ||||
|         commentVerticalPadding = (int) infoItemBuilder.getContext() | ||||
|                 .getResources().getDimension(R.dimen.comments_vertical_padding); | ||||
|  | ||||
|         textEllipsizer = new TextEllipsizer(itemContentView, COMMENT_DEFAULT_LINES, null); | ||||
|         textEllipsizer.setStateChangeListener(isEllipsized -> { | ||||
|             if (Boolean.TRUE.equals(isEllipsized)) { | ||||
|                 denyLinkFocus(); | ||||
|             } else { | ||||
|                 determineMovementMethod(); | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void updateFromItem(final InfoItem infoItem, | ||||
|                                final HistoryRecordManager historyRecordManager) { | ||||
|         if (!(infoItem instanceof CommentsInfoItem)) { | ||||
|             return; | ||||
|         } | ||||
|         final CommentsInfoItem item = (CommentsInfoItem) infoItem; | ||||
|  | ||||
|  | ||||
|         // load the author avatar | ||||
|         PicassoHelper.loadAvatar(item.getUploaderAvatars()).into(itemThumbnailView); | ||||
|         if (ImageStrategy.shouldLoadImages()) { | ||||
|             itemThumbnailView.setVisibility(View.VISIBLE); | ||||
|             itemRoot.setPadding(commentVerticalPadding, commentVerticalPadding, | ||||
|                     commentVerticalPadding, commentVerticalPadding); | ||||
|         } else { | ||||
|             itemThumbnailView.setVisibility(View.GONE); | ||||
|             itemRoot.setPadding(commentHorizontalPadding, commentVerticalPadding, | ||||
|                     commentHorizontalPadding, commentVerticalPadding); | ||||
|         } | ||||
|         itemThumbnailView.setOnClickListener(view -> openCommentAuthor(item)); | ||||
|  | ||||
|  | ||||
|         // setup the top row, with pinned icon, author name and comment date | ||||
|         itemPinnedView.setVisibility(item.isPinned() ? View.VISIBLE : View.GONE); | ||||
|         itemTitleView.setText(Localization.concatenateStrings(item.getUploaderName(), | ||||
|                 Localization.relativeTimeOrTextual(itemBuilder.getContext(), item.getUploadDate(), | ||||
|                         item.getTextualUploadDate()))); | ||||
|  | ||||
|  | ||||
|         // setup bottom row, with likes, heart and replies button | ||||
|         itemLikesCountView.setText( | ||||
|                 Localization.likeCount(itemBuilder.getContext(), item.getLikeCount())); | ||||
|  | ||||
|         itemHeartView.setVisibility(item.isHeartedByUploader() ? View.VISIBLE : View.GONE); | ||||
|  | ||||
|         final boolean hasReplies = item.getReplies() != null; | ||||
|         repliesButton.setOnClickListener(hasReplies ? v -> openCommentReplies(item) : null); | ||||
|         repliesButton.setVisibility(hasReplies ? View.VISIBLE : View.GONE); | ||||
|         repliesButton.setText(hasReplies | ||||
|                 ? Localization.replyCount(itemBuilder.getContext(), item.getReplyCount()) : ""); | ||||
|         ((RelativeLayout.LayoutParams) itemThumbsUpView.getLayoutParams()).topMargin = | ||||
|                 hasReplies ? 0 : DeviceUtils.dpToPx(6, itemBuilder.getContext()); | ||||
|  | ||||
|  | ||||
|         // setup comment content and click listeners to expand/ellipsize it | ||||
|         textEllipsizer.setStreamingService(getServiceById(item.getServiceId())); | ||||
|         textEllipsizer.setStreamUrl(item.getUrl()); | ||||
|         textEllipsizer.setContent(item.getCommentText()); | ||||
|         textEllipsizer.ellipsize(); | ||||
|  | ||||
|         //noinspection ClickableViewAccessibility | ||||
|         itemContentView.setOnTouchListener(CommentTextOnTouchListener.INSTANCE); | ||||
|  | ||||
|         itemView.setOnClickListener(view -> { | ||||
|             textEllipsizer.toggle(); | ||||
|             if (itemBuilder.getOnCommentsSelectedListener() != null) { | ||||
|                 itemBuilder.getOnCommentsSelectedListener().selected(item); | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         itemView.setOnLongClickListener(view -> { | ||||
|             if (DeviceUtils.isTv(itemBuilder.getContext())) { | ||||
|                 openCommentAuthor(item); | ||||
|             } else { | ||||
|                 final CharSequence text = itemContentView.getText(); | ||||
|                 if (text != null) { | ||||
|                     ShareUtils.copyToClipboard(itemBuilder.getContext(), text.toString()); | ||||
|                 } | ||||
|             } | ||||
|             return true; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     private void openCommentAuthor(@NonNull final CommentsInfoItem item) { | ||||
|         NavigationHelper.openCommentAuthorIfPresent((FragmentActivity) itemBuilder.getContext(), | ||||
|                 item); | ||||
|     } | ||||
|  | ||||
|     private void openCommentReplies(@NonNull final CommentsInfoItem item) { | ||||
|         NavigationHelper.openCommentRepliesFragment((FragmentActivity) itemBuilder.getContext(), | ||||
|                 item); | ||||
|     } | ||||
|  | ||||
|     private void allowLinkFocus() { | ||||
|         itemContentView.setMovementMethod(LinkMovementMethod.getInstance()); | ||||
|     } | ||||
|  | ||||
|     private void denyLinkFocus() { | ||||
|         itemContentView.setMovementMethod(null); | ||||
|     } | ||||
|  | ||||
|     private boolean shouldFocusLinks() { | ||||
|         if (itemView.isInTouchMode()) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         final URLSpan[] urls = itemContentView.getUrls(); | ||||
|  | ||||
|         return urls != null && urls.length != 0; | ||||
|     } | ||||
|  | ||||
|     private void determineMovementMethod() { | ||||
|         if (shouldFocusLinks()) { | ||||
|             allowLinkFocus(); | ||||
|         } else { | ||||
|             denyLinkFocus(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,63 +0,0 @@ | ||||
| package org.schabi.newpipe.info_list.holder; | ||||
|  | ||||
| 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.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
|  | ||||
| /* | ||||
|  * Created by Christian Schabesberger on 12.02.17. | ||||
|  * | ||||
|  * Copyright (C) Christian Schabesberger 2016 <chris.schabesberger@mailbox.org> | ||||
|  * ChannelInfoItemHolder .java is part of NewPipe. | ||||
|  * | ||||
|  * NewPipe is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * NewPipe is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with NewPipe.  If not, see <http://www.gnu.org/licenses/>. | ||||
|  */ | ||||
|  | ||||
| public class CommentsInfoItemHolder extends CommentsMiniInfoItemHolder { | ||||
|     public final TextView itemTitleView; | ||||
|     private final ImageView itemHeartView; | ||||
|     private final ImageView itemPinnedView; | ||||
|  | ||||
|     public CommentsInfoItemHolder(final InfoItemBuilder infoItemBuilder, final ViewGroup parent) { | ||||
|         super(infoItemBuilder, R.layout.list_comments_item, parent); | ||||
|  | ||||
|         itemTitleView = itemView.findViewById(R.id.itemTitleView); | ||||
|         itemHeartView = itemView.findViewById(R.id.detail_heart_image_view); | ||||
|         itemPinnedView = itemView.findViewById(R.id.detail_pinned_view); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void updateFromItem(final InfoItem infoItem, | ||||
|                                final HistoryRecordManager historyRecordManager) { | ||||
|         super.updateFromItem(infoItem, historyRecordManager); | ||||
|  | ||||
|         if (!(infoItem instanceof CommentsInfoItem)) { | ||||
|             return; | ||||
|         } | ||||
|         final CommentsInfoItem item = (CommentsInfoItem) infoItem; | ||||
|  | ||||
|         itemTitleView.setText(item.getUploaderName()); | ||||
|  | ||||
|         itemHeartView.setVisibility(item.isHeartedByUploader() ? View.VISIBLE : View.GONE); | ||||
|  | ||||
|         itemPinnedView.setVisibility(item.isPinned() ? View.VISIBLE : View.GONE); | ||||
|     } | ||||
| } | ||||
| @@ -1,255 +0,0 @@ | ||||
| package org.schabi.newpipe.info_list.holder; | ||||
|  | ||||
| import android.text.TextUtils; | ||||
| import android.text.method.LinkMovementMethod; | ||||
| import android.text.style.URLSpan; | ||||
| import android.text.util.Linkify; | ||||
| import android.util.Log; | ||||
| import android.view.View; | ||||
| import android.view.ViewGroup; | ||||
| import android.widget.ImageView; | ||||
| import android.widget.RelativeLayout; | ||||
| import android.widget.TextView; | ||||
|  | ||||
| import androidx.appcompat.app.AppCompatActivity; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.error.ErrorUtil; | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.comments.CommentsInfoItem; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| import org.schabi.newpipe.util.CommentTextOnTouchListener; | ||||
| import org.schabi.newpipe.util.DeviceUtils; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.NavigationHelper; | ||||
| import org.schabi.newpipe.util.external_communication.TimestampExtractor; | ||||
| import org.schabi.newpipe.util.external_communication.ShareUtils; | ||||
| import org.schabi.newpipe.util.PicassoHelper; | ||||
|  | ||||
| import java.util.regex.Matcher; | ||||
|  | ||||
| public class CommentsMiniInfoItemHolder extends InfoItemHolder { | ||||
|     private static final String TAG = "CommentsMiniIIHolder"; | ||||
|  | ||||
|     private static final int COMMENT_DEFAULT_LINES = 2; | ||||
|     private static final int COMMENT_EXPANDED_LINES = 1000; | ||||
|  | ||||
|     private final int commentHorizontalPadding; | ||||
|     private final int commentVerticalPadding; | ||||
|  | ||||
|     private final RelativeLayout itemRoot; | ||||
|     public final ImageView itemThumbnailView; | ||||
|     private final TextView itemContentView; | ||||
|     private final TextView itemLikesCountView; | ||||
|     private final TextView itemPublishedTime; | ||||
|  | ||||
|     private String commentText; | ||||
|     private String streamUrl; | ||||
|  | ||||
|     private final Linkify.TransformFilter timestampLink = new Linkify.TransformFilter() { | ||||
|         @Override | ||||
|         public String transformUrl(final Matcher match, final String url) { | ||||
|             try { | ||||
|                 final TimestampExtractor.TimestampMatchDTO timestampMatchDTO = | ||||
|                         TimestampExtractor.getTimestampFromMatcher(match, commentText); | ||||
|  | ||||
|                 if (timestampMatchDTO == null) { | ||||
|                     return url; | ||||
|                 } | ||||
|  | ||||
|                 return streamUrl + url.replace( | ||||
|                         match.group(0), | ||||
|                         "#timestamp=" + timestampMatchDTO.seconds()); | ||||
|             } catch (final Exception ex) { | ||||
|                 Log.e(TAG, "Unable to process url='" + url + "' as timestampLink", ex); | ||||
|                 return url; | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     CommentsMiniInfoItemHolder(final InfoItemBuilder infoItemBuilder, final int layoutId, | ||||
|                                final ViewGroup parent) { | ||||
|         super(infoItemBuilder, layoutId, parent); | ||||
|  | ||||
|         itemRoot = itemView.findViewById(R.id.itemRoot); | ||||
|         itemThumbnailView = itemView.findViewById(R.id.itemThumbnailView); | ||||
|         itemLikesCountView = itemView.findViewById(R.id.detail_thumbs_up_count_view); | ||||
|         itemPublishedTime = itemView.findViewById(R.id.itemPublishedTime); | ||||
|         itemContentView = itemView.findViewById(R.id.itemCommentContentView); | ||||
|  | ||||
|         commentHorizontalPadding = (int) infoItemBuilder.getContext() | ||||
|                 .getResources().getDimension(R.dimen.comments_horizontal_padding); | ||||
|         commentVerticalPadding = (int) infoItemBuilder.getContext() | ||||
|                 .getResources().getDimension(R.dimen.comments_vertical_padding); | ||||
|     } | ||||
|  | ||||
|     public CommentsMiniInfoItemHolder(final InfoItemBuilder infoItemBuilder, | ||||
|                                       final ViewGroup parent) { | ||||
|         this(infoItemBuilder, R.layout.list_comments_mini_item, parent); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void updateFromItem(final InfoItem infoItem, | ||||
|                                final HistoryRecordManager historyRecordManager) { | ||||
|         if (!(infoItem instanceof CommentsInfoItem)) { | ||||
|             return; | ||||
|         } | ||||
|         final CommentsInfoItem item = (CommentsInfoItem) infoItem; | ||||
|  | ||||
|         PicassoHelper.loadAvatar(item.getUploaderAvatarUrl()).into(itemThumbnailView); | ||||
|         if (PicassoHelper.getShouldLoadImages()) { | ||||
|             itemThumbnailView.setVisibility(View.VISIBLE); | ||||
|             itemRoot.setPadding(commentVerticalPadding, commentVerticalPadding, | ||||
|                     commentVerticalPadding, commentVerticalPadding); | ||||
|         } else { | ||||
|             itemThumbnailView.setVisibility(View.GONE); | ||||
|             itemRoot.setPadding(commentHorizontalPadding, commentVerticalPadding, | ||||
|                     commentHorizontalPadding, commentVerticalPadding); | ||||
|         } | ||||
|  | ||||
|  | ||||
|         itemThumbnailView.setOnClickListener(view -> openCommentAuthor(item)); | ||||
|  | ||||
|         streamUrl = item.getUrl(); | ||||
|  | ||||
|         itemContentView.setLines(COMMENT_DEFAULT_LINES); | ||||
|         commentText = item.getCommentText(); | ||||
|         itemContentView.setText(commentText); | ||||
|         itemContentView.setOnTouchListener(CommentTextOnTouchListener.INSTANCE); | ||||
|  | ||||
|         if (itemContentView.getLineCount() == 0) { | ||||
|             itemContentView.post(this::ellipsize); | ||||
|         } else { | ||||
|             ellipsize(); | ||||
|         } | ||||
|  | ||||
|         if (item.getLikeCount() >= 0) { | ||||
|             itemLikesCountView.setText( | ||||
|                     Localization.shortCount( | ||||
|                             itemBuilder.getContext(), | ||||
|                             item.getLikeCount())); | ||||
|         } else { | ||||
|             itemLikesCountView.setText("-"); | ||||
|         } | ||||
|  | ||||
|         if (item.getUploadDate() != null) { | ||||
|             itemPublishedTime.setText(Localization.relativeTime(item.getUploadDate() | ||||
|                     .offsetDateTime())); | ||||
|         } else { | ||||
|             itemPublishedTime.setText(item.getTextualUploadDate()); | ||||
|         } | ||||
|  | ||||
|         itemView.setOnClickListener(view -> { | ||||
|             toggleEllipsize(); | ||||
|             if (itemBuilder.getOnCommentsSelectedListener() != null) { | ||||
|                 itemBuilder.getOnCommentsSelectedListener().selected(item); | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|  | ||||
|         itemView.setOnLongClickListener(view -> { | ||||
|             if (DeviceUtils.isTv(itemBuilder.getContext())) { | ||||
|                 openCommentAuthor(item); | ||||
|             } else { | ||||
|                 ShareUtils.copyToClipboard(itemBuilder.getContext(), commentText); | ||||
|             } | ||||
|             return true; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     private void openCommentAuthor(final CommentsInfoItem item) { | ||||
|         if (TextUtils.isEmpty(item.getUploaderUrl())) { | ||||
|             return; | ||||
|         } | ||||
|         final AppCompatActivity activity = (AppCompatActivity) itemBuilder.getContext(); | ||||
|         try { | ||||
|             NavigationHelper.openChannelFragment( | ||||
|                     activity.getSupportFragmentManager(), | ||||
|                     item.getServiceId(), | ||||
|                     item.getUploaderUrl(), | ||||
|                     item.getUploaderName()); | ||||
|         } catch (final Exception e) { | ||||
|             ErrorUtil.showUiErrorSnackbar(activity, "Opening channel fragment", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void allowLinkFocus() { | ||||
|         itemContentView.setMovementMethod(LinkMovementMethod.getInstance()); | ||||
|     } | ||||
|  | ||||
|     private void denyLinkFocus() { | ||||
|         itemContentView.setMovementMethod(null); | ||||
|     } | ||||
|  | ||||
|     private boolean shouldFocusLinks() { | ||||
|         if (itemView.isInTouchMode()) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         final URLSpan[] urls = itemContentView.getUrls(); | ||||
|  | ||||
|         return urls != null && urls.length != 0; | ||||
|     } | ||||
|  | ||||
|     private void determineLinkFocus() { | ||||
|         if (shouldFocusLinks()) { | ||||
|             allowLinkFocus(); | ||||
|         } else { | ||||
|             denyLinkFocus(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void ellipsize() { | ||||
|         boolean hasEllipsis = false; | ||||
|  | ||||
|         if (itemContentView.getLineCount() > COMMENT_DEFAULT_LINES) { | ||||
|             final int endOfLastLine | ||||
|                     = itemContentView.getLayout().getLineEnd(COMMENT_DEFAULT_LINES - 1); | ||||
|             int end = itemContentView.getText().toString().lastIndexOf(' ', endOfLastLine - 2); | ||||
|             if (end == -1) { | ||||
|                 end = Math.max(endOfLastLine - 2, 0); | ||||
|             } | ||||
|             final String newVal = itemContentView.getText().subSequence(0, end) + " …"; | ||||
|             itemContentView.setText(newVal); | ||||
|             hasEllipsis = true; | ||||
|         } | ||||
|  | ||||
|         linkify(); | ||||
|  | ||||
|         if (hasEllipsis) { | ||||
|             denyLinkFocus(); | ||||
|         } else { | ||||
|             determineLinkFocus(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void toggleEllipsize() { | ||||
|         if (itemContentView.getText().toString().equals(commentText)) { | ||||
|             if (itemContentView.getLineCount() > COMMENT_DEFAULT_LINES) { | ||||
|                 ellipsize(); | ||||
|             } | ||||
|         } else { | ||||
|             expand(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private void expand() { | ||||
|         itemContentView.setMaxLines(COMMENT_EXPANDED_LINES); | ||||
|         itemContentView.setText(commentText); | ||||
|         linkify(); | ||||
|         determineLinkFocus(); | ||||
|     } | ||||
|  | ||||
|     private void linkify() { | ||||
|         Linkify.addLinks( | ||||
|                 itemContentView, | ||||
|                 Linkify.WEB_URLS); | ||||
|         Linkify.addLinks( | ||||
|                 itemContentView, | ||||
|                 TimestampExtractor.TIMESTAMPS_PATTERN, | ||||
|                 null, | ||||
|                 null, | ||||
|                 timestampLink); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,17 @@ | ||||
| package org.schabi.newpipe.info_list.holder; | ||||
|  | ||||
| import android.view.ViewGroup; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
|  | ||||
| /** | ||||
|  * Playlist card layout. | ||||
|  */ | ||||
| public class PlaylistCardInfoItemHolder extends PlaylistMiniInfoItemHolder { | ||||
|  | ||||
|     public PlaylistCardInfoItemHolder(final InfoItemBuilder infoItemBuilder, | ||||
|                                       final ViewGroup parent) { | ||||
|         super(infoItemBuilder, R.layout.list_playlist_card_item, parent); | ||||
|     } | ||||
| } | ||||
| @@ -9,7 +9,7 @@ import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.playlist.PlaylistInfoItem; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| import org.schabi.newpipe.util.PicassoHelper; | ||||
| import org.schabi.newpipe.util.image.PicassoHelper; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
|  | ||||
| public class PlaylistMiniInfoItemHolder extends InfoItemHolder { | ||||
| @@ -46,7 +46,7 @@ public class PlaylistMiniInfoItemHolder extends InfoItemHolder { | ||||
|                 .localizeStreamCountMini(itemStreamCountView.getContext(), item.getStreamCount())); | ||||
|         itemUploaderView.setText(item.getUploaderName()); | ||||
|  | ||||
|         PicassoHelper.loadPlaylistThumbnail(item.getThumbnailUrl()).into(itemThumbnailView); | ||||
|         PicassoHelper.loadPlaylistThumbnail(item.getThumbnails()).into(itemThumbnailView); | ||||
|  | ||||
|         itemView.setOnClickListener(view -> { | ||||
|             if (itemBuilder.getOnPlaylistSelectedListener() != null) { | ||||
|   | ||||
| @@ -0,0 +1,16 @@ | ||||
| package org.schabi.newpipe.info_list.holder; | ||||
|  | ||||
| import android.view.ViewGroup; | ||||
|  | ||||
| import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
|  | ||||
| /** | ||||
|  * Card layout for stream. | ||||
|  */ | ||||
| public class StreamCardInfoItemHolder extends StreamInfoItemHolder { | ||||
|  | ||||
|     public StreamCardInfoItemHolder(final InfoItemBuilder infoItemBuilder, final ViewGroup parent) { | ||||
|         super(infoItemBuilder, R.layout.list_stream_card_item, parent); | ||||
|     } | ||||
| } | ||||
| @@ -12,10 +12,6 @@ import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
|  | ||||
| import androidx.preference.PreferenceManager; | ||||
|  | ||||
| import static org.schabi.newpipe.MainActivity.DEBUG; | ||||
|  | ||||
| /* | ||||
|  * Created by Christian Schabesberger on 01.08.16. | ||||
|  * <p> | ||||
| @@ -81,7 +77,9 @@ public class StreamInfoItemHolder extends StreamMiniInfoItemHolder { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         final String uploadDate = getFormattedRelativeUploadDate(infoItem); | ||||
|         final String uploadDate = Localization.relativeTimeOrTextual(itemBuilder.getContext(), | ||||
|                 infoItem.getUploadDate(), | ||||
|                 infoItem.getTextualUploadDate()); | ||||
|         if (!TextUtils.isEmpty(uploadDate)) { | ||||
|             if (viewsAndDate.isEmpty()) { | ||||
|                 return uploadDate; | ||||
| @@ -92,20 +90,4 @@ public class StreamInfoItemHolder extends StreamMiniInfoItemHolder { | ||||
|  | ||||
|         return viewsAndDate; | ||||
|     } | ||||
|  | ||||
|     private String getFormattedRelativeUploadDate(final StreamInfoItem infoItem) { | ||||
|         if (infoItem.getUploadDate() != null) { | ||||
|             String formattedRelativeTime = Localization | ||||
|                     .relativeTime(infoItem.getUploadDate().offsetDateTime()); | ||||
|  | ||||
|             if (DEBUG && PreferenceManager.getDefaultSharedPreferences(itemBuilder.getContext()) | ||||
|                     .getBoolean(itemBuilder.getContext() | ||||
|                             .getString(R.string.show_original_time_ago_key), false)) { | ||||
|                 formattedRelativeTime += " (" + infoItem.getTextualUploadDate() + ")"; | ||||
|             } | ||||
|             return formattedRelativeTime; | ||||
|         } else { | ||||
|             return infoItem.getTextualUploadDate(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -11,12 +11,13 @@ import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.database.stream.model.StreamStateEntity; | ||||
| import org.schabi.newpipe.extractor.InfoItem; | ||||
| import org.schabi.newpipe.extractor.stream.StreamInfoItem; | ||||
| import org.schabi.newpipe.extractor.stream.StreamType; | ||||
| import org.schabi.newpipe.info_list.InfoItemBuilder; | ||||
| import org.schabi.newpipe.ktx.ViewUtils; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| import org.schabi.newpipe.util.PicassoHelper; | ||||
| import org.schabi.newpipe.util.DependentPreferenceHelper; | ||||
| import org.schabi.newpipe.util.Localization; | ||||
| import org.schabi.newpipe.util.image.PicassoHelper; | ||||
| import org.schabi.newpipe.util.StreamTypeUtil; | ||||
| import org.schabi.newpipe.views.AnimatedProgressBar; | ||||
|  | ||||
| import java.util.concurrent.TimeUnit; | ||||
| @@ -60,8 +61,12 @@ public class StreamMiniInfoItemHolder extends InfoItemHolder { | ||||
|                     R.color.duration_background_color)); | ||||
|             itemDurationView.setVisibility(View.VISIBLE); | ||||
|  | ||||
|             final StreamStateEntity state2 = historyRecordManager.loadStreamState(infoItem) | ||||
|                     .blockingGet()[0]; | ||||
|             StreamStateEntity state2 = null; | ||||
|             if (DependentPreferenceHelper | ||||
|                     .getPositionsInListsEnabled(itemProgressView.getContext())) { | ||||
|                 state2 = historyRecordManager.loadStreamState(infoItem) | ||||
|                         .blockingGet()[0]; | ||||
|             } | ||||
|             if (state2 != null) { | ||||
|                 itemProgressView.setVisibility(View.VISIBLE); | ||||
|                 itemProgressView.setMax((int) item.getDuration()); | ||||
| @@ -70,8 +75,7 @@ public class StreamMiniInfoItemHolder extends InfoItemHolder { | ||||
|             } else { | ||||
|                 itemProgressView.setVisibility(View.GONE); | ||||
|             } | ||||
|         } else if (item.getStreamType() == StreamType.LIVE_STREAM | ||||
|                 || item.getStreamType() == StreamType.AUDIO_LIVE_STREAM) { | ||||
|         } else if (StreamTypeUtil.isLiveStream(item.getStreamType())) { | ||||
|             itemDurationView.setText(R.string.duration_live); | ||||
|             itemDurationView.setBackgroundColor(ContextCompat.getColor(itemBuilder.getContext(), | ||||
|                     R.color.live_duration_background_color)); | ||||
| @@ -83,7 +87,7 @@ public class StreamMiniInfoItemHolder extends InfoItemHolder { | ||||
|         } | ||||
|  | ||||
|         // Default thumbnail is shown on error, while loading and if the url is empty | ||||
|         PicassoHelper.loadThumbnail(item.getThumbnailUrl()).into(itemThumbnailView); | ||||
|         PicassoHelper.loadThumbnail(item.getThumbnails()).into(itemThumbnailView); | ||||
|  | ||||
|         itemView.setOnClickListener(view -> { | ||||
|             if (itemBuilder.getOnStreamSelectedListener() != null) { | ||||
| @@ -96,9 +100,10 @@ public class StreamMiniInfoItemHolder extends InfoItemHolder { | ||||
|             case VIDEO_STREAM: | ||||
|             case LIVE_STREAM: | ||||
|             case AUDIO_LIVE_STREAM: | ||||
|             case POST_LIVE_STREAM: | ||||
|             case POST_LIVE_AUDIO_STREAM: | ||||
|                 enableLongClick(item); | ||||
|                 break; | ||||
|             case FILE: | ||||
|             case NONE: | ||||
|             default: | ||||
|                 disableLongClick(); | ||||
| @@ -111,10 +116,14 @@ public class StreamMiniInfoItemHolder extends InfoItemHolder { | ||||
|                             final HistoryRecordManager historyRecordManager) { | ||||
|         final StreamInfoItem item = (StreamInfoItem) infoItem; | ||||
|  | ||||
|         final StreamStateEntity state | ||||
|                 = historyRecordManager.loadStreamState(infoItem).blockingGet()[0]; | ||||
|         StreamStateEntity state = null; | ||||
|         if (DependentPreferenceHelper.getPositionsInListsEnabled(itemProgressView.getContext())) { | ||||
|             state = historyRecordManager | ||||
|                     .loadStreamState(infoItem) | ||||
|                     .blockingGet()[0]; | ||||
|         } | ||||
|         if (state != null && item.getDuration() > 0 | ||||
|                 && item.getStreamType() != StreamType.LIVE_STREAM) { | ||||
|                 && !StreamTypeUtil.isLiveStream(item.getStreamType())) { | ||||
|             itemProgressView.setMax((int) item.getDuration()); | ||||
|             if (itemProgressView.getVisibility() == View.VISIBLE) { | ||||
|                 itemProgressView.setProgressAnimated((int) TimeUnit.MILLISECONDS | ||||
|   | ||||
							
								
								
									
										9
									
								
								app/src/main/java/org/schabi/newpipe/ktx/Bundle.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								app/src/main/java/org/schabi/newpipe/ktx/Bundle.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| package org.schabi.newpipe.ktx | ||||
|  | ||||
| import android.os.Bundle | ||||
| import android.os.Parcelable | ||||
| import androidx.core.os.BundleCompat | ||||
|  | ||||
| inline fun <reified T : Parcelable> Bundle.parcelableArrayList(key: String?): ArrayList<T>? { | ||||
|     return BundleCompat.getParcelableArrayList(this, key, T::class.java) | ||||
| } | ||||
| @@ -21,10 +21,6 @@ import org.schabi.newpipe.MainActivity | ||||
|  | ||||
| private const val TAG = "ViewUtils" | ||||
|  | ||||
| inline var View.backgroundTintListCompat: ColorStateList? | ||||
|     get() = ViewCompat.getBackgroundTintList(this) | ||||
|     set(value) = ViewCompat.setBackgroundTintList(this, value) | ||||
|  | ||||
| /** | ||||
|  * Animate the view. | ||||
|  * | ||||
| @@ -96,62 +92,43 @@ fun View.animateBackgroundColor(duration: Long, @ColorInt colorStart: Int, @Colo | ||||
|     if (MainActivity.DEBUG) { | ||||
|         Log.d( | ||||
|             TAG, | ||||
|             "animateBackgroundColor() called with: " + | ||||
|                 "view = [" + this + "], duration = [" + duration + "], " + | ||||
|                 "colorStart = [" + colorStart + "], colorEnd = [" + colorEnd + "]" | ||||
|             "animateBackgroundColor() called with: view = [$this], duration = [$duration], " + | ||||
|                 "colorStart = [$colorStart], colorEnd = [$colorEnd]" | ||||
|         ) | ||||
|     } | ||||
|     val empty = arrayOf(IntArray(0)) | ||||
|     val viewPropertyAnimator = ValueAnimator.ofObject(ArgbEvaluator(), colorStart, colorEnd) | ||||
|     viewPropertyAnimator.interpolator = FastOutSlowInInterpolator() | ||||
|     viewPropertyAnimator.duration = duration | ||||
|     viewPropertyAnimator.addUpdateListener { animation: ValueAnimator -> | ||||
|         backgroundTintListCompat = ColorStateList(empty, intArrayOf(animation.animatedValue as Int)) | ||||
|  | ||||
|     fun listenerAction(color: Int) { | ||||
|         ViewCompat.setBackgroundTintList(this, ColorStateList.valueOf(color)) | ||||
|     } | ||||
|     viewPropertyAnimator.addListener( | ||||
|         onCancel = { backgroundTintListCompat = ColorStateList(empty, intArrayOf(colorEnd)) }, | ||||
|         onEnd = { backgroundTintListCompat = ColorStateList(empty, intArrayOf(colorEnd)) } | ||||
|     ) | ||||
|     viewPropertyAnimator.addUpdateListener { listenerAction(it.animatedValue as Int) } | ||||
|     viewPropertyAnimator.addListener(onCancel = { listenerAction(colorEnd) }, onEnd = { listenerAction(colorEnd) }) | ||||
|     viewPropertyAnimator.start() | ||||
| } | ||||
|  | ||||
| fun View.animateHeight(duration: Long, targetHeight: Int): ValueAnimator { | ||||
|     if (MainActivity.DEBUG) { | ||||
|         Log.d( | ||||
|             TAG, | ||||
|             "animateHeight: duration = [" + duration + "], " + | ||||
|                 "from " + height + " to → " + targetHeight + " in: " + this | ||||
|         ) | ||||
|         Log.d(TAG, "animateHeight: duration = [$duration], from $height to → $targetHeight in: $this") | ||||
|     } | ||||
|     val animator = ValueAnimator.ofFloat(height.toFloat(), targetHeight.toFloat()) | ||||
|     animator.interpolator = FastOutSlowInInterpolator() | ||||
|     animator.duration = duration | ||||
|     animator.addUpdateListener { animation: ValueAnimator -> | ||||
|         val value = animation.animatedValue as Float | ||||
|         layoutParams.height = value.toInt() | ||||
|  | ||||
|     fun listenerAction(value: Int) { | ||||
|         layoutParams.height = value | ||||
|         requestLayout() | ||||
|     } | ||||
|     animator.addListener( | ||||
|         onCancel = { | ||||
|             layoutParams.height = targetHeight | ||||
|             requestLayout() | ||||
|         }, | ||||
|         onEnd = { | ||||
|             layoutParams.height = targetHeight | ||||
|             requestLayout() | ||||
|         } | ||||
|     ) | ||||
|     animator.addUpdateListener { listenerAction((it.animatedValue as Float).toInt()) } | ||||
|     animator.addListener(onCancel = { listenerAction(targetHeight) }, onEnd = { listenerAction(targetHeight) }) | ||||
|     animator.start() | ||||
|     return animator | ||||
| } | ||||
|  | ||||
| fun View.animateRotation(duration: Long, targetRotation: Int) { | ||||
|     if (MainActivity.DEBUG) { | ||||
|         Log.d( | ||||
|             TAG, | ||||
|             "animateRotation: duration = [" + duration + "], " + | ||||
|                 "from " + rotation + " to → " + targetRotation + " in: " + this | ||||
|         ) | ||||
|         Log.d(TAG, "animateRotation: duration = [$duration], from $rotation to → $targetRotation in: $this") | ||||
|     } | ||||
|     animate().setListener(null).cancel() | ||||
|     animate() | ||||
| @@ -172,20 +149,13 @@ private fun View.animateAlpha(enterOrExit: Boolean, duration: Long, delay: Long, | ||||
|     if (enterOrExit) { | ||||
|         animate().setInterpolator(FastOutSlowInInterpolator()).alpha(1f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(ExecOnEndListener(execOnEnd)) | ||||
|             .start() | ||||
|     } else { | ||||
|         animate().setInterpolator(FastOutSlowInInterpolator()).alpha(0f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     isGone = true | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(HideAndExecOnEndListener(this, execOnEnd)) | ||||
|             .start() | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -197,11 +167,8 @@ private fun View.animateScaleAndAlpha(enterOrExit: Boolean, duration: Long, dela | ||||
|             .setInterpolator(FastOutSlowInInterpolator()) | ||||
|             .alpha(1f).scaleX(1f).scaleY(1f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(ExecOnEndListener(execOnEnd)) | ||||
|             .start() | ||||
|     } else { | ||||
|         scaleX = 1f | ||||
|         scaleY = 1f | ||||
| @@ -209,12 +176,8 @@ private fun View.animateScaleAndAlpha(enterOrExit: Boolean, duration: Long, dela | ||||
|             .setInterpolator(FastOutSlowInInterpolator()) | ||||
|             .alpha(0f).scaleX(.8f).scaleY(.8f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     isGone = true | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(HideAndExecOnEndListener(this, execOnEnd)) | ||||
|             .start() | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -227,11 +190,8 @@ private fun View.animateLightScaleAndAlpha(enterOrExit: Boolean, duration: Long, | ||||
|             .setInterpolator(FastOutSlowInInterpolator()) | ||||
|             .alpha(1f).scaleX(1f).scaleY(1f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(ExecOnEndListener(execOnEnd)) | ||||
|             .start() | ||||
|     } else { | ||||
|         alpha = 1f | ||||
|         scaleX = 1f | ||||
| @@ -240,12 +200,8 @@ private fun View.animateLightScaleAndAlpha(enterOrExit: Boolean, duration: Long, | ||||
|             .setInterpolator(FastOutSlowInInterpolator()) | ||||
|             .alpha(0f).scaleX(.95f).scaleY(.95f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     isGone = true | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(HideAndExecOnEndListener(this, execOnEnd)) | ||||
|             .start() | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -256,22 +212,15 @@ private fun View.animateSlideAndAlpha(enterOrExit: Boolean, duration: Long, dela | ||||
|         animate() | ||||
|             .setInterpolator(FastOutSlowInInterpolator()).alpha(1f).translationY(0f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(ExecOnEndListener(execOnEnd)) | ||||
|             .start() | ||||
|     } else { | ||||
|         animate() | ||||
|             .setInterpolator(FastOutSlowInInterpolator()) | ||||
|             .alpha(0f).translationY(-height.toFloat()) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     isGone = true | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(HideAndExecOnEndListener(this, execOnEnd)) | ||||
|             .start() | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -282,32 +231,18 @@ private fun View.animateLightSlideAndAlpha(enterOrExit: Boolean, duration: Long, | ||||
|         animate() | ||||
|             .setInterpolator(FastOutSlowInInterpolator()).alpha(1f).translationY(0f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(ExecOnEndListener(execOnEnd)) | ||||
|             .start() | ||||
|     } else { | ||||
|         animate().setInterpolator(FastOutSlowInInterpolator()) | ||||
|             .alpha(0f).translationY(-height / 2.0f) | ||||
|             .setDuration(duration).setStartDelay(delay) | ||||
|             .setListener(object : AnimatorListenerAdapter() { | ||||
|                 override fun onAnimationEnd(animation: Animator) { | ||||
|                     isGone = true | ||||
|                     execOnEnd?.run() | ||||
|                 } | ||||
|             }).start() | ||||
|             .setListener(HideAndExecOnEndListener(this, execOnEnd)) | ||||
|             .start() | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun View.slideUp( | ||||
|     duration: Long, | ||||
|     delay: Long, | ||||
|     @FloatRange(from = 0.0, to = 1.0) translationPercent: Float | ||||
| ) { | ||||
|     slideUp(duration, delay, translationPercent, null) | ||||
| } | ||||
|  | ||||
| @JvmOverloads | ||||
| fun View.slideUp( | ||||
|     duration: Long, | ||||
|     delay: Long = 0L, | ||||
| @@ -325,11 +260,7 @@ fun View.slideUp( | ||||
|         .setStartDelay(delay) | ||||
|         .setDuration(duration) | ||||
|         .setInterpolator(FastOutSlowInInterpolator()) | ||||
|         .setListener(object : AnimatorListenerAdapter() { | ||||
|             override fun onAnimationEnd(animation: Animator) { | ||||
|                 execOnEnd?.run() | ||||
|             } | ||||
|         }) | ||||
|         .setListener(ExecOnEndListener(execOnEnd)) | ||||
|         .start() | ||||
| } | ||||
|  | ||||
| @@ -343,6 +274,20 @@ fun View.animateHideRecyclerViewAllowingScrolling() { | ||||
|     animate().alpha(0.0f).setDuration(200).start() | ||||
| } | ||||
|  | ||||
| private open class ExecOnEndListener(private val execOnEnd: Runnable?) : AnimatorListenerAdapter() { | ||||
|     override fun onAnimationEnd(animation: Animator) { | ||||
|         execOnEnd?.run() | ||||
|     } | ||||
| } | ||||
|  | ||||
| private class HideAndExecOnEndListener(private val view: View, execOnEnd: Runnable?) : | ||||
|     ExecOnEndListener(execOnEnd) { | ||||
|     override fun onAnimationEnd(animation: Animator) { | ||||
|         view.isGone = true | ||||
|         super.onAnimationEnd(animation) | ||||
|     } | ||||
| } | ||||
|  | ||||
| enum class AnimationType { | ||||
|     ALPHA, SCALE_AND_ALPHA, LIGHT_SCALE_AND_ALPHA, SLIDE_AND_ALPHA, LIGHT_SLIDE_AND_ALPHA | ||||
| } | ||||
|   | ||||
| @@ -22,10 +22,11 @@ import org.schabi.newpipe.R; | ||||
| import org.schabi.newpipe.databinding.PignateFooterBinding; | ||||
| import org.schabi.newpipe.fragments.BaseStateFragment; | ||||
| import org.schabi.newpipe.fragments.list.ListViewContract; | ||||
| import org.schabi.newpipe.info_list.ItemViewMode; | ||||
|  | ||||
| import static org.schabi.newpipe.ktx.ViewUtils.animate; | ||||
| import static org.schabi.newpipe.ktx.ViewUtils.animateHideRecyclerViewAllowingScrolling; | ||||
| import static org.schabi.newpipe.util.ThemeHelper.shouldUseGridLayout; | ||||
| import static org.schabi.newpipe.util.ThemeHelper.getItemViewMode; | ||||
|  | ||||
| /** | ||||
|  * This fragment is design to be used with persistent data such as | ||||
| @@ -77,16 +78,23 @@ public abstract class BaseLocalListFragment<I, N> extends BaseStateFragment<I> | ||||
|         super.onResume(); | ||||
|         if (updateFlags != 0) { | ||||
|             if ((updateFlags & LIST_MODE_UPDATE_FLAG) != 0) { | ||||
|                 final boolean useGrid = shouldUseGridLayout(requireContext()); | ||||
|                 itemsList.setLayoutManager( | ||||
|                         useGrid ? getGridLayoutManager() : getListLayoutManager()); | ||||
|                 itemListAdapter.setUseGridVariant(useGrid); | ||||
|                 itemListAdapter.notifyDataSetChanged(); | ||||
|                 refreshItemViewMode(); | ||||
|             } | ||||
|             updateFlags = 0; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Updates the item view mode based on user preference. | ||||
|      */ | ||||
|     private void refreshItemViewMode() { | ||||
|         final ItemViewMode itemViewMode = getItemViewMode(requireContext()); | ||||
|         itemsList.setLayoutManager((itemViewMode == ItemViewMode.GRID) | ||||
|                 ? getGridLayoutManager() : getListLayoutManager()); | ||||
|         itemListAdapter.setItemViewMode(itemViewMode); | ||||
|         itemListAdapter.notifyDataSetChanged(); | ||||
|     } | ||||
|  | ||||
|     /*////////////////////////////////////////////////////////////////////////// | ||||
|     // Lifecycle - View | ||||
|     //////////////////////////////////////////////////////////////////////////*/ | ||||
| @@ -104,8 +112,7 @@ public abstract class BaseLocalListFragment<I, N> extends BaseStateFragment<I> | ||||
|         final Resources resources = activity.getResources(); | ||||
|         int width = resources.getDimensionPixelSize(R.dimen.video_item_grid_thumbnail_image_width); | ||||
|         width += (24 * resources.getDisplayMetrics().density); | ||||
|         final int spanCount = (int) Math.floor(resources.getDisplayMetrics().widthPixels | ||||
|                 / (double) width); | ||||
|         final int spanCount = Math.floorDiv(resources.getDisplayMetrics().widthPixels, width); | ||||
|         final GridLayoutManager lm = new GridLayoutManager(activity, spanCount); | ||||
|         lm.setSpanSizeLookup(itemListAdapter.getSpanSizeLookup(spanCount)); | ||||
|         return lm; | ||||
| @@ -121,11 +128,9 @@ public abstract class BaseLocalListFragment<I, N> extends BaseStateFragment<I> | ||||
|  | ||||
|         itemListAdapter = new LocalItemListAdapter(activity); | ||||
|  | ||||
|         final boolean useGrid = shouldUseGridLayout(requireContext()); | ||||
|         itemsList = rootView.findViewById(R.id.items_list); | ||||
|         itemsList.setLayoutManager(useGrid ? getGridLayoutManager() : getListLayoutManager()); | ||||
|         refreshItemViewMode(); | ||||
|  | ||||
|         itemListAdapter.setUseGridVariant(useGrid); | ||||
|         headerRootBinding = getListHeader(); | ||||
|         if (headerRootBinding != null) { | ||||
|             itemListAdapter.setHeader(headerRootBinding.getRoot()); | ||||
| @@ -256,7 +261,7 @@ public abstract class BaseLocalListFragment<I, N> extends BaseStateFragment<I> | ||||
|     @Override | ||||
|     public void onSharedPreferenceChanged(final SharedPreferences sharedPreferences, | ||||
|                                           final String key) { | ||||
|         if (key.equals(getString(R.string.list_view_mode_key))) { | ||||
|         if (getString(R.string.list_view_mode_key).equals(key)) { | ||||
|             updateFlags |= LIST_MODE_UPDATE_FLAG; | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -12,16 +12,21 @@ import androidx.recyclerview.widget.RecyclerView; | ||||
|  | ||||
| import org.schabi.newpipe.database.LocalItem; | ||||
| import org.schabi.newpipe.database.stream.model.StreamStateEntity; | ||||
| import org.schabi.newpipe.info_list.ItemViewMode; | ||||
| import org.schabi.newpipe.local.history.HistoryRecordManager; | ||||
| import org.schabi.newpipe.local.holder.LocalBookmarkPlaylistItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalPlaylistCardItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalPlaylistGridItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalPlaylistItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalPlaylistStreamCardItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalPlaylistStreamGridItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalPlaylistStreamItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalStatisticStreamCardItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalStatisticStreamGridItemHolder; | ||||
| import org.schabi.newpipe.local.holder.LocalStatisticStreamItemHolder; | ||||
| import org.schabi.newpipe.local.holder.RemoteBookmarkPlaylistItemHolder; | ||||
| import org.schabi.newpipe.local.holder.RemotePlaylistCardItemHolder; | ||||
| import org.schabi.newpipe.local.holder.RemotePlaylistGridItemHolder; | ||||
| import org.schabi.newpipe.local.holder.RemotePlaylistItemHolder; | ||||
| import org.schabi.newpipe.util.FallbackViewHolder; | ||||
| @@ -63,13 +68,19 @@ public class LocalItemListAdapter extends RecyclerView.Adapter<RecyclerView.View | ||||
|     private static final int STREAM_STATISTICS_HOLDER_TYPE = 0x1000; | ||||
|     private static final int STREAM_PLAYLIST_HOLDER_TYPE = 0x1001; | ||||
|     private static final int STREAM_STATISTICS_GRID_HOLDER_TYPE = 0x1002; | ||||
|     private static final int STREAM_STATISTICS_CARD_HOLDER_TYPE = 0x1003; | ||||
|     private static final int STREAM_PLAYLIST_GRID_HOLDER_TYPE = 0x1004; | ||||
|     private static final int STREAM_PLAYLIST_CARD_HOLDER_TYPE = 0x1005; | ||||
|  | ||||
|     private static final int LOCAL_PLAYLIST_HOLDER_TYPE = 0x2000; | ||||
|     private static final int REMOTE_PLAYLIST_HOLDER_TYPE = 0x2001; | ||||
|     private static final int LOCAL_PLAYLIST_GRID_HOLDER_TYPE = 0x2002; | ||||
|     private static final int REMOTE_PLAYLIST_GRID_HOLDER_TYPE = 0x2004; | ||||
|     private static final int LOCAL_BOOKMARK_PLAYLIST_HOLDER_TYPE = 0x2008; | ||||
|     private static final int REMOTE_BOOKMARK_PLAYLIST_HOLDER_TYPE = 0x2010; | ||||
|     private static final int LOCAL_PLAYLIST_GRID_HOLDER_TYPE = 0x2001; | ||||
|     private static final int LOCAL_PLAYLIST_CARD_HOLDER_TYPE = 0x2002; | ||||
|     private static final int LOCAL_BOOKMARK_PLAYLIST_HOLDER_TYPE = 0x2003; | ||||
|  | ||||
|     private static final int REMOTE_PLAYLIST_HOLDER_TYPE = 0x3000; | ||||
|     private static final int REMOTE_PLAYLIST_GRID_HOLDER_TYPE = 0x3001; | ||||
|     private static final int REMOTE_PLAYLIST_CARD_HOLDER_TYPE = 0x3002; | ||||
|     private static final int REMOTE_BOOKMARK_PLAYLIST_HOLDER_TYPE = 0x3003; | ||||
|  | ||||
|     private final LocalItemBuilder localItemBuilder; | ||||
|     private final ArrayList<LocalItem> localItems; | ||||
| @@ -77,10 +88,10 @@ public class LocalItemListAdapter extends RecyclerView.Adapter<RecyclerView.View | ||||
|     private final DateTimeFormatter dateTimeFormatter; | ||||
|  | ||||
|     private boolean showFooter = false; | ||||
|     private boolean useGridVariant = false; | ||||
|     private boolean useItemHandle = false; | ||||
|     private View header = null; | ||||
|     private View footer = null; | ||||
|     private ItemViewMode itemViewMode = ItemViewMode.LIST; | ||||
|     private boolean useItemHandle = false; | ||||
|  | ||||
|     public LocalItemListAdapter(final Context context) { | ||||
|         recordManager = new HistoryRecordManager(context); | ||||
| @@ -170,8 +181,8 @@ public class LocalItemListAdapter extends RecyclerView.Adapter<RecyclerView.View | ||||
|         notifyDataSetChanged(); | ||||
|     } | ||||
|  | ||||
|     public void setUseGridVariant(final boolean useGridVariant) { | ||||
|         this.useGridVariant = useGridVariant; | ||||
|     public void setItemViewMode(final ItemViewMode itemViewMode) { | ||||
|         this.itemViewMode = itemViewMode; | ||||
|     } | ||||
|  | ||||
|     public void setUseItemHandle(final boolean useItemHandle) { | ||||
| @@ -253,26 +264,43 @@ public class LocalItemListAdapter extends RecyclerView.Adapter<RecyclerView.View | ||||
|             return FOOTER_TYPE; | ||||
|         } | ||||
|         final LocalItem item = localItems.get(position); | ||||
|  | ||||
|         switch (item.getLocalItemType()) { | ||||
|             case PLAYLIST_LOCAL_ITEM: | ||||
|                 if (useItemHandle) { | ||||
|                     return LOCAL_BOOKMARK_PLAYLIST_HOLDER_TYPE; | ||||
|                 } else if (itemViewMode == ItemViewMode.CARD) { | ||||
|                     return LOCAL_PLAYLIST_CARD_HOLDER_TYPE; | ||||
|                 } else if (itemViewMode == ItemViewMode.GRID) { | ||||
|                     return LOCAL_PLAYLIST_GRID_HOLDER_TYPE; | ||||
|                 } else { | ||||
|                     return LOCAL_PLAYLIST_HOLDER_TYPE; | ||||
|                 } | ||||
|                 return useGridVariant ? LOCAL_PLAYLIST_GRID_HOLDER_TYPE | ||||
|                         : LOCAL_PLAYLIST_HOLDER_TYPE; | ||||
|             case PLAYLIST_REMOTE_ITEM: | ||||
|                 if (useItemHandle) { | ||||
|                     return REMOTE_BOOKMARK_PLAYLIST_HOLDER_TYPE; | ||||
|                 } else if (itemViewMode == ItemViewMode.CARD) { | ||||
|                     return REMOTE_PLAYLIST_CARD_HOLDER_TYPE; | ||||
|                 } else if (itemViewMode == ItemViewMode.GRID) { | ||||
|                     return REMOTE_PLAYLIST_GRID_HOLDER_TYPE; | ||||
|                 } else { | ||||
|                     return REMOTE_PLAYLIST_HOLDER_TYPE; | ||||
|                 } | ||||
|                 return useGridVariant ? REMOTE_PLAYLIST_GRID_HOLDER_TYPE | ||||
|                         : REMOTE_PLAYLIST_HOLDER_TYPE; | ||||
|             case PLAYLIST_STREAM_ITEM: | ||||
|                 return useGridVariant | ||||
|                         ? STREAM_PLAYLIST_GRID_HOLDER_TYPE : STREAM_PLAYLIST_HOLDER_TYPE; | ||||
|                 if (itemViewMode == ItemViewMode.CARD) { | ||||
|                     return STREAM_PLAYLIST_CARD_HOLDER_TYPE; | ||||
|                 } else if (itemViewMode == ItemViewMode.GRID) { | ||||
|                     return STREAM_PLAYLIST_GRID_HOLDER_TYPE; | ||||
|                 } else { | ||||
|                     return STREAM_PLAYLIST_HOLDER_TYPE; | ||||
|                 } | ||||
|             case STATISTIC_STREAM_ITEM: | ||||
|                 return useGridVariant | ||||
|                         ? STREAM_STATISTICS_GRID_HOLDER_TYPE : STREAM_STATISTICS_HOLDER_TYPE; | ||||
|                 if (itemViewMode == ItemViewMode.CARD) { | ||||
|                     return STREAM_STATISTICS_CARD_HOLDER_TYPE; | ||||
|                 } else if (itemViewMode == ItemViewMode.GRID) { | ||||
|                     return STREAM_STATISTICS_GRID_HOLDER_TYPE; | ||||
|                 } else { | ||||
|                     return STREAM_STATISTICS_HOLDER_TYPE; | ||||
|                 } | ||||
|             default: | ||||
|                 Log.e(TAG, "No holder type has been considered for item: [" | ||||
|                         + item.getLocalItemType() + "]"); | ||||
| @@ -297,22 +325,30 @@ public class LocalItemListAdapter extends RecyclerView.Adapter<RecyclerView.View | ||||
|                 return new LocalPlaylistItemHolder(localItemBuilder, parent); | ||||
|             case LOCAL_PLAYLIST_GRID_HOLDER_TYPE: | ||||
|                 return new LocalPlaylistGridItemHolder(localItemBuilder, parent); | ||||
|             case LOCAL_PLAYLIST_CARD_HOLDER_TYPE: | ||||
|                 return new LocalPlaylistCardItemHolder(localItemBuilder, parent); | ||||
|             case LOCAL_BOOKMARK_PLAYLIST_HOLDER_TYPE: | ||||
|                 return new LocalBookmarkPlaylistItemHolder(localItemBuilder, parent); | ||||
|             case REMOTE_PLAYLIST_HOLDER_TYPE: | ||||
|                 return new RemotePlaylistItemHolder(localItemBuilder, parent); | ||||
|             case REMOTE_PLAYLIST_GRID_HOLDER_TYPE: | ||||
|                 return new RemotePlaylistGridItemHolder(localItemBuilder, parent); | ||||
|             case REMOTE_PLAYLIST_CARD_HOLDER_TYPE: | ||||
|                 return new RemotePlaylistCardItemHolder(localItemBuilder, parent); | ||||
|             case REMOTE_BOOKMARK_PLAYLIST_HOLDER_TYPE: | ||||
|                 return new RemoteBookmarkPlaylistItemHolder(localItemBuilder, parent); | ||||
|             case STREAM_PLAYLIST_HOLDER_TYPE: | ||||
|                 return new LocalPlaylistStreamItemHolder(localItemBuilder, parent); | ||||
|             case STREAM_PLAYLIST_GRID_HOLDER_TYPE: | ||||
|                 return new LocalPlaylistStreamGridItemHolder(localItemBuilder, parent); | ||||
|             case STREAM_PLAYLIST_CARD_HOLDER_TYPE: | ||||
|                 return new LocalPlaylistStreamCardItemHolder(localItemBuilder, parent); | ||||
|             case STREAM_STATISTICS_HOLDER_TYPE: | ||||
|                 return new LocalStatisticStreamItemHolder(localItemBuilder, parent); | ||||
|             case STREAM_STATISTICS_GRID_HOLDER_TYPE: | ||||
|                 return new LocalStatisticStreamGridItemHolder(localItemBuilder, parent); | ||||
|             case STREAM_STATISTICS_CARD_HOLDER_TYPE: | ||||
|                 return new LocalStatisticStreamCardItemHolder(localItemBuilder, parent); | ||||
|             default: | ||||
|                 Log.e(TAG, "No view type has been considered for holder: [" + type + "]"); | ||||
|                 return new FallbackViewHolder(new View(parent.getContext())); | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	 Stypox
					Stypox