mirror of
https://github.com/ReVanced/revanced-patches.git
synced 2026-01-20 17:43:56 +00:00
Compare commits
356 Commits
v5.29.0-de
...
v5.40.0-de
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
729997ec3e | ||
|
|
767f1e3695 | ||
|
|
7857876551 | ||
|
|
04057c6e56 | ||
|
|
8ba9a19ade | ||
|
|
6862200a28 | ||
|
|
dfff3d7c0a | ||
|
|
e6cce85541 | ||
|
|
8502eb8eac | ||
|
|
0652c56d0d | ||
|
|
b7026b7086 | ||
|
|
fa4f422a15 | ||
|
|
38e0cbd724 | ||
|
|
0bdebd927d | ||
|
|
3eac25cf7f | ||
|
|
c9f741e616 | ||
|
|
cba44ccfc8 | ||
|
|
a84db7be7f | ||
|
|
2520129ace | ||
|
|
7eeffd3392 | ||
|
|
6c3391164e | ||
|
|
0b8b46c73e | ||
|
|
cbe576bc38 | ||
|
|
3a29f2a805 | ||
|
|
50069c7e05 | ||
|
|
2e9c9dc244 | ||
|
|
56166896d9 | ||
|
|
b4c695b1d5 | ||
|
|
1475643f84 | ||
|
|
9a7179f9cf | ||
|
|
6fb94a7a41 | ||
|
|
3776dda710 | ||
|
|
f88b3a5162 | ||
|
|
0eeaf7ad67 | ||
|
|
2726231404 | ||
|
|
9f0558e494 | ||
|
|
01f7bc9f8d | ||
|
|
5e20bd80f1 | ||
|
|
f304c178e2 | ||
|
|
1d65887e01 | ||
|
|
6b6eea8414 | ||
|
|
1db131e90e | ||
|
|
abe3943f98 | ||
|
|
cb6d802de3 | ||
|
|
f11d1ef990 | ||
|
|
3d25da18bc | ||
|
|
fa04c8eecf | ||
|
|
105f6e0e97 | ||
|
|
7d59efe05d | ||
|
|
81ff5576b0 | ||
|
|
9a5c102c0d | ||
|
|
e6c79f1383 | ||
|
|
2a582eced8 | ||
|
|
2db0948bea | ||
|
|
a3ba92e742 | ||
|
|
2a85a3b290 | ||
|
|
eee72208dd | ||
|
|
dcd42454bd | ||
|
|
782353c18a | ||
|
|
b53b870e8f | ||
|
|
09b941abf0 | ||
|
|
678ef4052e | ||
|
|
0abfab79d7 | ||
|
|
61cadf72cd | ||
|
|
e12359b94f | ||
|
|
c001daba4a | ||
|
|
e136f62d6e | ||
|
|
8ec405a359 | ||
|
|
2f4b3a887b | ||
|
|
d1fabb242b | ||
|
|
a53b00dd51 | ||
|
|
850c13e98e | ||
|
|
4310789a26 | ||
|
|
c4a720fbd3 | ||
|
|
3bdb8dbce0 | ||
|
|
4894f33c96 | ||
|
|
7f6093ee66 | ||
|
|
9d4aa5cd16 | ||
|
|
5ace6f587c | ||
|
|
796f56745e | ||
|
|
88b47ef414 | ||
|
|
8cd8e59bbc | ||
|
|
6e72b14d07 | ||
|
|
52b088327b | ||
|
|
8e934cc56b | ||
|
|
b3140d909b | ||
|
|
97645aa9f4 | ||
|
|
603e2d018c | ||
|
|
144af2f07e | ||
|
|
b8629aacb6 | ||
|
|
3951527f51 | ||
|
|
7a8b618c4e | ||
|
|
c66c42e946 | ||
|
|
b340769cf3 | ||
|
|
0a8cd7a7db | ||
|
|
39f90e4b11 | ||
|
|
9256aa4548 | ||
|
|
7973c75552 | ||
|
|
2b2307416a | ||
|
|
1dbc2d4057 | ||
|
|
f6917dc361 | ||
|
|
d2f043e11a | ||
|
|
a392bc0dfd | ||
|
|
dfc127048a | ||
|
|
ed31d0cab6 | ||
|
|
0df6315f9c | ||
|
|
f14259f9ef | ||
|
|
1473db0bef | ||
|
|
829ca58a55 | ||
|
|
aace741e25 | ||
|
|
189529151a | ||
|
|
51237c177a | ||
|
|
23496c7c36 | ||
|
|
e6823d8924 | ||
|
|
43597dab21 | ||
|
|
c0824db142 | ||
|
|
1b7f84b7fa | ||
|
|
6d87c848d6 | ||
|
|
150bee2833 | ||
|
|
c3ee6eca44 | ||
|
|
01a04c338c | ||
|
|
3130225d9d | ||
|
|
16b27fb872 | ||
|
|
bedabd3fa3 | ||
|
|
84f3c6f02d | ||
|
|
25470baeee | ||
|
|
b86da73a87 | ||
|
|
4aaa7ca895 | ||
|
|
d3f63461e7 | ||
|
|
7a3ace2231 | ||
|
|
c89668a540 | ||
|
|
40ac8e1142 | ||
|
|
26c6420de5 | ||
|
|
bfd3989995 | ||
|
|
7e812ae1a8 | ||
|
|
c23a926b07 | ||
|
|
fe66baedb7 | ||
|
|
959f23d1e4 | ||
|
|
56fbd8cce0 | ||
|
|
1bb8c53ed3 | ||
|
|
5fc0631a15 | ||
|
|
bdbe96beba | ||
|
|
6bd9e49c7a | ||
|
|
f904ca6d7e | ||
|
|
e579c56921 | ||
|
|
83f239065a | ||
|
|
6499318f33 | ||
|
|
809e013c4e | ||
|
|
182829d51c | ||
|
|
61824ade23 | ||
|
|
ff4308e961 | ||
|
|
b5eb13c0a8 | ||
|
|
b702dceda0 | ||
|
|
d616652058 | ||
|
|
c3e571e765 | ||
|
|
30176a3318 | ||
|
|
9c0638d128 | ||
|
|
d7eb6e87a5 | ||
|
|
562e005772 | ||
|
|
f61218de52 | ||
|
|
a19b670e19 | ||
|
|
300d816350 | ||
|
|
63d64a5c87 | ||
|
|
0cfc31c8f7 | ||
|
|
a28891e5f3 | ||
|
|
36036b082d | ||
|
|
1bc63e50a7 | ||
|
|
4b2b5e3029 | ||
|
|
9afa7d2ac6 | ||
|
|
1a8146dbc8 | ||
|
|
178eed7fcd | ||
|
|
621292644c | ||
|
|
1dd01cf54a | ||
|
|
8c31374c53 | ||
|
|
2e177a8839 | ||
|
|
cfffd422f8 | ||
|
|
37aab8382e | ||
|
|
f4950ec2ea | ||
|
|
7bdc32867a | ||
|
|
6e60ac6963 | ||
|
|
1adbd563b2 | ||
|
|
9ccf13b680 | ||
|
|
7b8ca9c018 | ||
|
|
ae6dd23d08 | ||
|
|
b1d164b446 | ||
|
|
87c39dd485 | ||
|
|
1549ac12aa | ||
|
|
5d08fdddb8 | ||
|
|
98114e5bde | ||
|
|
a4817dfdd0 | ||
|
|
d4f05351e1 | ||
|
|
d92362b0d9 | ||
|
|
afc7c75df1 | ||
|
|
f0d4e9bfb4 | ||
|
|
e9e4cf39b6 | ||
|
|
0579a9f760 | ||
|
|
1c0acef3f3 | ||
|
|
2419adb77b | ||
|
|
9e4113555b | ||
|
|
125855540b | ||
|
|
a8eee825e6 | ||
|
|
63859f0ef9 | ||
|
|
1c9000dbda | ||
|
|
8ec857a175 | ||
|
|
f56c7868f5 | ||
|
|
cfd77800d6 | ||
|
|
707deaef0b | ||
|
|
9ddb3ac39d | ||
|
|
a7d3b7c287 | ||
|
|
30bac0397e | ||
|
|
c5fc187a35 | ||
|
|
f46dbcd084 | ||
|
|
2136573cb6 | ||
|
|
86ec08993c | ||
|
|
44da5a71c5 | ||
|
|
e4e81b89ea | ||
|
|
165df659a1 | ||
|
|
bb87afe0f6 | ||
|
|
ac5fb17937 | ||
|
|
e88356b3c5 | ||
|
|
dead9c2d94 | ||
|
|
ca640b2839 | ||
|
|
c972267cd8 | ||
|
|
d0d2c13d16 | ||
|
|
e7b4ab53cf | ||
|
|
f994264d9c | ||
|
|
eb61c1f5d1 | ||
|
|
e578347277 | ||
|
|
294b2dce2e | ||
|
|
aa37105ea3 | ||
|
|
eb57a2697b | ||
|
|
19bc5b63c5 | ||
|
|
2b93ff6cfc | ||
|
|
cc6984e919 | ||
|
|
8bf575e778 | ||
|
|
2e625ee1a2 | ||
|
|
6bcba48ee7 | ||
|
|
c3034edc43 | ||
|
|
82255a09d3 | ||
|
|
594dce13cd | ||
|
|
479e205808 | ||
|
|
3d1b7e8101 | ||
|
|
e951184b7a | ||
|
|
d088b1e7ed | ||
|
|
a38f635514 | ||
|
|
b3e6c215cc | ||
|
|
c9cc3d5c41 | ||
|
|
536e64565c | ||
|
|
65cbf3c1eb | ||
|
|
61c1a7a75a | ||
|
|
1e39db06b8 | ||
|
|
e019f83232 | ||
|
|
3b57a5f8c0 | ||
|
|
eafe3dfc45 | ||
|
|
d56d8d990c | ||
|
|
37a8682901 | ||
|
|
11ba7d4e3e | ||
|
|
6833d37c26 | ||
|
|
e6f72bcb7d | ||
|
|
e8a227c082 | ||
|
|
0472ec2830 | ||
|
|
6412a5cb1a | ||
|
|
cc548689ac | ||
|
|
a3d47e72e3 | ||
|
|
f37482443a | ||
|
|
cc4aef89d3 | ||
|
|
1c0a0eb4b5 | ||
|
|
b1d6c46763 | ||
|
|
42195b9f63 | ||
|
|
a4e08ea13d | ||
|
|
bd2a939a72 | ||
|
|
a89179ab79 | ||
|
|
b0129d383a | ||
|
|
23b6c42630 | ||
|
|
10f4464735 | ||
|
|
4e5addbba5 | ||
|
|
8d11ede927 | ||
|
|
83a3f4da00 | ||
|
|
caf3b69731 | ||
|
|
3135203b55 | ||
|
|
8d113a7c67 | ||
|
|
4e742075f3 | ||
|
|
04caa66662 | ||
|
|
dacc85f5e7 | ||
|
|
f9abec358a | ||
|
|
7e11514cc1 | ||
|
|
2e9c8df8f6 | ||
|
|
4c8cfc8800 | ||
|
|
0ba6fad33f | ||
|
|
3eac215e13 | ||
|
|
90a3262f68 | ||
|
|
f7f49b834e | ||
|
|
89ec5d5bc6 | ||
|
|
e3bc8be936 | ||
|
|
6c5c3f5a4d | ||
|
|
629bd0644b | ||
|
|
b4005079e3 | ||
|
|
a354c443ad | ||
|
|
d1313e3ea1 | ||
|
|
11338008c6 | ||
|
|
8b9e04475d | ||
|
|
d3c9dc6ed7 | ||
|
|
d7ed32571f | ||
|
|
d3935f03c0 | ||
|
|
b2e601f0f0 | ||
|
|
d3ec219a29 | ||
|
|
5ed07d4aaa | ||
|
|
209a3a3626 | ||
|
|
2b3419571f | ||
|
|
bbe504e616 | ||
|
|
6c32591f62 | ||
|
|
ad6da67281 | ||
|
|
14dc593eba | ||
|
|
e52ee41222 | ||
|
|
6ee94f8532 | ||
|
|
21688201af | ||
|
|
f08474369b | ||
|
|
ed617094ea | ||
|
|
9131c50f1b | ||
|
|
69600d08a4 | ||
|
|
5dba77612b | ||
|
|
92b588c866 | ||
|
|
da20e565cd | ||
|
|
ca694c78d2 | ||
|
|
e169056b70 | ||
|
|
b6bf1e026c | ||
|
|
9fa89d48c0 | ||
|
|
5d2c21540c | ||
|
|
1a8aacdff6 | ||
|
|
1804bd9bfc | ||
|
|
7eb4e62762 | ||
|
|
b8e10b5c1f | ||
|
|
a7c11b9b08 | ||
|
|
443c0a74d5 | ||
|
|
84a0f7f7d7 | ||
|
|
558bf8bca8 | ||
|
|
e22d4e6a4b | ||
|
|
a07f946633 | ||
|
|
29c86ac6a3 | ||
|
|
19cf5667d8 | ||
|
|
fb83e58f79 | ||
|
|
9844081d04 | ||
|
|
439ca37e99 | ||
|
|
113a3d9f19 | ||
|
|
978c24458b | ||
|
|
957bece3e9 | ||
|
|
d32c3ac51d | ||
|
|
26102a70a2 | ||
|
|
2b44bf4c23 | ||
|
|
0e63f49e13 | ||
|
|
674a5b8d29 | ||
|
|
7be374100b | ||
|
|
e48c152b95 | ||
|
|
a678f178e1 | ||
|
|
2d8f5641f9 | ||
|
|
0dbd058099 |
2
.github/workflows/build_pull_request.yml
vendored
2
.github/workflows/build_pull_request.yml
vendored
@@ -13,8 +13,6 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
|
||||||
fetch-depth: 0
|
|
||||||
|
|
||||||
- name: Setup Java
|
- name: Setup Java
|
||||||
uses: actions/setup-java@v4
|
uses: actions/setup-java@v4
|
||||||
|
|||||||
1
.github/workflows/pull_strings.yml
vendored
1
.github/workflows/pull_strings.yml
vendored
@@ -17,7 +17,6 @@ jobs:
|
|||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
ref: dev
|
ref: dev
|
||||||
fetch-depth: 0
|
|
||||||
clean: true
|
clean: true
|
||||||
|
|
||||||
- name: Pull strings
|
- name: Pull strings
|
||||||
|
|||||||
2
.github/workflows/push_strings.yml
vendored
2
.github/workflows/push_strings.yml
vendored
@@ -15,8 +15,6 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
|
||||||
fetch-depth: 0
|
|
||||||
|
|
||||||
- name: Preprocess strings
|
- name: Preprocess strings
|
||||||
env:
|
env:
|
||||||
|
|||||||
2
.github/workflows/release.yml
vendored
2
.github/workflows/release.yml
vendored
@@ -19,8 +19,6 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
|
||||||
fetch-depth: 0
|
|
||||||
|
|
||||||
- name: Setup Java
|
- name: Setup Java
|
||||||
uses: actions/setup-java@v4
|
uses: actions/setup-java@v4
|
||||||
|
|||||||
1111
CHANGELOG.md
1111
CHANGELOG.md
File diff suppressed because it is too large
Load Diff
8
adsfund.json
Normal file
8
adsfund.json
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"info": "This is verification file for ads.fund project",
|
||||||
|
"project": {
|
||||||
|
"name": "Revanced Patches",
|
||||||
|
"walletAddress": "0x7ab4091e00363654bf84B34151225742cd92FCE5",
|
||||||
|
"tokenAddress": "0xadf325f255083a3f3d9a9d01ffb3db52a148d802"
|
||||||
|
}
|
||||||
|
}
|
||||||
3
build.gradle.kts
Normal file
3
build.gradle.kts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
plugins {
|
||||||
|
alias(libs.plugins.android.library) apply false
|
||||||
|
}
|
||||||
5
extensions/baconreader/build.gradle.kts
Normal file
5
extensions/baconreader/build.gradle.kts
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
dependencies {
|
||||||
|
compileOnly(project(":extensions:shared:library"))
|
||||||
|
compileOnly(libs.annotation)
|
||||||
|
compileOnly(libs.okhttp)
|
||||||
|
}
|
||||||
1
extensions/baconreader/src/main/AndroidManifest.xml
Normal file
1
extensions/baconreader/src/main/AndroidManifest.xml
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<manifest/>
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
package app.revanced.extension.baconreader;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.fixes.redgifs.BaseFixRedgifsApiPatch;
|
||||||
|
import okhttp3.OkHttpClient;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @noinspection unused
|
||||||
|
*/
|
||||||
|
public class FixRedgifsApiPatch extends BaseFixRedgifsApiPatch {
|
||||||
|
static {
|
||||||
|
INSTANCE = new FixRedgifsApiPatch();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getDefaultUserAgent() {
|
||||||
|
// BaconReader uses a static user agent for Redgifs API calls
|
||||||
|
return "BaconReader";
|
||||||
|
}
|
||||||
|
|
||||||
|
public static OkHttpClient install(OkHttpClient.Builder builder) {
|
||||||
|
return builder.addInterceptor(INSTANCE).build();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
dependencies {
|
dependencies {
|
||||||
compileOnly(project(":extensions:shared:library"))
|
compileOnly(project(":extensions:shared:library"))
|
||||||
compileOnly(project(":extensions:boostforreddit:stub"))
|
compileOnly(project(":extensions:boostforreddit:stub"))
|
||||||
|
compileOnly(libs.annotation)
|
||||||
|
compileOnly(libs.okhttp)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
package app.revanced.extension.boostforreddit;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.fixes.redgifs.BaseFixRedgifsApiPatch;
|
||||||
|
import okhttp3.OkHttpClient;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @noinspection unused
|
||||||
|
*/
|
||||||
|
public class FixRedgifsApiPatch extends BaseFixRedgifsApiPatch {
|
||||||
|
static {
|
||||||
|
INSTANCE = new FixRedgifsApiPatch();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getDefaultUserAgent() {
|
||||||
|
// Boost uses a static user agent for Redgifs API calls
|
||||||
|
return "Boost";
|
||||||
|
}
|
||||||
|
|
||||||
|
public static OkHttpClient createClient() {
|
||||||
|
return new OkHttpClient.Builder().addInterceptor(INSTANCE).build();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
4
extensions/cricbuzz/build.gradle.kts
Normal file
4
extensions/cricbuzz/build.gradle.kts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
dependencies {
|
||||||
|
compileOnly(project(":extensions:shared:library"))
|
||||||
|
compileOnly(project(":extensions:cricbuzz:stub"))
|
||||||
|
}
|
||||||
1
extensions/cricbuzz/src/main/AndroidManifest.xml
Normal file
1
extensions/cricbuzz/src/main/AndroidManifest.xml
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<manifest/>
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
package app.revanced.extension.cricbuzz.ads;
|
||||||
|
|
||||||
|
import com.cricbuzz.android.data.rest.model.BottomBar;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class HideAdsPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*/
|
||||||
|
public static void filterCb11(List<BottomBar> list) {
|
||||||
|
try {
|
||||||
|
Iterator<BottomBar> iterator = list.iterator();
|
||||||
|
while (iterator.hasNext()) {
|
||||||
|
BottomBar bar = iterator.next();
|
||||||
|
if (bar.getName().equals("Cricbuzz11")) {
|
||||||
|
Logger.printInfo(() -> "Removing Cricbuzz11 bar: " + bar);
|
||||||
|
iterator.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception ex) {
|
||||||
|
Logger.printException(() -> "filterCb11 failure", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
17
extensions/cricbuzz/stub/build.gradle.kts
Normal file
17
extensions/cricbuzz/stub/build.gradle.kts
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
plugins {
|
||||||
|
alias(libs.plugins.android.library)
|
||||||
|
}
|
||||||
|
|
||||||
|
android {
|
||||||
|
namespace = "app.revanced.extension"
|
||||||
|
compileSdk = 34
|
||||||
|
|
||||||
|
defaultConfig {
|
||||||
|
minSdk = 21
|
||||||
|
}
|
||||||
|
|
||||||
|
compileOptions {
|
||||||
|
sourceCompatibility = JavaVersion.VERSION_11
|
||||||
|
targetCompatibility = JavaVersion.VERSION_11
|
||||||
|
}
|
||||||
|
}
|
||||||
1
extensions/cricbuzz/stub/src/main/AndroidManifest.xml
Normal file
1
extensions/cricbuzz/stub/src/main/AndroidManifest.xml
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<manifest/>
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
package com.cricbuzz.android.data.rest.model;
|
||||||
|
|
||||||
|
public final class BottomBar {
|
||||||
|
public final String getName() { throw new UnsupportedOperationException(); }
|
||||||
|
}
|
||||||
3
extensions/instagram/build.gradle.kts
Normal file
3
extensions/instagram/build.gradle.kts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
dependencies {
|
||||||
|
compileOnly(project(":extensions:shared:library"))
|
||||||
|
}
|
||||||
1
extensions/instagram/src/main/AndroidManifest.xml
Normal file
1
extensions/instagram/src/main/AndroidManifest.xml
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<manifest/>
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
package app.revanced.extension.instagram.feed;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class LimitFeedToFollowedProfiles {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*/
|
||||||
|
public static Map<String, String> setFollowingHeader(Map<String, String> requestHeaderMap) {
|
||||||
|
// Create new map as original is unmodifiable.
|
||||||
|
Map<String, String> patchedRequestHeaderMap = new HashMap<>(requestHeaderMap);
|
||||||
|
patchedRequestHeaderMap.put("pagination_source", "following");
|
||||||
|
return patchedRequestHeaderMap;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,3 +1,9 @@
|
|||||||
|
dependencies {
|
||||||
|
compileOnly(project(":extensions:shared:library"))
|
||||||
|
compileOnly(project(":extensions:youtube:stub"))
|
||||||
|
compileOnly(libs.annotation)
|
||||||
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
defaultConfig {
|
defaultConfig {
|
||||||
minSdk = 26
|
minSdk = 26
|
||||||
|
|||||||
@@ -0,0 +1,14 @@
|
|||||||
|
package app.revanced.extension.music.patches;
|
||||||
|
|
||||||
|
import app.revanced.extension.music.settings.Settings;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class HideCategoryBarPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point
|
||||||
|
*/
|
||||||
|
public static boolean hideCategoryBar() {
|
||||||
|
return Settings.HIDE_CATEGORY_BAR.get();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
package app.revanced.extension.music.patches;
|
||||||
|
|
||||||
|
import app.revanced.extension.music.settings.Settings;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class HideGetPremiumPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point
|
||||||
|
*/
|
||||||
|
public static boolean hideGetPremiumLabel() {
|
||||||
|
return Settings.HIDE_GET_PREMIUM_LABEL.get();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
package app.revanced.extension.music.patches;
|
||||||
|
|
||||||
|
import app.revanced.extension.music.settings.Settings;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class HideUpgradeButtonPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point
|
||||||
|
*/
|
||||||
|
public static boolean hideUpgradeButton() {
|
||||||
|
return Settings.HIDE_UPGRADE_BUTTON.get();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
package app.revanced.extension.music.patches;
|
||||||
|
|
||||||
|
import app.revanced.extension.music.settings.Settings;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class HideVideoAdsPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point
|
||||||
|
*/
|
||||||
|
public static boolean showVideoAds(boolean original) {
|
||||||
|
if (Settings.HIDE_VIDEO_ADS.get()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return original;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
package app.revanced.extension.music.patches;
|
||||||
|
|
||||||
|
import app.revanced.extension.music.settings.Settings;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class PermanentRepeatPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point
|
||||||
|
*/
|
||||||
|
public static boolean permanentRepeat() {
|
||||||
|
return Settings.PERMANENT_REPEAT.get();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
package app.revanced.extension.music.patches.spoof;
|
||||||
|
|
||||||
|
import static app.revanced.extension.music.settings.Settings.SPOOF_VIDEO_STREAMS_CLIENT_TYPE;
|
||||||
|
import static app.revanced.extension.shared.spoof.ClientType.ANDROID_VR_1_43_32;
|
||||||
|
import static app.revanced.extension.shared.spoof.ClientType.ANDROID_VR_1_61_48;
|
||||||
|
import static app.revanced.extension.shared.spoof.ClientType.VISIONOS;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.spoof.ClientType;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class SpoofVideoStreamsPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*/
|
||||||
|
public static void setClientOrderToUse() {
|
||||||
|
List<ClientType> availableClients = List.of(
|
||||||
|
ANDROID_VR_1_43_32,
|
||||||
|
ANDROID_VR_1_61_48,
|
||||||
|
VISIONOS
|
||||||
|
);
|
||||||
|
|
||||||
|
app.revanced.extension.shared.spoof.SpoofVideoStreamsPatch.setClientsToUse(
|
||||||
|
availableClients, SPOOF_VIDEO_STREAMS_CLIENT_TYPE.get());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,87 @@
|
|||||||
|
package app.revanced.extension.music.settings;
|
||||||
|
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.graphics.PorterDuff;
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
|
import android.preference.PreferenceFragment;
|
||||||
|
import android.view.View;
|
||||||
|
|
||||||
|
import app.revanced.extension.music.settings.preference.ReVancedPreferenceFragment;
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import app.revanced.extension.shared.Utils;
|
||||||
|
import app.revanced.extension.shared.settings.BaseActivityHook;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hooks GoogleApiActivity to inject a custom ReVancedPreferenceFragment with a toolbar.
|
||||||
|
*/
|
||||||
|
public class GoogleApiActivityHook extends BaseActivityHook {
|
||||||
|
/**
|
||||||
|
* Injection point
|
||||||
|
* <p>
|
||||||
|
* Creates an instance of GoogleApiActivityHook for use in static initialization.
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public static GoogleApiActivityHook createInstance() {
|
||||||
|
// Must touch the Music settings to ensure the class is loaded and
|
||||||
|
// the values can be found when setting the UI preferences.
|
||||||
|
// Logging anything under non debug ensures this is set.
|
||||||
|
Logger.printInfo(() -> "Permanent repeat enabled: " + Settings.PERMANENT_REPEAT.get());
|
||||||
|
|
||||||
|
// YT Music always uses dark mode.
|
||||||
|
Utils.setIsDarkModeEnabled(true);
|
||||||
|
|
||||||
|
return new GoogleApiActivityHook();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the fixed theme for the activity.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected void customizeActivityTheme(Activity activity) {
|
||||||
|
// Override the default YouTube Music theme to increase start padding of list items.
|
||||||
|
// Custom style located in resources/music/values/style.xml
|
||||||
|
activity.setTheme(Utils.getResourceIdentifier("Theme.ReVanced.YouTubeMusic.Settings", "style"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the resource ID for the YouTube Music settings layout.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected int getContentViewResourceId() {
|
||||||
|
return Utils.getResourceIdentifier("revanced_music_settings_with_toolbar", "layout");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the fixed background color for the toolbar.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected int getToolbarBackgroundColor() {
|
||||||
|
return Utils.getResourceColor("ytm_color_black");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the navigation icon with a color filter applied.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected Drawable getNavigationIcon() {
|
||||||
|
Drawable navigationIcon = ReVancedPreferenceFragment.getBackButtonDrawable();
|
||||||
|
navigationIcon.setColorFilter(Utils.getAppForegroundColor(), PorterDuff.Mode.SRC_IN);
|
||||||
|
return navigationIcon;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the click listener that finishes the activity when the navigation icon is clicked.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected View.OnClickListener getNavigationClickListener(Activity activity) {
|
||||||
|
return view -> activity.finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new ReVancedPreferenceFragment for the activity.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected PreferenceFragment createPreferenceFragment() {
|
||||||
|
return new ReVancedPreferenceFragment();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
package app.revanced.extension.music.settings;
|
||||||
|
|
||||||
|
import static java.lang.Boolean.FALSE;
|
||||||
|
import static java.lang.Boolean.TRUE;
|
||||||
|
|
||||||
|
import static app.revanced.extension.shared.settings.Setting.parent;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.settings.BaseSettings;
|
||||||
|
import app.revanced.extension.shared.settings.BooleanSetting;
|
||||||
|
import app.revanced.extension.shared.settings.EnumSetting;
|
||||||
|
import app.revanced.extension.shared.spoof.ClientType;
|
||||||
|
|
||||||
|
public class Settings extends BaseSettings {
|
||||||
|
|
||||||
|
// Ads
|
||||||
|
public static final BooleanSetting HIDE_VIDEO_ADS = new BooleanSetting("revanced_music_hide_video_ads", TRUE, true);
|
||||||
|
public static final BooleanSetting HIDE_GET_PREMIUM_LABEL = new BooleanSetting("revanced_music_hide_get_premium_label", TRUE, true);
|
||||||
|
public static final BooleanSetting HIDE_UPGRADE_BUTTON = new BooleanSetting("revanced_music_hide_upgrade_button", TRUE, true);
|
||||||
|
|
||||||
|
// General
|
||||||
|
public static final BooleanSetting HIDE_CATEGORY_BAR = new BooleanSetting("revanced_music_hide_category_bar", FALSE, true);
|
||||||
|
|
||||||
|
// Player
|
||||||
|
public static final BooleanSetting PERMANENT_REPEAT = new BooleanSetting("revanced_music_play_permanent_repeat", FALSE, true);
|
||||||
|
|
||||||
|
// Miscellaneous
|
||||||
|
public static final EnumSetting<ClientType> SPOOF_VIDEO_STREAMS_CLIENT_TYPE = new EnumSetting<>("revanced_spoof_video_streams_client_type",
|
||||||
|
ClientType.ANDROID_VR_1_43_32, true, parent(SPOOF_VIDEO_STREAMS));
|
||||||
|
}
|
||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package app.revanced.extension.music.settings.preference;
|
||||||
|
|
||||||
|
import android.widget.Toolbar;
|
||||||
|
|
||||||
|
import app.revanced.extension.music.settings.GoogleApiActivityHook;
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import app.revanced.extension.shared.Utils;
|
||||||
|
import app.revanced.extension.shared.settings.preference.ToolbarPreferenceFragment;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Preference fragment for ReVanced settings.
|
||||||
|
*/
|
||||||
|
@SuppressWarnings({"deprecation", "NewApi"})
|
||||||
|
public class ReVancedPreferenceFragment extends ToolbarPreferenceFragment {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes the preference fragment.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected void initialize() {
|
||||||
|
super.initialize();
|
||||||
|
|
||||||
|
try {
|
||||||
|
Utils.sortPreferenceGroups(getPreferenceScreen());
|
||||||
|
setPreferenceScreenToolbar(getPreferenceScreen());
|
||||||
|
} catch (Exception ex) {
|
||||||
|
Logger.printException(() -> "initialize failure", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets toolbar for all nested preference screens.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected void customizeToolbar(Toolbar toolbar) {
|
||||||
|
GoogleApiActivityHook.setToolbarLayoutParams(toolbar);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
package app.revanced.extension.music.spoof;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @noinspection unused
|
|
||||||
*/
|
|
||||||
public class SpoofClientPatch {
|
|
||||||
private static final int CLIENT_TYPE_ID = 26;
|
|
||||||
private static final String CLIENT_VERSION = "6.21";
|
|
||||||
private static final String DEVICE_MODEL = "iPhone16,2";
|
|
||||||
private static final String OS_VERSION = "17.7.2.21H221";
|
|
||||||
|
|
||||||
public static int getClientId() {
|
|
||||||
return CLIENT_TYPE_ID;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String getClientVersion() {
|
|
||||||
return CLIENT_VERSION;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String getClientModel() {
|
|
||||||
return DEVICE_MODEL;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String getOsVersion() {
|
|
||||||
return OS_VERSION;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
@@ -0,0 +1,207 @@
|
|||||||
|
package app.revanced.extension.primevideo.videoplayer;
|
||||||
|
|
||||||
|
import android.app.AlertDialog;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.graphics.RectF;
|
||||||
|
import android.view.View;
|
||||||
|
import android.widget.ImageView;
|
||||||
|
import android.widget.LinearLayout;
|
||||||
|
import android.graphics.Color;
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
|
import android.graphics.Canvas;
|
||||||
|
import android.graphics.Paint;
|
||||||
|
import android.graphics.ColorFilter;
|
||||||
|
import android.graphics.PixelFormat;
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import app.revanced.extension.shared.Utils;
|
||||||
|
|
||||||
|
import com.amazon.video.sdk.player.Player;
|
||||||
|
|
||||||
|
public class PlaybackSpeedPatch {
|
||||||
|
private static Player player;
|
||||||
|
private static final float[] SPEED_VALUES = {0.5f, 0.7f, 0.8f, 0.9f, 0.95f, 1.0f, 1.05f, 1.1f, 1.2f, 1.3f, 1.5f, 2.0f};
|
||||||
|
private static final String SPEED_BUTTON_TAG = "speed_overlay";
|
||||||
|
|
||||||
|
public static void setPlayer(Player playerInstance) {
|
||||||
|
player = playerInstance;
|
||||||
|
if (player != null) {
|
||||||
|
// Reset playback rate when switching between episodes to ensure correct display.
|
||||||
|
player.setPlaybackRate(1.0f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void initializeSpeedOverlay(View userControlsView) {
|
||||||
|
try {
|
||||||
|
LinearLayout buttonContainer = Utils.getChildViewByResourceName(userControlsView, "ButtonContainerPlayerTop");
|
||||||
|
|
||||||
|
// If the speed overlay exists we should return early.
|
||||||
|
if (Utils.getChildView(buttonContainer, false, child ->
|
||||||
|
child instanceof ImageView && SPEED_BUTTON_TAG.equals(child.getTag())) != null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImageView speedButton = createSpeedButton(userControlsView.getContext());
|
||||||
|
speedButton.setOnClickListener(v -> changePlaybackSpeed(speedButton));
|
||||||
|
buttonContainer.addView(speedButton, 0);
|
||||||
|
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
Logger.printException(() -> "initializeSpeedOverlay, no button container found", e);
|
||||||
|
} catch (Exception e) {
|
||||||
|
Logger.printException(() -> "initializeSpeedOverlay failure", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static ImageView createSpeedButton(Context context) {
|
||||||
|
ImageView speedButton = new ImageView(context);
|
||||||
|
speedButton.setContentDescription("Playback Speed");
|
||||||
|
speedButton.setTag(SPEED_BUTTON_TAG);
|
||||||
|
speedButton.setClickable(true);
|
||||||
|
speedButton.setFocusable(true);
|
||||||
|
speedButton.setScaleType(ImageView.ScaleType.CENTER);
|
||||||
|
|
||||||
|
SpeedIconDrawable speedIcon = new SpeedIconDrawable();
|
||||||
|
speedButton.setImageDrawable(speedIcon);
|
||||||
|
|
||||||
|
int buttonSize = Utils.dipToPixels(48);
|
||||||
|
speedButton.setMinimumWidth(buttonSize);
|
||||||
|
speedButton.setMinimumHeight(buttonSize);
|
||||||
|
|
||||||
|
return speedButton;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String[] getSpeedOptions() {
|
||||||
|
String[] options = new String[SPEED_VALUES.length];
|
||||||
|
for (int i = 0; i < SPEED_VALUES.length; i++) {
|
||||||
|
options[i] = SPEED_VALUES[i] + "x";
|
||||||
|
}
|
||||||
|
return options;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void changePlaybackSpeed(ImageView imageView) {
|
||||||
|
if (player == null) {
|
||||||
|
Logger.printException(() -> "Player not available");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
player.pause();
|
||||||
|
AlertDialog dialog = createSpeedPlaybackDialog(imageView);
|
||||||
|
dialog.setOnDismissListener(dialogInterface -> player.play());
|
||||||
|
dialog.show();
|
||||||
|
|
||||||
|
} catch (Exception e) {
|
||||||
|
Logger.printException(() -> "changePlaybackSpeed", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static AlertDialog createSpeedPlaybackDialog(ImageView imageView) {
|
||||||
|
Context context = imageView.getContext();
|
||||||
|
int currentSelection = getCurrentSpeedSelection();
|
||||||
|
|
||||||
|
return new AlertDialog.Builder(context)
|
||||||
|
.setTitle("Select Playback Speed")
|
||||||
|
.setSingleChoiceItems(getSpeedOptions(), currentSelection,
|
||||||
|
PlaybackSpeedPatch::handleSpeedSelection)
|
||||||
|
.create();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int getCurrentSpeedSelection() {
|
||||||
|
try {
|
||||||
|
float currentRate = player.getPlaybackRate();
|
||||||
|
int index = Arrays.binarySearch(SPEED_VALUES, currentRate);
|
||||||
|
return Math.max(index, 0); // Use slowest speed if not found.
|
||||||
|
} catch (Exception e) {
|
||||||
|
Logger.printException(() -> "getCurrentSpeedSelection error getting current playback speed", e);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void handleSpeedSelection(android.content.DialogInterface dialog, int selectedIndex) {
|
||||||
|
try {
|
||||||
|
float selectedSpeed = SPEED_VALUES[selectedIndex];
|
||||||
|
player.setPlaybackRate(selectedSpeed);
|
||||||
|
player.play();
|
||||||
|
} catch (Exception e) {
|
||||||
|
Logger.printException(() -> "handleSpeedSelection error setting playback speed", e);
|
||||||
|
} finally {
|
||||||
|
dialog.dismiss();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class SpeedIconDrawable extends Drawable {
|
||||||
|
private final Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void draw(Canvas canvas) {
|
||||||
|
int w = getBounds().width();
|
||||||
|
int h = getBounds().height();
|
||||||
|
float centerX = w / 2f;
|
||||||
|
// Position gauge in lower portion.
|
||||||
|
float centerY = h * 0.7f;
|
||||||
|
float radius = Math.min(w, h) / 2f * 0.8f;
|
||||||
|
|
||||||
|
paint.setColor(Color.WHITE);
|
||||||
|
paint.setStyle(Paint.Style.STROKE);
|
||||||
|
paint.setStrokeWidth(radius * 0.1f);
|
||||||
|
|
||||||
|
// Draw semicircle.
|
||||||
|
RectF oval = new RectF(centerX - radius, centerY - radius, centerX + radius, centerY + radius);
|
||||||
|
canvas.drawArc(oval, 180, 180, false, paint);
|
||||||
|
|
||||||
|
// Draw three tick marks.
|
||||||
|
paint.setStrokeWidth(radius * 0.06f);
|
||||||
|
for (int i = 0; i < 3; i++) {
|
||||||
|
float angle = 180 + (i * 45); // 180°, 225°, 270°.
|
||||||
|
float angleRad = (float) Math.toRadians(angle);
|
||||||
|
|
||||||
|
float startX = centerX + (radius * 0.8f) * (float) Math.cos(angleRad);
|
||||||
|
float startY = centerY + (radius * 0.8f) * (float) Math.sin(angleRad);
|
||||||
|
float endX = centerX + radius * (float) Math.cos(angleRad);
|
||||||
|
float endY = centerY + radius * (float) Math.sin(angleRad);
|
||||||
|
|
||||||
|
canvas.drawLine(startX, startY, endX, endY, paint);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Draw needle.
|
||||||
|
paint.setStrokeWidth(radius * 0.08f);
|
||||||
|
float needleAngle = 200; // Slightly right of center.
|
||||||
|
float needleAngleRad = (float) Math.toRadians(needleAngle);
|
||||||
|
|
||||||
|
float needleEndX = centerX + (radius * 0.6f) * (float) Math.cos(needleAngleRad);
|
||||||
|
float needleEndY = centerY + (radius * 0.6f) * (float) Math.sin(needleAngleRad);
|
||||||
|
|
||||||
|
canvas.drawLine(centerX, centerY, needleEndX, needleEndY, paint);
|
||||||
|
|
||||||
|
// Center dot.
|
||||||
|
paint.setStyle(Paint.Style.FILL);
|
||||||
|
canvas.drawCircle(centerX, centerY, radius * 0.06f, paint);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setAlpha(int alpha) {
|
||||||
|
paint.setAlpha(alpha);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setColorFilter(ColorFilter colorFilter) {
|
||||||
|
paint.setColorFilter(colorFilter);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getOpacity() {
|
||||||
|
return PixelFormat.TRANSLUCENT;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getIntrinsicWidth() {
|
||||||
|
return Utils.dipToPixels(32);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getIntrinsicHeight() {
|
||||||
|
return Utils.dipToPixels(32);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
@@ -4,4 +4,10 @@ public interface VideoPlayer {
|
|||||||
long getCurrentPosition();
|
long getCurrentPosition();
|
||||||
|
|
||||||
void seekTo(long positionMs);
|
void seekTo(long positionMs);
|
||||||
|
|
||||||
|
void pause();
|
||||||
|
|
||||||
|
void play();
|
||||||
|
|
||||||
|
boolean isPlaying();
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
package com.amazon.video.sdk.player;
|
||||||
|
|
||||||
|
public interface Player {
|
||||||
|
float getPlaybackRate();
|
||||||
|
|
||||||
|
void setPlaybackRate(float rate);
|
||||||
|
|
||||||
|
void play();
|
||||||
|
|
||||||
|
void pause();
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
dependencies {
|
dependencies {
|
||||||
implementation(project(":extensions:shared:library"))
|
implementation(project(":extensions:shared:library"))
|
||||||
|
compileOnly(libs.okhttp)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id("com.android.library")
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
@@ -18,4 +18,5 @@ android {
|
|||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
compileOnly(libs.annotation)
|
compileOnly(libs.annotation)
|
||||||
|
compileOnly(libs.okhttp)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
package app.revanced.extension.youtube;
|
package app.revanced.extension.shared;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
|
||||||
|
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
|
|
||||||
@@ -39,7 +37,7 @@ public final class ByteTrieSearch extends TrieSearch<byte[]> {
|
|||||||
return replacement;
|
return replacement;
|
||||||
}
|
}
|
||||||
|
|
||||||
public ByteTrieSearch(@NonNull byte[]... patterns) {
|
public ByteTrieSearch(byte[]... patterns) {
|
||||||
super(new ByteTrieNode(), patterns);
|
super(new ByteTrieNode(), patterns);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -19,7 +19,6 @@ import android.util.Pair;
|
|||||||
import android.widget.LinearLayout;
|
import android.widget.LinearLayout;
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
import androidx.annotation.Nullable;
|
||||||
import androidx.annotation.RequiresApi;
|
|
||||||
|
|
||||||
import java.net.HttpURLConnection;
|
import java.net.HttpURLConnection;
|
||||||
import java.net.MalformedURLException;
|
import java.net.MalformedURLException;
|
||||||
@@ -28,7 +27,6 @@ import java.util.Locale;
|
|||||||
|
|
||||||
import app.revanced.extension.shared.requests.Requester;
|
import app.revanced.extension.shared.requests.Requester;
|
||||||
import app.revanced.extension.shared.requests.Route;
|
import app.revanced.extension.shared.requests.Route;
|
||||||
import app.revanced.extension.shared.Utils;
|
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
public class GmsCoreSupport {
|
public class GmsCoreSupport {
|
||||||
@@ -109,7 +107,6 @@ public class GmsCoreSupport {
|
|||||||
/**
|
/**
|
||||||
* Injection point.
|
* Injection point.
|
||||||
*/
|
*/
|
||||||
@RequiresApi(api = Build.VERSION_CODES.N)
|
|
||||||
public static void checkGmsCore(Activity context) {
|
public static void checkGmsCore(Activity context) {
|
||||||
try {
|
try {
|
||||||
// Verify the user has not included GmsCore for a root installation.
|
// Verify the user has not included GmsCore for a root installation.
|
||||||
@@ -157,7 +154,9 @@ public class GmsCoreSupport {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Check if GmsCore is currently running in the background.
|
// Check if GmsCore is currently running in the background.
|
||||||
try (var client = context.getContentResolver().acquireContentProviderClient(GMS_CORE_PROVIDER)) {
|
var client = context.getContentResolver().acquireContentProviderClient(GMS_CORE_PROVIDER);
|
||||||
|
//noinspection TryFinallyCanBeTryWithResources
|
||||||
|
try {
|
||||||
if (client == null) {
|
if (client == null) {
|
||||||
Logger.printInfo(() -> "GmsCore is not running in the background");
|
Logger.printInfo(() -> "GmsCore is not running in the background");
|
||||||
checkIfDontKillMyAppSupportsManufacturer();
|
checkIfDontKillMyAppSupportsManufacturer();
|
||||||
@@ -167,6 +166,8 @@ public class GmsCoreSupport {
|
|||||||
"gms_core_dialog_open_website_text",
|
"gms_core_dialog_open_website_text",
|
||||||
(dialog, id) -> openDontKillMyApp());
|
(dialog, id) -> openDontKillMyApp());
|
||||||
}
|
}
|
||||||
|
} finally {
|
||||||
|
if (client != null) client.close();
|
||||||
}
|
}
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.printException(() -> "checkGmsCore failure", ex);
|
Logger.printException(() -> "checkGmsCore failure", ex);
|
||||||
@@ -226,6 +227,11 @@ public class GmsCoreSupport {
|
|||||||
* @return If GmsCore is not whitelisted from battery optimizations.
|
* @return If GmsCore is not whitelisted from battery optimizations.
|
||||||
*/
|
*/
|
||||||
private static boolean batteryOptimizationsEnabled(Context context) {
|
private static boolean batteryOptimizationsEnabled(Context context) {
|
||||||
|
//noinspection ObsoleteSdkInt
|
||||||
|
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
|
||||||
|
// Android 5.0 does not have battery optimization settings.
|
||||||
|
return false;
|
||||||
|
}
|
||||||
var powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
var powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||||
return !powerManager.isIgnoringBatteryOptimizations(GMS_CORE_PACKAGE_NAME);
|
return !powerManager.isIgnoringBatteryOptimizations(GMS_CORE_PACKAGE_NAME);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
package app.revanced.extension.youtube;
|
package app.revanced.extension.shared;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Text pattern searching using a prefix tree (trie).
|
* Text pattern searching using a prefix tree (trie).
|
||||||
@@ -28,7 +26,7 @@ public final class StringTrieSearch extends TrieSearch<String> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public StringTrieSearch(@NonNull String... patterns) {
|
public StringTrieSearch(String... patterns) {
|
||||||
super(new StringTrieNode(), patterns);
|
super(new StringTrieNode(), patterns);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
package app.revanced.extension.youtube;
|
package app.revanced.extension.shared;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
|
||||||
import androidx.annotation.Nullable;
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
@@ -57,11 +56,13 @@ public abstract class TrieSearch<T> {
|
|||||||
if (searchTextLength - searchTextIndex < patternLength - patternStartIndex) {
|
if (searchTextLength - searchTextIndex < patternLength - patternStartIndex) {
|
||||||
return false; // Remaining search text is shorter than the remaining leaf pattern and they cannot match.
|
return false; // Remaining search text is shorter than the remaining leaf pattern and they cannot match.
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i = searchTextIndex, j = patternStartIndex; j < patternLength; i++, j++) {
|
for (int i = searchTextIndex, j = patternStartIndex; j < patternLength; i++, j++) {
|
||||||
if (enclosingNode.getCharValue(searchText, i) != enclosingNode.getCharValue(pattern, j)) {
|
if (enclosingNode.getCharValue(searchText, i) != enclosingNode.getCharValue(pattern, j)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return callback == null || callback.patternMatched(searchText,
|
return callback == null || callback.patternMatched(searchText,
|
||||||
searchTextIndex - patternStartIndex, patternLength, callbackParameter);
|
searchTextIndex - patternStartIndex, patternLength, callbackParameter);
|
||||||
}
|
}
|
||||||
@@ -136,7 +137,7 @@ public abstract class TrieSearch<T> {
|
|||||||
* @param patternLength Length of the pattern.
|
* @param patternLength Length of the pattern.
|
||||||
* @param callback Callback, where a value of NULL indicates to always accept a pattern match.
|
* @param callback Callback, where a value of NULL indicates to always accept a pattern match.
|
||||||
*/
|
*/
|
||||||
private void addPattern(@NonNull T pattern, int patternIndex, int patternLength,
|
private void addPattern(T pattern, int patternIndex, int patternLength,
|
||||||
@Nullable TriePatternMatchedCallback<T> callback) {
|
@Nullable TriePatternMatchedCallback<T> callback) {
|
||||||
if (patternIndex == patternLength) { // Reached the end of the pattern.
|
if (patternIndex == patternLength) { // Reached the end of the pattern.
|
||||||
if (endOfPatternCallback == null) {
|
if (endOfPatternCallback == null) {
|
||||||
@@ -145,6 +146,7 @@ public abstract class TrieSearch<T> {
|
|||||||
endOfPatternCallback.add(callback);
|
endOfPatternCallback.add(callback);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (leaf != null) {
|
if (leaf != null) {
|
||||||
// Reached end of the graph and a leaf exist.
|
// Reached end of the graph and a leaf exist.
|
||||||
// Recursively call back into this method and push the existing leaf down 1 level.
|
// Recursively call back into this method and push the existing leaf down 1 level.
|
||||||
@@ -159,6 +161,7 @@ public abstract class TrieSearch<T> {
|
|||||||
leaf = new TrieCompressedPath<>(pattern, patternIndex, patternLength, callback);
|
leaf = new TrieCompressedPath<>(pattern, patternIndex, patternLength, callback);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final char character = getCharValue(pattern, patternIndex);
|
final char character = getCharValue(pattern, patternIndex);
|
||||||
final int arrayIndex = hashIndexForTableSize(children.length, character);
|
final int arrayIndex = hashIndexForTableSize(children.length, character);
|
||||||
TrieNode<T> child = children[arrayIndex];
|
TrieNode<T> child = children[arrayIndex];
|
||||||
@@ -183,6 +186,7 @@ public abstract class TrieSearch<T> {
|
|||||||
//noinspection unchecked
|
//noinspection unchecked
|
||||||
TrieNode<T>[] replacement = new TrieNode[replacementArraySize];
|
TrieNode<T>[] replacement = new TrieNode[replacementArraySize];
|
||||||
addNodeToArray(replacement, child);
|
addNodeToArray(replacement, child);
|
||||||
|
|
||||||
boolean collision = false;
|
boolean collision = false;
|
||||||
for (TrieNode<T> existingChild : children) {
|
for (TrieNode<T> existingChild : children) {
|
||||||
if (existingChild != null) {
|
if (existingChild != null) {
|
||||||
@@ -195,6 +199,7 @@ public abstract class TrieSearch<T> {
|
|||||||
if (collision) {
|
if (collision) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
children = replacement;
|
children = replacement;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -234,6 +239,7 @@ public abstract class TrieSearch<T> {
|
|||||||
if (leaf != null && leaf.matches(startNode, searchText, searchTextEndIndex, searchTextIndex, callbackParameter)) {
|
if (leaf != null && leaf.matches(startNode, searchText, searchTextEndIndex, searchTextIndex, callbackParameter)) {
|
||||||
return true; // Leaf exists and it matched the search text.
|
return true; // Leaf exists and it matched the search text.
|
||||||
}
|
}
|
||||||
|
|
||||||
List<TriePatternMatchedCallback<T>> endOfPatternCallback = node.endOfPatternCallback;
|
List<TriePatternMatchedCallback<T>> endOfPatternCallback = node.endOfPatternCallback;
|
||||||
if (endOfPatternCallback != null) {
|
if (endOfPatternCallback != null) {
|
||||||
final int matchStartIndex = searchTextIndex - currentMatchLength;
|
final int matchStartIndex = searchTextIndex - currentMatchLength;
|
||||||
@@ -246,6 +252,7 @@ public abstract class TrieSearch<T> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TrieNode<T>[] children = node.children;
|
TrieNode<T>[] children = node.children;
|
||||||
if (children == null) {
|
if (children == null) {
|
||||||
return false; // Reached a graph end point and there's no further patterns to search.
|
return false; // Reached a graph end point and there's no further patterns to search.
|
||||||
@@ -278,9 +285,11 @@ public abstract class TrieSearch<T> {
|
|||||||
if (leaf != null) {
|
if (leaf != null) {
|
||||||
numberOfPointers += 4; // Number of fields in leaf node.
|
numberOfPointers += 4; // Number of fields in leaf node.
|
||||||
}
|
}
|
||||||
|
|
||||||
if (endOfPatternCallback != null) {
|
if (endOfPatternCallback != null) {
|
||||||
numberOfPointers += endOfPatternCallback.size();
|
numberOfPointers += endOfPatternCallback.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (children != null) {
|
if (children != null) {
|
||||||
numberOfPointers += children.length;
|
numberOfPointers += children.length;
|
||||||
for (TrieNode<T> child : children) {
|
for (TrieNode<T> child : children) {
|
||||||
@@ -308,13 +317,13 @@ public abstract class TrieSearch<T> {
|
|||||||
private final List<T> patterns = new ArrayList<>();
|
private final List<T> patterns = new ArrayList<>();
|
||||||
|
|
||||||
@SafeVarargs
|
@SafeVarargs
|
||||||
TrieSearch(@NonNull TrieNode<T> root, @NonNull T... patterns) {
|
TrieSearch(TrieNode<T> root, T... patterns) {
|
||||||
this.root = Objects.requireNonNull(root);
|
this.root = Objects.requireNonNull(root);
|
||||||
addPatterns(patterns);
|
addPatterns(patterns);
|
||||||
}
|
}
|
||||||
|
|
||||||
@SafeVarargs
|
@SafeVarargs
|
||||||
public final void addPatterns(@NonNull T... patterns) {
|
public final void addPatterns(T... patterns) {
|
||||||
for (T pattern : patterns) {
|
for (T pattern : patterns) {
|
||||||
addPattern(pattern);
|
addPattern(pattern);
|
||||||
}
|
}
|
||||||
@@ -325,7 +334,7 @@ public abstract class TrieSearch<T> {
|
|||||||
*
|
*
|
||||||
* @param pattern Pattern to add. Calling this with a zero length pattern does nothing.
|
* @param pattern Pattern to add. Calling this with a zero length pattern does nothing.
|
||||||
*/
|
*/
|
||||||
public void addPattern(@NonNull T pattern) {
|
public void addPattern(T pattern) {
|
||||||
addPattern(pattern, root.getTextLength(pattern), null);
|
addPattern(pattern, root.getTextLength(pattern), null);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -333,31 +342,31 @@ public abstract class TrieSearch<T> {
|
|||||||
* @param pattern Pattern to add. Calling this with a zero length pattern does nothing.
|
* @param pattern Pattern to add. Calling this with a zero length pattern does nothing.
|
||||||
* @param callback Callback to determine if searching should halt when a match is found.
|
* @param callback Callback to determine if searching should halt when a match is found.
|
||||||
*/
|
*/
|
||||||
public void addPattern(@NonNull T pattern, @NonNull TriePatternMatchedCallback<T> callback) {
|
public void addPattern(T pattern, TriePatternMatchedCallback<T> callback) {
|
||||||
addPattern(pattern, root.getTextLength(pattern), Objects.requireNonNull(callback));
|
addPattern(pattern, root.getTextLength(pattern), Objects.requireNonNull(callback));
|
||||||
}
|
}
|
||||||
|
|
||||||
void addPattern(@NonNull T pattern, int patternLength, @Nullable TriePatternMatchedCallback<T> callback) {
|
void addPattern(T pattern, int patternLength, @Nullable TriePatternMatchedCallback<T> callback) {
|
||||||
if (patternLength == 0) return; // Nothing to match
|
if (patternLength == 0) return; // Nothing to match
|
||||||
|
|
||||||
patterns.add(pattern);
|
patterns.add(pattern);
|
||||||
root.addPattern(pattern, 0, patternLength, callback);
|
root.addPattern(pattern, 0, patternLength, callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
public final boolean matches(@NonNull T textToSearch) {
|
public final boolean matches(T textToSearch) {
|
||||||
return matches(textToSearch, 0);
|
return matches(textToSearch, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean matches(@NonNull T textToSearch, @NonNull Object callbackParameter) {
|
public boolean matches(T textToSearch, Object callbackParameter) {
|
||||||
return matches(textToSearch, 0, root.getTextLength(textToSearch),
|
return matches(textToSearch, 0, root.getTextLength(textToSearch),
|
||||||
Objects.requireNonNull(callbackParameter));
|
Objects.requireNonNull(callbackParameter));
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean matches(@NonNull T textToSearch, int startIndex) {
|
public boolean matches(T textToSearch, int startIndex) {
|
||||||
return matches(textToSearch, startIndex, root.getTextLength(textToSearch));
|
return matches(textToSearch, startIndex, root.getTextLength(textToSearch));
|
||||||
}
|
}
|
||||||
|
|
||||||
public final boolean matches(@NonNull T textToSearch, int startIndex, int endIndex) {
|
public final boolean matches(T textToSearch, int startIndex, int endIndex) {
|
||||||
return matches(textToSearch, startIndex, endIndex, null);
|
return matches(textToSearch, startIndex, endIndex, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -370,11 +379,11 @@ public abstract class TrieSearch<T> {
|
|||||||
* @param callbackParameter Optional parameter passed to the callbacks.
|
* @param callbackParameter Optional parameter passed to the callbacks.
|
||||||
* @return If any pattern matched, and it's callback halted searching.
|
* @return If any pattern matched, and it's callback halted searching.
|
||||||
*/
|
*/
|
||||||
public boolean matches(@NonNull T textToSearch, int startIndex, int endIndex, @Nullable Object callbackParameter) {
|
public boolean matches(T textToSearch, int startIndex, int endIndex, @Nullable Object callbackParameter) {
|
||||||
return matches(textToSearch, root.getTextLength(textToSearch), startIndex, endIndex, callbackParameter);
|
return matches(textToSearch, root.getTextLength(textToSearch), startIndex, endIndex, callbackParameter);
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean matches(@NonNull T textToSearch, int textToSearchLength, int startIndex, int endIndex,
|
private boolean matches(T textToSearch, int textToSearchLength, int startIndex, int endIndex,
|
||||||
@Nullable Object callbackParameter) {
|
@Nullable Object callbackParameter) {
|
||||||
if (endIndex > textToSearchLength) {
|
if (endIndex > textToSearchLength) {
|
||||||
throw new IllegalArgumentException("endIndex: " + endIndex
|
throw new IllegalArgumentException("endIndex: " + endIndex
|
||||||
@@ -311,6 +311,10 @@ public class Utils {
|
|||||||
return getContext().getResources().getDimension(getResourceIdentifier(resourceIdentifierName, "dimen"));
|
return getContext().getResources().getDimension(getResourceIdentifier(resourceIdentifierName, "dimen"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static String[] getResourceStringArray(String resourceIdentifierName) throws Resources.NotFoundException {
|
||||||
|
return getContext().getResources().getStringArray(getResourceIdentifier(resourceIdentifierName, "array"));
|
||||||
|
}
|
||||||
|
|
||||||
public interface MatchFilter<T> {
|
public interface MatchFilter<T> {
|
||||||
boolean matches(T object);
|
boolean matches(T object);
|
||||||
}
|
}
|
||||||
@@ -325,7 +329,7 @@ public class Utils {
|
|||||||
return (R) child;
|
return (R) child;
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new IllegalArgumentException("View with resource name '" + str + "' not found");
|
throw new IllegalArgumentException("View with resource name not found: " + str);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -579,7 +583,7 @@ public class Utils {
|
|||||||
Context currentContext = context;
|
Context currentContext = context;
|
||||||
|
|
||||||
if (currentContext == null) {
|
if (currentContext == null) {
|
||||||
Logger.printException(() -> "Cannot show toast (context is null): " + messageToToast, null);
|
Logger.printException(() -> "Cannot show toast (context is null): " + messageToToast);
|
||||||
} else {
|
} else {
|
||||||
Logger.printDebug(() -> "Showing toast: " + messageToToast);
|
Logger.printDebug(() -> "Showing toast: " + messageToToast);
|
||||||
Toast.makeText(currentContext, messageToToast, toastDuration).show();
|
Toast.makeText(currentContext, messageToToast, toastDuration).show();
|
||||||
@@ -809,7 +813,7 @@ public class Utils {
|
|||||||
|
|
||||||
// Create content container (message/EditText) inside a ScrollView only if message or editText is provided.
|
// Create content container (message/EditText) inside a ScrollView only if message or editText is provided.
|
||||||
ScrollView contentScrollView = null;
|
ScrollView contentScrollView = null;
|
||||||
LinearLayout contentContainer = null;
|
LinearLayout contentContainer;
|
||||||
if (message != null || editText != null) {
|
if (message != null || editText != null) {
|
||||||
contentScrollView = new ScrollView(context);
|
contentScrollView = new ScrollView(context);
|
||||||
contentScrollView.setVerticalScrollBarEnabled(false); // Disable the vertical scrollbar.
|
contentScrollView.setVerticalScrollBarEnabled(false); // Disable the vertical scrollbar.
|
||||||
@@ -833,7 +837,7 @@ public class Utils {
|
|||||||
contentScrollView.addView(contentContainer);
|
contentScrollView.addView(contentContainer);
|
||||||
|
|
||||||
// Message (if not replaced by EditText).
|
// Message (if not replaced by EditText).
|
||||||
if (editText == null && message != null) {
|
if (editText == null) {
|
||||||
TextView messageView = new TextView(context);
|
TextView messageView = new TextView(context);
|
||||||
messageView.setText(message); // Supports Spanned (HTML).
|
messageView.setText(message); // Supports Spanned (HTML).
|
||||||
messageView.setTextSize(16);
|
messageView.setTextSize(16);
|
||||||
@@ -1434,6 +1438,38 @@ public class Utils {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts a percentage of the screen height to actual device pixels.
|
||||||
|
*
|
||||||
|
* @param percentage The percentage of the screen height (e.g., 30 for 30%).
|
||||||
|
* @return The device pixel value corresponding to the percentage of screen height.
|
||||||
|
*/
|
||||||
|
public static int percentageHeightToPixels(int percentage) {
|
||||||
|
DisplayMetrics metrics = context.getResources().getDisplayMetrics();
|
||||||
|
return (int) (metrics.heightPixels * (percentage / 100.0f));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts a percentage of the screen width to actual device pixels.
|
||||||
|
*
|
||||||
|
* @param percentage The percentage of the screen width (e.g., 30 for 30%).
|
||||||
|
* @return The device pixel value corresponding to the percentage of screen width.
|
||||||
|
*/
|
||||||
|
public static int percentageWidthToPixels(int percentage) {
|
||||||
|
DisplayMetrics metrics = context.getResources().getDisplayMetrics();
|
||||||
|
return (int) (metrics.widthPixels * (percentage / 100.0f));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Uses {@link #adjustColorBrightness(int, float)} depending if light or dark mode is active.
|
||||||
|
*/
|
||||||
|
@ColorInt
|
||||||
|
public static int adjustColorBrightness(@ColorInt int baseColor, float lightThemeFactor, float darkThemeFactor) {
|
||||||
|
return isDarkModeEnabled()
|
||||||
|
? adjustColorBrightness(baseColor, darkThemeFactor)
|
||||||
|
: adjustColorBrightness(baseColor, lightThemeFactor);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adjusts the brightness of a color by lightening or darkening it based on the given factor.
|
* Adjusts the brightness of a color by lightening or darkening it based on the given factor.
|
||||||
* <p>
|
* <p>
|
||||||
|
|||||||
@@ -0,0 +1,71 @@
|
|||||||
|
package app.revanced.extension.shared.fixes.redgifs;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
|
import org.json.JSONException;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.HttpURLConnection;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import okhttp3.Interceptor;
|
||||||
|
import okhttp3.MediaType;
|
||||||
|
import okhttp3.Protocol;
|
||||||
|
import okhttp3.Request;
|
||||||
|
import okhttp3.Response;
|
||||||
|
import okhttp3.ResponseBody;
|
||||||
|
|
||||||
|
|
||||||
|
public abstract class BaseFixRedgifsApiPatch implements Interceptor {
|
||||||
|
protected static BaseFixRedgifsApiPatch INSTANCE;
|
||||||
|
public abstract String getDefaultUserAgent();
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public Response intercept(@NonNull Chain chain) throws IOException {
|
||||||
|
Request request = chain.request();
|
||||||
|
if (!request.url().host().equals("api.redgifs.com")) {
|
||||||
|
return chain.proceed(request);
|
||||||
|
}
|
||||||
|
|
||||||
|
String userAgent = getDefaultUserAgent();
|
||||||
|
|
||||||
|
if (request.header("Authorization") != null) {
|
||||||
|
Response response = chain.proceed(request.newBuilder().header("User-Agent", userAgent).build());
|
||||||
|
if (response.isSuccessful()) {
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
// It's possible that the user agent is being overwritten later down in the interceptor
|
||||||
|
// chain, so make sure we grab the new user agent from the request headers.
|
||||||
|
userAgent = response.request().header("User-Agent");
|
||||||
|
response.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
RedgifsTokenManager.RedgifsToken token = RedgifsTokenManager.refreshToken(userAgent);
|
||||||
|
|
||||||
|
// Emulate response for old OAuth endpoint
|
||||||
|
if (request.url().encodedPath().equals("/v2/oauth/client")) {
|
||||||
|
String responseBody = RedgifsTokenManager.getEmulatedOAuthResponseBody(token);
|
||||||
|
return new Response.Builder()
|
||||||
|
.message("OK")
|
||||||
|
.code(HttpURLConnection.HTTP_OK)
|
||||||
|
.protocol(Protocol.HTTP_1_1)
|
||||||
|
.request(request)
|
||||||
|
.header("Content-Type", "application/json")
|
||||||
|
.body(ResponseBody.create(
|
||||||
|
responseBody, MediaType.get("application/json")))
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
Request modifiedRequest = request.newBuilder()
|
||||||
|
.header("Authorization", "Bearer " + token.getAccessToken())
|
||||||
|
.header("User-Agent", userAgent)
|
||||||
|
.build();
|
||||||
|
return chain.proceed(modifiedRequest);
|
||||||
|
} catch (JSONException ex) {
|
||||||
|
Logger.printException(() -> "Could not parse Redgifs response", ex);
|
||||||
|
throw new IOException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,94 @@
|
|||||||
|
package app.revanced.extension.shared.fixes.redgifs;
|
||||||
|
|
||||||
|
import static app.revanced.extension.shared.requests.Route.Method.GET;
|
||||||
|
|
||||||
|
import androidx.annotation.GuardedBy;
|
||||||
|
|
||||||
|
import org.json.JSONException;
|
||||||
|
import org.json.JSONObject;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.HttpURLConnection;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.requests.Requester;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Manages Redgifs token lifecycle.
|
||||||
|
*/
|
||||||
|
public class RedgifsTokenManager {
|
||||||
|
public static class RedgifsToken {
|
||||||
|
// Expire after 23 hours to provide some breathing room
|
||||||
|
private static final long EXPIRY_SECONDS = 23 * 60 * 60;
|
||||||
|
|
||||||
|
private final String accessToken;
|
||||||
|
private final long refreshTimeInSeconds;
|
||||||
|
|
||||||
|
public RedgifsToken(String accessToken, long refreshTime) {
|
||||||
|
this.accessToken = accessToken;
|
||||||
|
this.refreshTimeInSeconds = refreshTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getAccessToken() {
|
||||||
|
return accessToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getExpiryTimeInSeconds() {
|
||||||
|
return refreshTimeInSeconds + EXPIRY_SECONDS;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isValid() {
|
||||||
|
if (accessToken == null) return false;
|
||||||
|
return getExpiryTimeInSeconds() >= System.currentTimeMillis() / 1000;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public static final String REDGIFS_API_HOST = "https://api.redgifs.com";
|
||||||
|
private static final String GET_TEMPORARY_TOKEN = REDGIFS_API_HOST + "/v2/auth/temporary";
|
||||||
|
@GuardedBy("itself")
|
||||||
|
private static final Map<String, RedgifsToken> tokenMap = new HashMap<>();
|
||||||
|
|
||||||
|
private static String getToken(String userAgent) throws IOException, JSONException {
|
||||||
|
HttpURLConnection connection = (HttpURLConnection) new URL(GET_TEMPORARY_TOKEN).openConnection();
|
||||||
|
connection.setFixedLengthStreamingMode(0);
|
||||||
|
connection.setRequestMethod(GET.name());
|
||||||
|
connection.setRequestProperty("User-Agent", userAgent);
|
||||||
|
connection.setRequestProperty("Content-Type", "application/json");
|
||||||
|
connection.setRequestProperty("Accept", "application/json");
|
||||||
|
connection.setUseCaches(false);
|
||||||
|
|
||||||
|
JSONObject responseObject = Requester.parseJSONObject(connection);
|
||||||
|
return responseObject.getString("token");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static RedgifsToken refreshToken(String userAgent) throws IOException, JSONException {
|
||||||
|
synchronized(tokenMap) {
|
||||||
|
// Reference: https://github.com/JeffreyCA/Apollo-ImprovedCustomApi/pull/67
|
||||||
|
RedgifsToken token = tokenMap.get(userAgent);
|
||||||
|
if (token != null && token.isValid()) {
|
||||||
|
return token;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy user agent from original request if present because Redgifs verifies
|
||||||
|
// that the user agent in subsequent requests matches the one in the OAuth token.
|
||||||
|
String accessToken = getToken(userAgent);
|
||||||
|
long refreshTime = System.currentTimeMillis() / 1000;
|
||||||
|
token = new RedgifsToken(accessToken, refreshTime);
|
||||||
|
tokenMap.put(userAgent, token);
|
||||||
|
return token;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String getEmulatedOAuthResponseBody(RedgifsToken token) throws JSONException {
|
||||||
|
// Reference: https://github.com/JeffreyCA/Apollo-ImprovedCustomApi/pull/67
|
||||||
|
JSONObject responseObject = new JSONObject();
|
||||||
|
responseObject.put("access_token", token.accessToken);
|
||||||
|
responseObject.put("expiry_time", token.getExpiryTimeInSeconds() - (System.currentTimeMillis() / 1000));
|
||||||
|
responseObject.put("scope", "read");
|
||||||
|
responseObject.put("token_type", "Bearer");
|
||||||
|
return responseObject.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,142 @@
|
|||||||
|
package app.revanced.extension.shared.settings;
|
||||||
|
|
||||||
|
import android.annotation.SuppressLint;
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
|
import android.preference.PreferenceFragment;
|
||||||
|
import android.util.TypedValue;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.widget.TextView;
|
||||||
|
import android.widget.Toolbar;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import app.revanced.extension.shared.Utils;
|
||||||
|
import app.revanced.extension.shared.settings.preference.ToolbarPreferenceFragment;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Base class for hooking activities to inject a custom PreferenceFragment with a toolbar.
|
||||||
|
* Provides common logic for initializing the activity and setting up the toolbar.
|
||||||
|
*/
|
||||||
|
@SuppressWarnings({"deprecation", "NewApi"})
|
||||||
|
public abstract class BaseActivityHook extends Activity {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Layout parameters for the toolbar, extracted from the dummy toolbar.
|
||||||
|
*/
|
||||||
|
protected static ViewGroup.LayoutParams toolbarLayoutParams;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the layout parameters for the toolbar.
|
||||||
|
*/
|
||||||
|
public static void setToolbarLayoutParams(Toolbar toolbar) {
|
||||||
|
if (toolbarLayoutParams != null) {
|
||||||
|
toolbar.setLayoutParams(toolbarLayoutParams);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initializes the activity by setting the theme, content view and injecting a PreferenceFragment.
|
||||||
|
*/
|
||||||
|
public static void initialize(BaseActivityHook hook, Activity activity) {
|
||||||
|
try {
|
||||||
|
hook.customizeActivityTheme(activity);
|
||||||
|
activity.setContentView(hook.getContentViewResourceId());
|
||||||
|
|
||||||
|
// Sanity check.
|
||||||
|
String dataString = activity.getIntent().getDataString();
|
||||||
|
if (!"revanced_settings_intent".equals(dataString)) {
|
||||||
|
Logger.printException(() -> "Unknown intent: " + dataString);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
PreferenceFragment fragment = hook.createPreferenceFragment();
|
||||||
|
hook.createToolbar(activity, fragment);
|
||||||
|
|
||||||
|
activity.getFragmentManager()
|
||||||
|
.beginTransaction()
|
||||||
|
.replace(Utils.getResourceIdentifier("revanced_settings_fragments", "id"), fragment)
|
||||||
|
.commit();
|
||||||
|
} catch (Exception ex) {
|
||||||
|
Logger.printException(() -> "initialize failure", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates and configures a toolbar for the activity, replacing a dummy placeholder.
|
||||||
|
*/
|
||||||
|
@SuppressLint("UseCompatLoadingForDrawables")
|
||||||
|
protected void createToolbar(Activity activity, PreferenceFragment fragment) {
|
||||||
|
// Replace dummy placeholder toolbar.
|
||||||
|
// This is required to fix submenu title alignment issue with Android ASOP 15+
|
||||||
|
ViewGroup toolBarParent = activity.findViewById(
|
||||||
|
Utils.getResourceIdentifier("revanced_toolbar_parent", "id"));
|
||||||
|
ViewGroup dummyToolbar = Utils.getChildViewByResourceName(toolBarParent, "revanced_toolbar");
|
||||||
|
toolbarLayoutParams = dummyToolbar.getLayoutParams();
|
||||||
|
toolBarParent.removeView(dummyToolbar);
|
||||||
|
|
||||||
|
// Sets appropriate system navigation bar color for the activity.
|
||||||
|
ToolbarPreferenceFragment.setNavigationBarColor(activity.getWindow());
|
||||||
|
|
||||||
|
Toolbar toolbar = new Toolbar(toolBarParent.getContext());
|
||||||
|
toolbar.setBackgroundColor(getToolbarBackgroundColor());
|
||||||
|
toolbar.setNavigationIcon(getNavigationIcon());
|
||||||
|
toolbar.setNavigationOnClickListener(getNavigationClickListener(activity));
|
||||||
|
toolbar.setTitle(Utils.getResourceIdentifier("revanced_settings_title", "string"));
|
||||||
|
|
||||||
|
final int margin = Utils.dipToPixels(16);
|
||||||
|
toolbar.setTitleMarginStart(margin);
|
||||||
|
toolbar.setTitleMarginEnd(margin);
|
||||||
|
TextView toolbarTextView = Utils.getChildView(toolbar, false, view -> view instanceof TextView);
|
||||||
|
if (toolbarTextView != null) {
|
||||||
|
toolbarTextView.setTextColor(Utils.getAppForegroundColor());
|
||||||
|
toolbarTextView.setTextSize(TypedValue.COMPLEX_UNIT_SP, 20);
|
||||||
|
}
|
||||||
|
setToolbarLayoutParams(toolbar);
|
||||||
|
|
||||||
|
onPostToolbarSetup(activity, toolbar, fragment);
|
||||||
|
|
||||||
|
toolBarParent.addView(toolbar, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Customizes the activity's theme.
|
||||||
|
*/
|
||||||
|
protected abstract void customizeActivityTheme(Activity activity);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the resource ID for the content view layout.
|
||||||
|
*/
|
||||||
|
protected abstract int getContentViewResourceId();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the background color for the toolbar.
|
||||||
|
*/
|
||||||
|
protected abstract int getToolbarBackgroundColor();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the navigation icon drawable for the toolbar.
|
||||||
|
*/
|
||||||
|
protected abstract Drawable getNavigationIcon();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the click listener for the toolbar's navigation icon.
|
||||||
|
*/
|
||||||
|
protected abstract View.OnClickListener getNavigationClickListener(Activity activity);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates the PreferenceFragment to be injected into the activity.
|
||||||
|
*/
|
||||||
|
protected PreferenceFragment createPreferenceFragment() {
|
||||||
|
return new ToolbarPreferenceFragment();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performs additional setup after the toolbar is configured.
|
||||||
|
*
|
||||||
|
* @param activity The activity hosting the toolbar.
|
||||||
|
* @param toolbar The configured toolbar.
|
||||||
|
* @param fragment The PreferenceFragment associated with the activity.
|
||||||
|
*/
|
||||||
|
protected void onPostToolbarSetup(Activity activity, Toolbar toolbar, PreferenceFragment fragment) {}
|
||||||
|
}
|
||||||
@@ -4,9 +4,6 @@ import static java.lang.Boolean.FALSE;
|
|||||||
import static java.lang.Boolean.TRUE;
|
import static java.lang.Boolean.TRUE;
|
||||||
import static app.revanced.extension.shared.settings.Setting.parent;
|
import static app.revanced.extension.shared.settings.Setting.parent;
|
||||||
import static app.revanced.extension.shared.spoof.SpoofVideoStreamsPatch.AudioStreamLanguageOverrideAvailability;
|
import static app.revanced.extension.shared.spoof.SpoofVideoStreamsPatch.AudioStreamLanguageOverrideAvailability;
|
||||||
import static app.revanced.extension.shared.spoof.SpoofVideoStreamsPatch.SpoofiOSAvailability;
|
|
||||||
|
|
||||||
import app.revanced.extension.shared.spoof.ClientType;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Settings shared across multiple apps.
|
* Settings shared across multiple apps.
|
||||||
@@ -31,9 +28,4 @@ public class BaseSettings {
|
|||||||
public static final BooleanSetting SPOOF_VIDEO_STREAMS = new BooleanSetting("revanced_spoof_video_streams", TRUE, true, "revanced_spoof_video_streams_user_dialog_message");
|
public static final BooleanSetting SPOOF_VIDEO_STREAMS = new BooleanSetting("revanced_spoof_video_streams", TRUE, true, "revanced_spoof_video_streams_user_dialog_message");
|
||||||
public static final EnumSetting<AppLanguage> SPOOF_VIDEO_STREAMS_LANGUAGE = new EnumSetting<>("revanced_spoof_video_streams_language", AppLanguage.DEFAULT, new AudioStreamLanguageOverrideAvailability());
|
public static final EnumSetting<AppLanguage> SPOOF_VIDEO_STREAMS_LANGUAGE = new EnumSetting<>("revanced_spoof_video_streams_language", AppLanguage.DEFAULT, new AudioStreamLanguageOverrideAvailability());
|
||||||
public static final BooleanSetting SPOOF_STREAMING_DATA_STATS_FOR_NERDS = new BooleanSetting("revanced_spoof_streaming_data_stats_for_nerds", TRUE, parent(SPOOF_VIDEO_STREAMS));
|
public static final BooleanSetting SPOOF_STREAMING_DATA_STATS_FOR_NERDS = new BooleanSetting("revanced_spoof_streaming_data_stats_for_nerds", TRUE, parent(SPOOF_VIDEO_STREAMS));
|
||||||
public static final BooleanSetting SPOOF_VIDEO_STREAMS_IOS_FORCE_AVC = new BooleanSetting("revanced_spoof_video_streams_ios_force_avc", FALSE, true,
|
|
||||||
"revanced_spoof_video_streams_ios_force_avc_user_dialog_message", new SpoofiOSAvailability());
|
|
||||||
// Client type must be last spoof setting due to cyclic references.
|
|
||||||
public static final EnumSetting<ClientType> SPOOF_VIDEO_STREAMS_CLIENT_TYPE = new EnumSetting<>("revanced_spoof_video_streams_client_type", ClientType.ANDROID_UNPLUGGED, true, parent(SPOOF_VIDEO_STREAMS));
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -71,15 +71,20 @@ public class EnumSetting<T extends Enum<?>> extends Setting<T> {
|
|||||||
json.put(importExportKey, value.name().toLowerCase(Locale.ENGLISH));
|
json.put(importExportKey, value.name().toLowerCase(Locale.ENGLISH));
|
||||||
}
|
}
|
||||||
|
|
||||||
@NonNull
|
/**
|
||||||
private T getEnumFromString(String enumName) {
|
* @param enumName Enum name. Casing does not matter.
|
||||||
|
* @return Enum of this type with the same declared name.
|
||||||
|
* @throws IllegalArgumentException if the name is not a valid enum of this type.
|
||||||
|
*/
|
||||||
|
protected T getEnumFromString(String enumName) {
|
||||||
//noinspection ConstantConditions
|
//noinspection ConstantConditions
|
||||||
for (Enum<?> value : defaultValue.getClass().getEnumConstants()) {
|
for (Enum<?> value : defaultValue.getClass().getEnumConstants()) {
|
||||||
if (value.name().equalsIgnoreCase(enumName)) {
|
if (value.name().equalsIgnoreCase(enumName)) {
|
||||||
// noinspection unchecked
|
//noinspection unchecked
|
||||||
return (T) value;
|
return (T) value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new IllegalArgumentException("Unknown enum value: " + enumName);
|
throw new IllegalArgumentException("Unknown enum value: " + enumName);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -103,7 +108,9 @@ public class EnumSetting<T extends Enum<?>> extends Setting<T> {
|
|||||||
* Availability based on if this setting is currently set to any of the provided types.
|
* Availability based on if this setting is currently set to any of the provided types.
|
||||||
*/
|
*/
|
||||||
@SafeVarargs
|
@SafeVarargs
|
||||||
public final Setting.Availability availability(@NonNull T... types) {
|
public final Setting.Availability availability(T... types) {
|
||||||
|
Objects.requireNonNull(types);
|
||||||
|
|
||||||
return () -> {
|
return () -> {
|
||||||
T currentEnumType = get();
|
T currentEnumType = get();
|
||||||
for (T enumType : types) {
|
for (T enumType : types) {
|
||||||
|
|||||||
@@ -28,16 +28,14 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* Availability based on a single parent setting being enabled.
|
* Availability based on a single parent setting being enabled.
|
||||||
*/
|
*/
|
||||||
@NonNull
|
public static Availability parent(BooleanSetting parent) {
|
||||||
public static Availability parent(@NonNull BooleanSetting parent) {
|
|
||||||
return parent::get;
|
return parent::get;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Availability based on all parents being enabled.
|
* Availability based on all parents being enabled.
|
||||||
*/
|
*/
|
||||||
@NonNull
|
public static Availability parentsAll(BooleanSetting... parents) {
|
||||||
public static Availability parentsAll(@NonNull BooleanSetting... parents) {
|
|
||||||
return () -> {
|
return () -> {
|
||||||
for (BooleanSetting parent : parents) {
|
for (BooleanSetting parent : parents) {
|
||||||
if (!parent.get()) return false;
|
if (!parent.get()) return false;
|
||||||
@@ -49,8 +47,7 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* Availability based on any parent being enabled.
|
* Availability based on any parent being enabled.
|
||||||
*/
|
*/
|
||||||
@NonNull
|
public static Availability parentsAny(BooleanSetting... parents) {
|
||||||
public static Availability parentsAny(@NonNull BooleanSetting... parents) {
|
|
||||||
return () -> {
|
return () -> {
|
||||||
for (BooleanSetting parent : parents) {
|
for (BooleanSetting parent : parents) {
|
||||||
if (parent.get()) return true;
|
if (parent.get()) return true;
|
||||||
@@ -79,7 +76,7 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* Adds a callback for {@link #importFromJSON(Context, String)} and {@link #exportToJson(Context)}.
|
* Adds a callback for {@link #importFromJSON(Context, String)} and {@link #exportToJson(Context)}.
|
||||||
*/
|
*/
|
||||||
public static void addImportExportCallback(@NonNull ImportExportCallback callback) {
|
public static void addImportExportCallback(ImportExportCallback callback) {
|
||||||
importExportCallbacks.add(Objects.requireNonNull(callback));
|
importExportCallbacks.add(Objects.requireNonNull(callback));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,14 +97,13 @@ public abstract class Setting<T> {
|
|||||||
public static final SharedPrefCategory preferences = new SharedPrefCategory("revanced_prefs");
|
public static final SharedPrefCategory preferences = new SharedPrefCategory("revanced_prefs");
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
public static Setting<?> getSettingFromPath(@NonNull String str) {
|
public static Setting<?> getSettingFromPath(String str) {
|
||||||
return PATH_TO_SETTINGS.get(str);
|
return PATH_TO_SETTINGS.get(str);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return All settings that have been created.
|
* @return All settings that have been created.
|
||||||
*/
|
*/
|
||||||
@NonNull
|
|
||||||
public static List<Setting<?>> allLoadedSettings() {
|
public static List<Setting<?>> allLoadedSettings() {
|
||||||
return Collections.unmodifiableList(SETTINGS);
|
return Collections.unmodifiableList(SETTINGS);
|
||||||
}
|
}
|
||||||
@@ -115,7 +111,6 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* @return All settings that have been created, sorted by keys.
|
* @return All settings that have been created, sorted by keys.
|
||||||
*/
|
*/
|
||||||
@NonNull
|
|
||||||
private static List<Setting<?>> allLoadedSettingsSorted() {
|
private static List<Setting<?>> allLoadedSettingsSorted() {
|
||||||
Collections.sort(SETTINGS, (Setting<?> o1, Setting<?> o2) -> o1.key.compareTo(o2.key));
|
Collections.sort(SETTINGS, (Setting<?> o1, Setting<?> o2) -> o1.key.compareTo(o2.key));
|
||||||
return allLoadedSettings();
|
return allLoadedSettings();
|
||||||
@@ -124,13 +119,11 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* The key used to store the value in the shared preferences.
|
* The key used to store the value in the shared preferences.
|
||||||
*/
|
*/
|
||||||
@NonNull
|
|
||||||
public final String key;
|
public final String key;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The default value of the setting.
|
* The default value of the setting.
|
||||||
*/
|
*/
|
||||||
@NonNull
|
|
||||||
public final T defaultValue;
|
public final T defaultValue;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -161,7 +154,6 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* The value of the setting.
|
* The value of the setting.
|
||||||
*/
|
*/
|
||||||
@NonNull
|
|
||||||
protected volatile T value;
|
protected volatile T value;
|
||||||
|
|
||||||
public Setting(String key, T defaultValue) {
|
public Setting(String key, T defaultValue) {
|
||||||
@@ -199,8 +191,8 @@ public abstract class Setting<T> {
|
|||||||
* @param userDialogMessage Confirmation message to display, if the user tries to change the setting from the default value.
|
* @param userDialogMessage Confirmation message to display, if the user tries to change the setting from the default value.
|
||||||
* @param availability Condition that must be true, for this setting to be available to configure.
|
* @param availability Condition that must be true, for this setting to be available to configure.
|
||||||
*/
|
*/
|
||||||
public Setting(@NonNull String key,
|
public Setting(String key,
|
||||||
@NonNull T defaultValue,
|
T defaultValue,
|
||||||
boolean rebootApp,
|
boolean rebootApp,
|
||||||
boolean includeWithImportExport,
|
boolean includeWithImportExport,
|
||||||
@Nullable String userDialogMessage,
|
@Nullable String userDialogMessage,
|
||||||
@@ -227,7 +219,7 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* Migrate a setting value if the path is renamed but otherwise the old and new settings are identical.
|
* Migrate a setting value if the path is renamed but otherwise the old and new settings are identical.
|
||||||
*/
|
*/
|
||||||
public static <T> void migrateOldSettingToNew(@NonNull Setting<T> oldSetting, @NonNull Setting<T> newSetting) {
|
public static <T> void migrateOldSettingToNew(Setting<T> oldSetting, Setting<T> newSetting) {
|
||||||
if (oldSetting == newSetting) throw new IllegalArgumentException();
|
if (oldSetting == newSetting) throw new IllegalArgumentException();
|
||||||
|
|
||||||
if (!oldSetting.isSetToDefault()) {
|
if (!oldSetting.isSetToDefault()) {
|
||||||
@@ -243,7 +235,7 @@ public abstract class Setting<T> {
|
|||||||
* This method will be deleted in the future.
|
* This method will be deleted in the future.
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("rawtypes")
|
@SuppressWarnings("rawtypes")
|
||||||
public static void migrateFromOldPreferences(@NonNull SharedPrefCategory oldPrefs, @NonNull Setting setting, String settingKey) {
|
public static void migrateFromOldPreferences(SharedPrefCategory oldPrefs, Setting setting, String settingKey) {
|
||||||
if (!oldPrefs.preferences.contains(settingKey)) {
|
if (!oldPrefs.preferences.contains(settingKey)) {
|
||||||
return; // Nothing to do.
|
return; // Nothing to do.
|
||||||
}
|
}
|
||||||
@@ -285,7 +277,7 @@ public abstract class Setting<T> {
|
|||||||
* This intentionally is a static method to deter
|
* This intentionally is a static method to deter
|
||||||
* accidental usage when {@link #save(Object)} was intended.
|
* accidental usage when {@link #save(Object)} was intended.
|
||||||
*/
|
*/
|
||||||
public static void privateSetValueFromString(@NonNull Setting<?> setting, @NonNull String newValue) {
|
public static void privateSetValueFromString(Setting<?> setting, String newValue) {
|
||||||
setting.setValueFromString(newValue);
|
setting.setValueFromString(newValue);
|
||||||
|
|
||||||
// Clear the preference value since default is used, to allow changing
|
// Clear the preference value since default is used, to allow changing
|
||||||
@@ -299,7 +291,7 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* Sets the value of {@link #value}, but do not save to {@link #preferences}.
|
* Sets the value of {@link #value}, but do not save to {@link #preferences}.
|
||||||
*/
|
*/
|
||||||
protected abstract void setValueFromString(@NonNull String newValue);
|
protected abstract void setValueFromString(String newValue);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load and set the value of {@link #value}.
|
* Load and set the value of {@link #value}.
|
||||||
@@ -309,7 +301,7 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* Persistently saves the value.
|
* Persistently saves the value.
|
||||||
*/
|
*/
|
||||||
public final void save(@NonNull T newValue) {
|
public final void save(T newValue) {
|
||||||
if (value.equals(newValue)) {
|
if (value.equals(newValue)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -406,7 +398,6 @@ public abstract class Setting<T> {
|
|||||||
json.put(importExportKey, value);
|
json.put(importExportKey, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
@NonNull
|
|
||||||
public static String exportToJson(@Nullable Context alertDialogContext) {
|
public static String exportToJson(@Nullable Context alertDialogContext) {
|
||||||
try {
|
try {
|
||||||
JSONObject json = new JSONObject();
|
JSONObject json = new JSONObject();
|
||||||
@@ -445,7 +436,7 @@ public abstract class Setting<T> {
|
|||||||
/**
|
/**
|
||||||
* @return if any settings that require a reboot were changed.
|
* @return if any settings that require a reboot were changed.
|
||||||
*/
|
*/
|
||||||
public static boolean importFromJSON(@NonNull Context alertDialogContext, @NonNull String settingsJsonString) {
|
public static boolean importFromJSON(Context alertDialogContext, String settingsJsonString) {
|
||||||
try {
|
try {
|
||||||
if (!settingsJsonString.matches("[\\s\\S]*\\{")) {
|
if (!settingsJsonString.matches("[\\s\\S]*\\{")) {
|
||||||
settingsJsonString = '{' + settingsJsonString + '}'; // Restore outer JSON braces
|
settingsJsonString = '{' + settingsJsonString + '}'; // Restore outer JSON braces
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
package app.revanced.extension.youtube.settings.preference;
|
package app.revanced.extension.shared.settings.preference;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.util.AttributeSet;
|
import android.util.AttributeSet;
|
||||||
import android.preference.Preference;
|
import android.preference.Preference;
|
||||||
import app.revanced.extension.shared.settings.preference.LogBufferManager;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A custom preference that clears the ReVanced debug log buffer when clicked.
|
* A custom preference that clears the ReVanced debug log buffer when clicked.
|
||||||
@@ -1,7 +1,5 @@
|
|||||||
package app.revanced.extension.shared.settings.preference;
|
package app.revanced.extension.shared.settings.preference;
|
||||||
|
|
||||||
import static app.revanced.extension.shared.Utils.dipToPixels;
|
|
||||||
|
|
||||||
import android.app.Dialog;
|
import android.app.Dialog;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
@@ -26,7 +24,7 @@ public class CustomDialogListPreference extends ListPreference {
|
|||||||
/**
|
/**
|
||||||
* Custom ArrayAdapter to handle checkmark visibility.
|
* Custom ArrayAdapter to handle checkmark visibility.
|
||||||
*/
|
*/
|
||||||
private static class ListPreferenceArrayAdapter extends ArrayAdapter<CharSequence> {
|
public static class ListPreferenceArrayAdapter extends ArrayAdapter<CharSequence> {
|
||||||
private static class SubViewDataContainer {
|
private static class SubViewDataContainer {
|
||||||
ImageView checkIcon;
|
ImageView checkIcon;
|
||||||
View placeholder;
|
View placeholder;
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
package app.revanced.extension.youtube.settings.preference;
|
package app.revanced.extension.shared.settings.preference;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.util.AttributeSet;
|
import android.util.AttributeSet;
|
||||||
import android.preference.Preference;
|
import android.preference.Preference;
|
||||||
import app.revanced.extension.shared.settings.preference.LogBufferManager;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A custom preference that triggers exporting ReVanced debug logs to the clipboard when clicked.
|
* A custom preference that triggers exporting ReVanced debug logs to the clipboard when clicked.
|
||||||
@@ -1,28 +1,15 @@
|
|||||||
package app.revanced.extension.shared.settings.preference;
|
package app.revanced.extension.shared.settings.preference;
|
||||||
|
|
||||||
import static app.revanced.extension.shared.StringRef.str;
|
import static app.revanced.extension.shared.StringRef.str;
|
||||||
import static app.revanced.extension.shared.Utils.dipToPixels;
|
|
||||||
|
|
||||||
import android.app.Dialog;
|
import android.app.Dialog;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.graphics.Color;
|
|
||||||
import android.graphics.drawable.Drawable;
|
|
||||||
import android.graphics.drawable.LayerDrawable;
|
|
||||||
import android.graphics.drawable.shapes.RectShape;
|
|
||||||
import android.graphics.drawable.shapes.RoundRectShape;
|
|
||||||
import android.graphics.drawable.ShapeDrawable;
|
|
||||||
import android.graphics.Paint.Style;
|
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.preference.EditTextPreference;
|
import android.preference.EditTextPreference;
|
||||||
import android.text.TextUtils;
|
|
||||||
import android.util.AttributeSet;
|
import android.util.AttributeSet;
|
||||||
import android.util.Pair;
|
import android.util.Pair;
|
||||||
import android.view.ViewGroup;
|
|
||||||
import android.widget.Button;
|
|
||||||
import android.widget.EditText;
|
import android.widget.EditText;
|
||||||
import android.widget.LinearLayout;
|
import android.widget.LinearLayout;
|
||||||
import android.widget.LinearLayout;
|
|
||||||
import android.widget.TextView;
|
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,150 @@
|
|||||||
|
package app.revanced.extension.shared.settings.preference;
|
||||||
|
|
||||||
|
import android.annotation.SuppressLint;
|
||||||
|
import android.app.Dialog;
|
||||||
|
import android.graphics.Insets;
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
|
import android.os.Build;
|
||||||
|
import android.preference.Preference;
|
||||||
|
import android.preference.PreferenceScreen;
|
||||||
|
import android.util.TypedValue;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.view.Window;
|
||||||
|
import android.view.WindowInsets;
|
||||||
|
import android.widget.TextView;
|
||||||
|
import android.widget.Toolbar;
|
||||||
|
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import app.revanced.extension.shared.Utils;
|
||||||
|
import app.revanced.extension.shared.settings.BaseActivityHook;
|
||||||
|
|
||||||
|
@SuppressWarnings({"deprecation", "NewApi"})
|
||||||
|
public class ToolbarPreferenceFragment extends AbstractPreferenceFragment {
|
||||||
|
/**
|
||||||
|
* Sets toolbar for all nested preference screens.
|
||||||
|
*/
|
||||||
|
protected void setPreferenceScreenToolbar(PreferenceScreen parentScreen) {
|
||||||
|
for (int i = 0, count = parentScreen.getPreferenceCount(); i < count; i++) {
|
||||||
|
Preference childPreference = parentScreen.getPreference(i);
|
||||||
|
if (childPreference instanceof PreferenceScreen) {
|
||||||
|
// Recursively set sub preferences.
|
||||||
|
setPreferenceScreenToolbar((PreferenceScreen) childPreference);
|
||||||
|
|
||||||
|
childPreference.setOnPreferenceClickListener(
|
||||||
|
childScreen -> {
|
||||||
|
Dialog preferenceScreenDialog = ((PreferenceScreen) childScreen).getDialog();
|
||||||
|
ViewGroup rootView = (ViewGroup) preferenceScreenDialog
|
||||||
|
.findViewById(android.R.id.content)
|
||||||
|
.getParent();
|
||||||
|
|
||||||
|
// Allow package-specific background customization.
|
||||||
|
customizeDialogBackground(rootView);
|
||||||
|
|
||||||
|
// Fix the system navigation bar color for submenus.
|
||||||
|
setNavigationBarColor(preferenceScreenDialog.getWindow());
|
||||||
|
|
||||||
|
// Fix edge-to-edge screen with Android 15 and YT 19.45+
|
||||||
|
// https://developer.android.com/develop/ui/views/layout/edge-to-edge#system-bars-insets
|
||||||
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
|
||||||
|
rootView.setOnApplyWindowInsetsListener((v, insets) -> {
|
||||||
|
Insets statusInsets = insets.getInsets(WindowInsets.Type.statusBars());
|
||||||
|
Insets navInsets = insets.getInsets(WindowInsets.Type.navigationBars());
|
||||||
|
Insets cutoutInsets = insets.getInsets(WindowInsets.Type.displayCutout());
|
||||||
|
|
||||||
|
// Apply padding for display cutout in landscape.
|
||||||
|
int leftPadding = cutoutInsets.left;
|
||||||
|
int rightPadding = cutoutInsets.right;
|
||||||
|
int topPadding = statusInsets.top;
|
||||||
|
int bottomPadding = navInsets.bottom;
|
||||||
|
|
||||||
|
v.setPadding(leftPadding, topPadding, rightPadding, bottomPadding);
|
||||||
|
return insets;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
Toolbar toolbar = new Toolbar(childScreen.getContext());
|
||||||
|
toolbar.setTitle(childScreen.getTitle());
|
||||||
|
toolbar.setNavigationIcon(getBackButtonDrawable());
|
||||||
|
toolbar.setNavigationOnClickListener(view -> preferenceScreenDialog.dismiss());
|
||||||
|
|
||||||
|
final int margin = Utils.dipToPixels(16);
|
||||||
|
toolbar.setTitleMargin(margin, 0, margin, 0);
|
||||||
|
|
||||||
|
TextView toolbarTextView = Utils.getChildView(toolbar,
|
||||||
|
true, TextView.class::isInstance);
|
||||||
|
if (toolbarTextView != null) {
|
||||||
|
toolbarTextView.setTextColor(Utils.getAppForegroundColor());
|
||||||
|
toolbarTextView.setTextSize(TypedValue.COMPLEX_UNIT_SP, 20);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Allow package-specific toolbar customization.
|
||||||
|
customizeToolbar(toolbar);
|
||||||
|
|
||||||
|
// Allow package-specific post-toolbar setup.
|
||||||
|
onPostToolbarSetup(toolbar, preferenceScreenDialog);
|
||||||
|
|
||||||
|
rootView.addView(toolbar, 0);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the system navigation bar color for the activity.
|
||||||
|
* Applies the background color obtained from {@link Utils#getAppBackgroundColor()} to the navigation bar.
|
||||||
|
* For Android 10 (API 29) and above, enforces navigation bar contrast to ensure visibility.
|
||||||
|
*/
|
||||||
|
public static void setNavigationBarColor(@Nullable Window window) {
|
||||||
|
if (window == null) {
|
||||||
|
Logger.printDebug(() -> "Cannot set navigation bar color, window is null");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
window.setNavigationBarColor(Utils.getAppBackgroundColor());
|
||||||
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
|
||||||
|
window.setNavigationBarContrastEnforced(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the drawable for the back button.
|
||||||
|
*/
|
||||||
|
@SuppressLint("UseCompatLoadingForDrawables")
|
||||||
|
public static Drawable getBackButtonDrawable() {
|
||||||
|
final int backButtonResource = Utils.getResourceIdentifier(
|
||||||
|
"revanced_settings_toolbar_arrow_left", "drawable");
|
||||||
|
Drawable drawable = Utils.getContext().getResources().getDrawable(backButtonResource);
|
||||||
|
customizeBackButtonDrawable(drawable);
|
||||||
|
return drawable;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Customizes the back button drawable.
|
||||||
|
*/
|
||||||
|
protected static void customizeBackButtonDrawable(Drawable drawable) {
|
||||||
|
drawable.setTint(Utils.getAppForegroundColor());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allows subclasses to customize the dialog's root view background.
|
||||||
|
*/
|
||||||
|
protected void customizeDialogBackground(ViewGroup rootView) {
|
||||||
|
rootView.setBackgroundColor(Utils.getAppBackgroundColor());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allows subclasses to customize the toolbar.
|
||||||
|
*/
|
||||||
|
protected void customizeToolbar(Toolbar toolbar) {
|
||||||
|
BaseActivityHook.setToolbarLayoutParams(toolbar);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allows subclasses to perform actions after toolbar setup.
|
||||||
|
*/
|
||||||
|
protected void onPostToolbarSetup(Toolbar toolbar, Dialog preferenceScreenDialog) {}
|
||||||
|
}
|
||||||
@@ -2,17 +2,22 @@ package app.revanced.extension.shared.spoof;
|
|||||||
|
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
import androidx.annotation.Nullable;
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.settings.BaseSettings;
|
|
||||||
|
|
||||||
|
@SuppressWarnings("ConstantLocale")
|
||||||
public enum ClientType {
|
public enum ClientType {
|
||||||
|
/**
|
||||||
|
* Video not playable: Kids / Paid / Movie / Private / Age-restricted.
|
||||||
|
* This client can only be used when logged out.
|
||||||
|
*/
|
||||||
// https://dumps.tadiphone.dev/dumps/oculus/eureka
|
// https://dumps.tadiphone.dev/dumps/oculus/eureka
|
||||||
ANDROID_VR_NO_AUTH(
|
ANDROID_VR_1_61_48(
|
||||||
28,
|
28,
|
||||||
"ANDROID_VR",
|
"ANDROID_VR",
|
||||||
"com.google.android.apps.youtube.vr.oculus",
|
"com.google.android.apps.youtube.vr.oculus",
|
||||||
@@ -26,30 +31,31 @@ public enum ClientType {
|
|||||||
"132.0.6808.3",
|
"132.0.6808.3",
|
||||||
"1.61.48",
|
"1.61.48",
|
||||||
false,
|
false,
|
||||||
false,
|
"Android VR 1.61"
|
||||||
"Android VR No auth"
|
|
||||||
),
|
),
|
||||||
// Chromecast with Google TV 4K.
|
/**
|
||||||
// https://dumps.tadiphone.dev/dumps/google/kirkwood
|
* Uses non adaptive bitrate, which fixes audio stuttering with YT Music.
|
||||||
ANDROID_UNPLUGGED(
|
* Does not use AV1.
|
||||||
29,
|
*/
|
||||||
"ANDROID_UNPLUGGED",
|
ANDROID_VR_1_43_32(
|
||||||
"com.google.android.apps.youtube.unplugged",
|
ANDROID_VR_1_61_48.id,
|
||||||
"Google",
|
ANDROID_VR_1_61_48.clientName,
|
||||||
"Google TV Streamer",
|
Objects.requireNonNull(ANDROID_VR_1_61_48.packageName),
|
||||||
"Android",
|
ANDROID_VR_1_61_48.deviceMake,
|
||||||
"14",
|
ANDROID_VR_1_61_48.deviceModel,
|
||||||
"34",
|
ANDROID_VR_1_61_48.osName,
|
||||||
"UTT3.240625.001.K5",
|
ANDROID_VR_1_61_48.osVersion,
|
||||||
"132.0.6808.3",
|
Objects.requireNonNull(ANDROID_VR_1_61_48.androidSdkVersion),
|
||||||
"8.49.0",
|
Objects.requireNonNull(ANDROID_VR_1_61_48.buildId),
|
||||||
true,
|
"107.0.5284.2",
|
||||||
true,
|
"1.43.32",
|
||||||
"Android TV"
|
ANDROID_VR_1_61_48.useAuth,
|
||||||
|
"Android VR 1.43"
|
||||||
),
|
),
|
||||||
// Cannot play livestreams and lacks HDR, but can play videos with music and labeled "for children".
|
/**
|
||||||
// Google Pixel 9 Pro Fold
|
* Cannot play livestreams and lacks HDR, but can play videos with music and labeled "for children".
|
||||||
// https://dumps.tadiphone.dev/dumps/google/barbet
|
* <a href="https://dumps.tadiphone.dev/dumps/google/barbet">Google Pixel 9 Pro Fold</a>
|
||||||
|
*/
|
||||||
ANDROID_CREATOR(
|
ANDROID_CREATOR(
|
||||||
14,
|
14,
|
||||||
"ANDROID_CREATOR",
|
"ANDROID_CREATOR",
|
||||||
@@ -63,61 +69,47 @@ public enum ClientType {
|
|||||||
"132.0.6779.0",
|
"132.0.6779.0",
|
||||||
"23.47.101",
|
"23.47.101",
|
||||||
true,
|
true,
|
||||||
true,
|
"Android Studio"
|
||||||
"Android Creator"
|
|
||||||
),
|
),
|
||||||
IOS_UNPLUGGED(
|
/**
|
||||||
33,
|
* Internal YT client for an unreleased YT client. May stop working at any time.
|
||||||
"IOS_UNPLUGGED",
|
*/
|
||||||
"com.google.ios.youtubeunplugged",
|
VISIONOS(101,
|
||||||
|
"VISIONOS",
|
||||||
"Apple",
|
"Apple",
|
||||||
forceAVC()
|
"RealityDevice14,1",
|
||||||
// 11 Pro Max (last device with iOS 13)
|
"visionOS",
|
||||||
? "iPhone12,5"
|
"1.3.21O771",
|
||||||
// 15 Pro Max
|
"0.1",
|
||||||
: "iPhone16,2",
|
"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/18.0 Safari/605.1.15",
|
||||||
"iOS",
|
false,
|
||||||
forceAVC()
|
"visionOS"
|
||||||
// iOS 13 and earlier uses only AVC. 14+ adds VP9 and AV1.
|
|
||||||
? "13.7.17H35"
|
|
||||||
: "18.2.22C152",
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
// Version number should be a valid iOS release.
|
|
||||||
// https://www.ipa4fun.com/history/152043/
|
|
||||||
forceAVC()
|
|
||||||
// Some newer versions can also force AVC,
|
|
||||||
// but 6.45 is the last version that supports iOS 13.
|
|
||||||
? "6.45"
|
|
||||||
: "8.49",
|
|
||||||
true,
|
|
||||||
true,
|
|
||||||
forceAVC()
|
|
||||||
? "iOS TV Force AVC"
|
|
||||||
: "iOS TV"
|
|
||||||
),
|
),
|
||||||
ANDROID_VR_AUTH(
|
/**
|
||||||
ANDROID_VR_NO_AUTH.id,
|
* The device machine id for the iPad 6th Gen (iPad7,6).
|
||||||
ANDROID_VR_NO_AUTH.clientName,
|
* AV1 hardware decoding is not supported.
|
||||||
ANDROID_VR_NO_AUTH.packageName,
|
* See [this GitHub Gist](https://gist.github.com/adamawolf/3048717) for more information.
|
||||||
ANDROID_VR_NO_AUTH.deviceMake,
|
*
|
||||||
ANDROID_VR_NO_AUTH.deviceModel,
|
* Based on Google's actions to date, PoToken may not be required on devices with very low specs.
|
||||||
ANDROID_VR_NO_AUTH.osName,
|
* For example, suppose the User-Agent for a PlayStation 3 (with 256MB of RAM) is used.
|
||||||
ANDROID_VR_NO_AUTH.osVersion,
|
* Accessing 'Web' (https://www.youtube.com) will redirect to 'TV' (https://www.youtube.com/tv).
|
||||||
ANDROID_VR_NO_AUTH.androidSdkVersion,
|
* 'TV' target devices with very low specs, such as embedded devices, game consoles, and blu-ray players, so PoToken is not required.
|
||||||
ANDROID_VR_NO_AUTH.buildId,
|
*
|
||||||
ANDROID_VR_NO_AUTH.cronetVersion,
|
* For this reason, the device machine id for the iPad 6th Gen (with 2GB of RAM),
|
||||||
ANDROID_VR_NO_AUTH.clientVersion,
|
* the lowest spec device capable of running iPadOS 17, was used.
|
||||||
ANDROID_VR_NO_AUTH.requiresAuth,
|
*/
|
||||||
true,
|
IPADOS(5,
|
||||||
"Android VR Auth"
|
"IOS",
|
||||||
|
"Apple",
|
||||||
|
"iPad7,6",
|
||||||
|
"iPadOS",
|
||||||
|
"17.7.10.21H450",
|
||||||
|
"19.22.3",
|
||||||
|
"com.google.ios.youtube/19.22.3 (iPad7,6; U; CPU iPadOS 17_7_10 like Mac OS X; " + Locale.getDefault() + ")",
|
||||||
|
false,
|
||||||
|
"iPadOS"
|
||||||
);
|
);
|
||||||
|
|
||||||
private static boolean forceAVC() {
|
|
||||||
return BaseSettings.SPOOF_VIDEO_STREAMS_IOS_FORCE_AVC.get();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* YouTube
|
* YouTube
|
||||||
* <a href="https://github.com/zerodytrash/YouTube-Internal-Clients?tab=readme-ov-file#clients">client type</a>
|
* <a href="https://github.com/zerodytrash/YouTube-Internal-Clients?tab=readme-ov-file#clients">client type</a>
|
||||||
@@ -129,6 +121,7 @@ public enum ClientType {
|
|||||||
/**
|
/**
|
||||||
* App package name.
|
* App package name.
|
||||||
*/
|
*/
|
||||||
|
@Nullable
|
||||||
private final String packageName;
|
private final String packageName;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -182,12 +175,6 @@ public enum ClientType {
|
|||||||
*/
|
*/
|
||||||
public final String clientVersion;
|
public final String clientVersion;
|
||||||
|
|
||||||
/**
|
|
||||||
* If this client requires authentication and does not work
|
|
||||||
* if logged out or in incognito mode.
|
|
||||||
*/
|
|
||||||
public final boolean requiresAuth;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If the client should use authentication if available.
|
* If the client should use authentication if available.
|
||||||
*/
|
*/
|
||||||
@@ -198,19 +185,20 @@ public enum ClientType {
|
|||||||
*/
|
*/
|
||||||
public final String friendlyName;
|
public final String friendlyName;
|
||||||
|
|
||||||
@SuppressWarnings("ConstantLocale")
|
/**
|
||||||
|
* Android constructor.
|
||||||
|
*/
|
||||||
ClientType(int id,
|
ClientType(int id,
|
||||||
String clientName,
|
String clientName,
|
||||||
String packageName,
|
@NonNull String packageName,
|
||||||
String deviceMake,
|
String deviceMake,
|
||||||
String deviceModel,
|
String deviceModel,
|
||||||
String osName,
|
String osName,
|
||||||
String osVersion,
|
String osVersion,
|
||||||
@Nullable String androidSdkVersion,
|
@NonNull String androidSdkVersion,
|
||||||
@Nullable String buildId,
|
@NonNull String buildId,
|
||||||
@Nullable String cronetVersion,
|
@NonNull String cronetVersion,
|
||||||
String clientVersion,
|
String clientVersion,
|
||||||
boolean requiresAuth,
|
|
||||||
boolean useAuth,
|
boolean useAuth,
|
||||||
String friendlyName) {
|
String friendlyName) {
|
||||||
this.id = id;
|
this.id = id;
|
||||||
@@ -224,36 +212,46 @@ public enum ClientType {
|
|||||||
this.buildId = buildId;
|
this.buildId = buildId;
|
||||||
this.cronetVersion = cronetVersion;
|
this.cronetVersion = cronetVersion;
|
||||||
this.clientVersion = clientVersion;
|
this.clientVersion = clientVersion;
|
||||||
this.requiresAuth = requiresAuth;
|
|
||||||
this.useAuth = useAuth;
|
this.useAuth = useAuth;
|
||||||
this.friendlyName = friendlyName;
|
this.friendlyName = friendlyName;
|
||||||
|
|
||||||
Locale defaultLocale = Locale.getDefault();
|
Locale defaultLocale = Locale.getDefault();
|
||||||
if (androidSdkVersion == null) {
|
this.userAgent = String.format("%s/%s (Linux; U; Android %s; %s; %s; Build/%s; Cronet/%s)",
|
||||||
// Convert version from '18.2.22C152' into '18_2_22'
|
packageName,
|
||||||
String userAgentOsVersion = osVersion
|
clientVersion,
|
||||||
.replaceAll("(\\d+\\.\\d+\\.\\d+).*", "$1")
|
osVersion,
|
||||||
.replace(".", "_");
|
defaultLocale,
|
||||||
// https://github.com/mitmproxy/mitmproxy/issues/4836
|
deviceModel,
|
||||||
this.userAgent = String.format("%s/%s (%s; U; CPU iOS %s like Mac OS X; %s)",
|
Objects.requireNonNull(buildId),
|
||||||
packageName,
|
Objects.requireNonNull(cronetVersion)
|
||||||
clientVersion,
|
);
|
||||||
deviceModel,
|
|
||||||
userAgentOsVersion,
|
|
||||||
defaultLocale
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
this.userAgent = String.format("%s/%s (Linux; U; Android %s; %s; %s; Build/%s; Cronet/%s)",
|
|
||||||
packageName,
|
|
||||||
clientVersion,
|
|
||||||
osVersion,
|
|
||||||
defaultLocale,
|
|
||||||
deviceModel,
|
|
||||||
Objects.requireNonNull(buildId),
|
|
||||||
Objects.requireNonNull(cronetVersion)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
Logger.printDebug(() -> "userAgent: " + this.userAgent);
|
Logger.printDebug(() -> "userAgent: " + this.userAgent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("ConstantLocale")
|
||||||
|
ClientType(int id,
|
||||||
|
String clientName,
|
||||||
|
String deviceMake,
|
||||||
|
String deviceModel,
|
||||||
|
String osName,
|
||||||
|
String osVersion,
|
||||||
|
String clientVersion,
|
||||||
|
String userAgent,
|
||||||
|
boolean useAuth,
|
||||||
|
String friendlyName) {
|
||||||
|
this.id = id;
|
||||||
|
this.clientName = clientName;
|
||||||
|
this.deviceMake = deviceMake;
|
||||||
|
this.deviceModel = deviceModel;
|
||||||
|
this.osName = osName;
|
||||||
|
this.osVersion = osVersion;
|
||||||
|
this.clientVersion = clientVersion;
|
||||||
|
this.userAgent = userAgent;
|
||||||
|
this.useAuth = useAuth;
|
||||||
|
this.friendlyName = friendlyName;
|
||||||
|
this.packageName = null;
|
||||||
|
this.androidSdkVersion = null;
|
||||||
|
this.buildId = null;
|
||||||
|
this.cronetVersion = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,38 +6,70 @@ import android.text.TextUtils;
|
|||||||
import androidx.annotation.Nullable;
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.Utils;
|
import app.revanced.extension.shared.Utils;
|
||||||
|
import app.revanced.extension.shared.settings.AppLanguage;
|
||||||
import app.revanced.extension.shared.settings.BaseSettings;
|
import app.revanced.extension.shared.settings.BaseSettings;
|
||||||
import app.revanced.extension.shared.settings.Setting;
|
import app.revanced.extension.shared.settings.Setting;
|
||||||
import app.revanced.extension.shared.spoof.requests.StreamingDataRequest;
|
import app.revanced.extension.shared.spoof.requests.StreamingDataRequest;
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
public class SpoofVideoStreamsPatch {
|
public class SpoofVideoStreamsPatch {
|
||||||
|
/**
|
||||||
|
* Domain used for internet connectivity verification.
|
||||||
|
* It has an empty response body and is only used to check for a 204 response code.
|
||||||
|
* <p>
|
||||||
|
* If an unreachable IP address (127.0.0.1) is used, no response code is provided.
|
||||||
|
* <p>
|
||||||
|
* YouTube handles unreachable IP addresses without issue.
|
||||||
|
* YouTube Music has an issue with waiting for the Cronet connect timeout of 30s on mobile networks.
|
||||||
|
* <p>
|
||||||
|
* Using a VPN or DNS can temporarily resolve this issue,
|
||||||
|
* But the ideal workaround is to avoid using an unreachable IP address.
|
||||||
|
*/
|
||||||
|
private static final String INTERNET_CONNECTION_CHECK_URI_STRING = "https://www.google.com/gen_204";
|
||||||
|
private static final Uri INTERNET_CONNECTION_CHECK_URI = Uri.parse(INTERNET_CONNECTION_CHECK_URI_STRING);
|
||||||
|
|
||||||
private static final boolean SPOOF_STREAMING_DATA = BaseSettings.SPOOF_VIDEO_STREAMS.get();
|
private static final boolean SPOOF_STREAMING_DATA = BaseSettings.SPOOF_VIDEO_STREAMS.get();
|
||||||
|
|
||||||
private static final boolean FIX_HLS_CURRENT_TIME = SPOOF_STREAMING_DATA
|
@Nullable
|
||||||
&& BaseSettings.SPOOF_VIDEO_STREAMS_CLIENT_TYPE.get() == ClientType.IOS_UNPLUGGED;
|
private static volatile AppLanguage languageOverride;
|
||||||
|
|
||||||
/**
|
private static volatile ClientType preferredClient = ClientType.ANDROID_VR_1_61_48;
|
||||||
* Any unreachable ip address. Used to intentionally fail requests.
|
|
||||||
*/
|
|
||||||
private static final String UNREACHABLE_HOST_URI_STRING = "https://127.0.0.0";
|
|
||||||
private static final Uri UNREACHABLE_HOST_URI = Uri.parse(UNREACHABLE_HOST_URI_STRING);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return If this patch was included during patching.
|
* @return If this patch was included during patching.
|
||||||
*/
|
*/
|
||||||
private static boolean isPatchIncluded() {
|
public static boolean isPatchIncluded() {
|
||||||
return false; // Modified during patching.
|
return false; // Modified during patching.
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean notSpoofingToAndroid() {
|
@Nullable
|
||||||
return !isPatchIncluded()
|
public static AppLanguage getLanguageOverride() {
|
||||||
|| !BaseSettings.SPOOF_VIDEO_STREAMS.get()
|
return languageOverride;
|
||||||
|| BaseSettings.SPOOF_VIDEO_STREAMS_CLIENT_TYPE.get() == ClientType.IOS_UNPLUGGED;
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param language Language override for non-authenticated requests. If this is null then
|
||||||
|
* {@link BaseSettings#SPOOF_VIDEO_STREAMS_LANGUAGE} is used.
|
||||||
|
*/
|
||||||
|
public static void setLanguageOverride(@Nullable AppLanguage language) {
|
||||||
|
languageOverride = language;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setClientsToUse(List<ClientType> availableClients, ClientType client) {
|
||||||
|
preferredClient = Objects.requireNonNull(client);
|
||||||
|
StreamingDataRequest.setClientOrderToUse(availableClients, client);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean spoofingToClientWithNoMultiAudioStreams() {
|
||||||
|
return isPatchIncluded()
|
||||||
|
&& SPOOF_STREAMING_DATA
|
||||||
|
&& preferredClient != ClientType.IPADOS;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -53,9 +85,9 @@ public class SpoofVideoStreamsPatch {
|
|||||||
String path = playerRequestUri.getPath();
|
String path = playerRequestUri.getPath();
|
||||||
|
|
||||||
if (path != null && path.contains("get_watch")) {
|
if (path != null && path.contains("get_watch")) {
|
||||||
Logger.printDebug(() -> "Blocking 'get_watch' by returning unreachable uri");
|
Logger.printDebug(() -> "Blocking 'get_watch' by returning internet connection check uri");
|
||||||
|
|
||||||
return UNREACHABLE_HOST_URI;
|
return INTERNET_CONNECTION_CHECK_URI;
|
||||||
}
|
}
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.printException(() -> "blockGetWatchRequest failure", ex);
|
Logger.printException(() -> "blockGetWatchRequest failure", ex);
|
||||||
@@ -77,9 +109,9 @@ public class SpoofVideoStreamsPatch {
|
|||||||
String path = originalUri.getPath();
|
String path = originalUri.getPath();
|
||||||
|
|
||||||
if (path != null && path.contains("initplayback")) {
|
if (path != null && path.contains("initplayback")) {
|
||||||
Logger.printDebug(() -> "Blocking 'initplayback' by clearing query");
|
Logger.printDebug(() -> "Blocking 'initplayback' by returning internet connection check uri");
|
||||||
|
|
||||||
return originalUri.buildUpon().clearQuery().build().toString();
|
return INTERNET_CONNECTION_CHECK_URI_STRING;
|
||||||
}
|
}
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.printException(() -> "blockInitPlaybackRequest failure", ex);
|
Logger.printException(() -> "blockInitPlaybackRequest failure", ex);
|
||||||
@@ -252,16 +284,7 @@ public class SpoofVideoStreamsPatch {
|
|||||||
public static final class AudioStreamLanguageOverrideAvailability implements Setting.Availability {
|
public static final class AudioStreamLanguageOverrideAvailability implements Setting.Availability {
|
||||||
@Override
|
@Override
|
||||||
public boolean isAvailable() {
|
public boolean isAvailable() {
|
||||||
return BaseSettings.SPOOF_VIDEO_STREAMS.get()
|
return BaseSettings.SPOOF_VIDEO_STREAMS.get() && !preferredClient.useAuth;
|
||||||
&& BaseSettings.SPOOF_VIDEO_STREAMS_CLIENT_TYPE.get() == ClientType.ANDROID_VR_NO_AUTH;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static final class SpoofiOSAvailability implements Setting.Availability {
|
|
||||||
@Override
|
|
||||||
public boolean isAvailable() {
|
|
||||||
return BaseSettings.SPOOF_VIDEO_STREAMS.get()
|
|
||||||
&& BaseSettings.SPOOF_VIDEO_STREAMS_CLIENT_TYPE.get() == ClientType.IOS_UNPLUGGED;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
package app.revanced.extension.shared.spoof.requests;
|
package app.revanced.extension.shared.spoof.requests;
|
||||||
|
|
||||||
|
import static app.revanced.extension.shared.spoof.ClientType.ANDROID_VR_1_43_32;
|
||||||
|
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
|
|
||||||
@@ -10,8 +12,10 @@ import java.util.Locale;
|
|||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.requests.Requester;
|
import app.revanced.extension.shared.requests.Requester;
|
||||||
import app.revanced.extension.shared.requests.Route;
|
import app.revanced.extension.shared.requests.Route;
|
||||||
|
import app.revanced.extension.shared.settings.AppLanguage;
|
||||||
import app.revanced.extension.shared.settings.BaseSettings;
|
import app.revanced.extension.shared.settings.BaseSettings;
|
||||||
import app.revanced.extension.shared.spoof.ClientType;
|
import app.revanced.extension.shared.spoof.ClientType;
|
||||||
|
import app.revanced.extension.shared.spoof.SpoofVideoStreamsPatch;
|
||||||
|
|
||||||
final class PlayerRoutes {
|
final class PlayerRoutes {
|
||||||
static final Route.CompiledRoute GET_STREAMING_DATA = new Route(
|
static final Route.CompiledRoute GET_STREAMING_DATA = new Route(
|
||||||
@@ -37,14 +41,16 @@ final class PlayerRoutes {
|
|||||||
try {
|
try {
|
||||||
JSONObject context = new JSONObject();
|
JSONObject context = new JSONObject();
|
||||||
|
|
||||||
// Can override default language only if no login is used.
|
AppLanguage language = SpoofVideoStreamsPatch.getLanguageOverride();
|
||||||
// Could use preferred audio for all clients that do not login,
|
if (language == null || clientType == ANDROID_VR_1_43_32) {
|
||||||
// but if this is a fall over client it will set the language even though
|
// Force original audio has not overrode the language.
|
||||||
// the audio language is not selectable in the UI.
|
// Or if YT has fallen over to the last unauthenticated client (VR 1.43), then
|
||||||
ClientType userSelectedClient = BaseSettings.SPOOF_VIDEO_STREAMS_CLIENT_TYPE.get();
|
// always use the app language because forcing an audio stream of specific languages
|
||||||
Locale streamLocale = userSelectedClient == ClientType.ANDROID_VR_NO_AUTH
|
// can sometimes fail so it's better to try and load something rather than nothing.
|
||||||
? BaseSettings.SPOOF_VIDEO_STREAMS_LANGUAGE.get().getLocale()
|
language = BaseSettings.SPOOF_VIDEO_STREAMS_LANGUAGE.get();
|
||||||
: Locale.getDefault();
|
}
|
||||||
|
//noinspection ExtractMethodRecommender
|
||||||
|
Locale streamLocale = language.getLocale();
|
||||||
|
|
||||||
JSONObject client = new JSONObject();
|
JSONObject client = new JSONObject();
|
||||||
client.put("deviceMake", clientType.deviceMake);
|
client.put("deviceMake", clientType.deviceMake);
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
package app.revanced.extension.shared.spoof.requests;
|
package app.revanced.extension.shared.spoof.requests;
|
||||||
|
|
||||||
|
import static app.revanced.extension.shared.ByteTrieSearch.convertStringsToBytes;
|
||||||
import static app.revanced.extension.shared.spoof.requests.PlayerRoutes.GET_STREAMING_DATA;
|
import static app.revanced.extension.shared.spoof.requests.PlayerRoutes.GET_STREAMING_DATA;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
@@ -13,12 +14,18 @@ import java.net.HttpURLConnection;
|
|||||||
import java.net.SocketTimeoutException;
|
import java.net.SocketTimeoutException;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
import java.util.*;
|
import java.util.Arrays;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Objects;
|
||||||
import java.util.concurrent.ExecutionException;
|
import java.util.concurrent.ExecutionException;
|
||||||
import java.util.concurrent.Future;
|
import java.util.concurrent.Future;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.concurrent.TimeoutException;
|
import java.util.concurrent.TimeoutException;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.ByteTrieSearch;
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.Utils;
|
import app.revanced.extension.shared.Utils;
|
||||||
import app.revanced.extension.shared.settings.BaseSettings;
|
import app.revanced.extension.shared.settings.BaseSettings;
|
||||||
@@ -35,21 +42,27 @@ import app.revanced.extension.shared.spoof.ClientType;
|
|||||||
*/
|
*/
|
||||||
public class StreamingDataRequest {
|
public class StreamingDataRequest {
|
||||||
|
|
||||||
private static final ClientType[] CLIENT_ORDER_TO_USE;
|
private static volatile ClientType[] clientOrderToUse = ClientType.values();
|
||||||
|
|
||||||
static {
|
public static void setClientOrderToUse(List<ClientType> availableClients, ClientType preferredClient) {
|
||||||
ClientType[] allClientTypes = ClientType.values();
|
Objects.requireNonNull(preferredClient);
|
||||||
ClientType preferredClient = BaseSettings.SPOOF_VIDEO_STREAMS_CLIENT_TYPE.get();
|
|
||||||
|
|
||||||
CLIENT_ORDER_TO_USE = new ClientType[allClientTypes.length];
|
int availableClientSize = availableClients.size();
|
||||||
CLIENT_ORDER_TO_USE[0] = preferredClient;
|
if (!availableClients.contains(preferredClient)) {
|
||||||
|
availableClientSize++;
|
||||||
|
}
|
||||||
|
|
||||||
|
clientOrderToUse = new ClientType[availableClientSize];
|
||||||
|
clientOrderToUse[0] = preferredClient;
|
||||||
|
|
||||||
int i = 1;
|
int i = 1;
|
||||||
for (ClientType c : allClientTypes) {
|
for (ClientType c : availableClients) {
|
||||||
if (c != preferredClient) {
|
if (c != preferredClient) {
|
||||||
CLIENT_ORDER_TO_USE[i++] = c;
|
clientOrderToUse[i++] = c;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Logger.printDebug(() -> "Available spoof clients: " + Arrays.toString(clientOrderToUse));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static final String AUTHORIZATION_HEADER = "Authorization";
|
private static final String AUTHORIZATION_HEADER = "Authorization";
|
||||||
@@ -87,6 +100,16 @@ public class StreamingDataRequest {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Strings found in the response if the video is a livestream.
|
||||||
|
*/
|
||||||
|
private static final ByteTrieSearch liveStreamBufferSearch = new ByteTrieSearch(
|
||||||
|
convertStringsToBytes(
|
||||||
|
"yt_live_broadcast",
|
||||||
|
"yt_premiere_broadcast"
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
private static volatile ClientType lastSpoofedClientType;
|
private static volatile ClientType lastSpoofedClientType;
|
||||||
|
|
||||||
public static String getLastSpoofedClientName() {
|
public static String getLastSpoofedClientName() {
|
||||||
@@ -154,7 +177,7 @@ public class StreamingDataRequest {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!authHeadersIncludes && clientType.requiresAuth) {
|
if (!authHeadersIncludes && clientType.useAuth) {
|
||||||
Logger.printDebug(() -> "Skipping client since user is not logged in: " + clientType
|
Logger.printDebug(() -> "Skipping client since user is not logged in: " + clientType
|
||||||
+ " videoId: " + videoId);
|
+ " videoId: " + videoId);
|
||||||
return null;
|
return null;
|
||||||
@@ -193,9 +216,9 @@ public class StreamingDataRequest {
|
|||||||
|
|
||||||
// Retry with different client if empty response body is received.
|
// Retry with different client if empty response body is received.
|
||||||
int i = 0;
|
int i = 0;
|
||||||
for (ClientType clientType : CLIENT_ORDER_TO_USE) {
|
for (ClientType clientType : clientOrderToUse) {
|
||||||
// Show an error if the last client type fails, or if debug is enabled then show for all attempts.
|
// Show an error if the last client type fails, or if debug is enabled then show for all attempts.
|
||||||
final boolean showErrorToast = (++i == CLIENT_ORDER_TO_USE.length) || debugEnabled;
|
final boolean showErrorToast = (++i == clientOrderToUse.length) || debugEnabled;
|
||||||
|
|
||||||
HttpURLConnection connection = send(clientType, videoId, playerHeaders, showErrorToast);
|
HttpURLConnection connection = send(clientType, videoId, playerHeaders, showErrorToast);
|
||||||
if (connection != null) {
|
if (connection != null) {
|
||||||
@@ -215,9 +238,13 @@ public class StreamingDataRequest {
|
|||||||
while ((bytesRead = inputStream.read(buffer)) >= 0) {
|
while ((bytesRead = inputStream.read(buffer)) >= 0) {
|
||||||
baos.write(buffer, 0, bytesRead);
|
baos.write(buffer, 0, bytesRead);
|
||||||
}
|
}
|
||||||
lastSpoofedClientType = clientType;
|
if (clientType == ClientType.ANDROID_CREATOR && liveStreamBufferSearch.matches(buffer)) {
|
||||||
|
Logger.printDebug(() -> "Skipping Android Studio as video is a livestream: " + videoId);
|
||||||
|
} else {
|
||||||
|
lastSpoofedClientType = clientType;
|
||||||
|
|
||||||
return ByteBuffer.wrap(baos.toByteArray());
|
return ByteBuffer.wrap(baos.toByteArray());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (IOException ex) {
|
} catch (IOException ex) {
|
||||||
|
|||||||
@@ -1,7 +1,14 @@
|
|||||||
|
plugins {
|
||||||
|
alias(libs.plugins.protobuf)
|
||||||
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
compileOnly(project(":extensions:shared:library"))
|
compileOnly(project(":extensions:shared:library"))
|
||||||
compileOnly(project(":extensions:spotify:stub"))
|
compileOnly(project(":extensions:spotify:stub"))
|
||||||
compileOnly(libs.annotation)
|
compileOnly(libs.annotation)
|
||||||
|
|
||||||
|
implementation(libs.nanohttpd)
|
||||||
|
implementation(libs.protobuf.javalite)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
@@ -14,3 +21,19 @@ android {
|
|||||||
targetCompatibility = JavaVersion.VERSION_1_8
|
targetCompatibility = JavaVersion.VERSION_1_8
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protobuf {
|
||||||
|
protoc {
|
||||||
|
artifact = libs.protobuf.protoc.get().toString()
|
||||||
|
}
|
||||||
|
|
||||||
|
generateProtoTasks {
|
||||||
|
all().forEach { task ->
|
||||||
|
task.builtins {
|
||||||
|
create("java") {
|
||||||
|
option("lite")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,9 +1,11 @@
|
|||||||
package app.revanced.extension.spotify.layout.hide.createbutton;
|
package app.revanced.extension.spotify.layout.hide.createbutton;
|
||||||
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.spotify.shared.ComponentFilters.*;
|
import app.revanced.extension.spotify.shared.ComponentFilters.ComponentFilter;
|
||||||
|
import app.revanced.extension.spotify.shared.ComponentFilters.ResourceIdComponentFilter;
|
||||||
|
import app.revanced.extension.spotify.shared.ComponentFilters.StringComponentFilter;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
public final class HideCreateButtonPatch {
|
public final class HideCreateButtonPatch {
|
||||||
@@ -53,7 +55,9 @@ public final class HideCreateButtonPatch {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (Exception ex) {
|
} catch (Throwable ex) {
|
||||||
|
// Catch Throwable as calling toString can cause crashes with wrongfully generated code that throws
|
||||||
|
// NoSuchMethod errors.
|
||||||
Logger.printException(() -> "returnNullIfIsCreateButton failure", ex);
|
Logger.printException(() -> "returnNullIfIsCreateButton failure", ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,115 @@
|
|||||||
|
package app.revanced.extension.spotify.misc.fix;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import app.revanced.extension.spotify.misc.fix.clienttoken.data.v0.ClienttokenHttp.*;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.net.HttpURLConnection;
|
||||||
|
import java.net.URL;
|
||||||
|
|
||||||
|
import static app.revanced.extension.spotify.misc.fix.Constants.*;
|
||||||
|
|
||||||
|
class ClientTokenService {
|
||||||
|
private static final String IOS_CLIENT_ID = "58bd3c95768941ea9eb4350aaa033eb3";
|
||||||
|
private static final String IOS_USER_AGENT;
|
||||||
|
|
||||||
|
static {
|
||||||
|
String clientVersion = getClientVersion();
|
||||||
|
int commitHashIndex = clientVersion.lastIndexOf(".");
|
||||||
|
String version = clientVersion.substring(
|
||||||
|
clientVersion.indexOf("-") + 1,
|
||||||
|
clientVersion.lastIndexOf(".", commitHashIndex - 1)
|
||||||
|
);
|
||||||
|
|
||||||
|
IOS_USER_AGENT = "Spotify/" + version + " iOS/" + getSystemVersion() + " (" + getHardwareMachine() + ")";
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final ConnectivitySdkData.Builder IOS_CONNECTIVITY_SDK_DATA =
|
||||||
|
ConnectivitySdkData.newBuilder()
|
||||||
|
.setPlatformSpecificData(PlatformSpecificData.newBuilder()
|
||||||
|
.setIos(NativeIOSData.newBuilder()
|
||||||
|
.setHwMachine(getHardwareMachine())
|
||||||
|
.setSystemVersion(getSystemVersion())
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
private static final ClientDataRequest.Builder IOS_CLIENT_DATA_REQUEST =
|
||||||
|
ClientDataRequest.newBuilder()
|
||||||
|
.setClientVersion(getClientVersion())
|
||||||
|
.setClientId(IOS_CLIENT_ID);
|
||||||
|
|
||||||
|
private static final ClientTokenRequest.Builder IOS_CLIENT_TOKEN_REQUEST =
|
||||||
|
ClientTokenRequest.newBuilder()
|
||||||
|
.setRequestType(ClientTokenRequestType.REQUEST_CLIENT_DATA_REQUEST);
|
||||||
|
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
static ClientTokenRequest newIOSClientTokenRequest(String deviceId) {
|
||||||
|
Logger.printInfo(() -> "Creating new iOS client token request with device ID: " + deviceId);
|
||||||
|
|
||||||
|
return IOS_CLIENT_TOKEN_REQUEST
|
||||||
|
.setClientData(IOS_CLIENT_DATA_REQUEST
|
||||||
|
.setConnectivitySdkData(IOS_CONNECTIVITY_SDK_DATA
|
||||||
|
.setDeviceId(deviceId)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
static ClientTokenResponse getClientTokenResponse(@NonNull ClientTokenRequest request) {
|
||||||
|
if (request.getRequestType() == ClientTokenRequestType.REQUEST_CLIENT_DATA_REQUEST) {
|
||||||
|
Logger.printInfo(() -> "Requesting iOS client token");
|
||||||
|
String deviceId = request.getClientData().getConnectivitySdkData().getDeviceId();
|
||||||
|
request = newIOSClientTokenRequest(deviceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
ClientTokenResponse response;
|
||||||
|
try {
|
||||||
|
response = requestClientToken(request);
|
||||||
|
} catch (IOException ex) {
|
||||||
|
Logger.printException(() -> "Failed to handle request", ex);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
private static ClientTokenResponse requestClientToken(@NonNull ClientTokenRequest request) throws IOException {
|
||||||
|
HttpURLConnection urlConnection = (HttpURLConnection) new URL(CLIENT_TOKEN_API_URL).openConnection();
|
||||||
|
urlConnection.setRequestMethod("POST");
|
||||||
|
urlConnection.setDoOutput(true);
|
||||||
|
urlConnection.setRequestProperty("Content-Type", "application/x-protobuf");
|
||||||
|
urlConnection.setRequestProperty("Accept", "application/x-protobuf");
|
||||||
|
urlConnection.setRequestProperty("User-Agent", IOS_USER_AGENT);
|
||||||
|
|
||||||
|
byte[] requestArray = request.toByteArray();
|
||||||
|
urlConnection.setFixedLengthStreamingMode(requestArray.length);
|
||||||
|
urlConnection.getOutputStream().write(requestArray);
|
||||||
|
|
||||||
|
try (InputStream inputStream = urlConnection.getInputStream()) {
|
||||||
|
return ClientTokenResponse.parseFrom(inputStream);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
static ClientTokenResponse serveClientTokenRequest(@NonNull InputStream inputStream) {
|
||||||
|
ClientTokenRequest request;
|
||||||
|
try {
|
||||||
|
request = ClientTokenRequest.parseFrom(inputStream);
|
||||||
|
} catch (IOException ex) {
|
||||||
|
Logger.printException(() -> "Failed to parse request from input stream", ex);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
Logger.printInfo(() -> "Request of type: " + request.getRequestType());
|
||||||
|
|
||||||
|
ClientTokenResponse response = getClientTokenResponse(request);
|
||||||
|
if (response != null) Logger.printInfo(() -> "Response of type: " + response.getResponseType());
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
package app.revanced.extension.spotify.misc.fix;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
|
class Constants {
|
||||||
|
static final String CLIENT_TOKEN_API_PATH = "/v1/clienttoken";
|
||||||
|
static final String CLIENT_TOKEN_API_URL = "https://clienttoken.spotify.com" + CLIENT_TOKEN_API_PATH;
|
||||||
|
|
||||||
|
// Modified by a patch. Do not touch.
|
||||||
|
@NonNull
|
||||||
|
static String getClientVersion() {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Modified by a patch. Do not touch.
|
||||||
|
@NonNull
|
||||||
|
static String getSystemVersion() {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Modified by a patch. Do not touch.
|
||||||
|
@NonNull
|
||||||
|
static String getHardwareMachine() {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,94 @@
|
|||||||
|
package app.revanced.extension.spotify.misc.fix;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import app.revanced.extension.spotify.misc.fix.clienttoken.data.v0.ClienttokenHttp.ClientTokenResponse;
|
||||||
|
import com.google.protobuf.MessageLite;
|
||||||
|
import fi.iki.elonen.NanoHTTPD;
|
||||||
|
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.io.FilterInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
|
import static app.revanced.extension.spotify.misc.fix.ClientTokenService.serveClientTokenRequest;
|
||||||
|
import static app.revanced.extension.spotify.misc.fix.Constants.CLIENT_TOKEN_API_PATH;
|
||||||
|
import static fi.iki.elonen.NanoHTTPD.Response.Status.INTERNAL_ERROR;
|
||||||
|
|
||||||
|
class RequestListener extends NanoHTTPD {
|
||||||
|
RequestListener(int port) {
|
||||||
|
super(port);
|
||||||
|
|
||||||
|
try {
|
||||||
|
start();
|
||||||
|
} catch (IOException ex) {
|
||||||
|
Logger.printException(() -> "Failed to start request listener on port " + port, ex);
|
||||||
|
throw new RuntimeException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public Response serve(@NonNull IHTTPSession session) {
|
||||||
|
String uri = session.getUri();
|
||||||
|
if (!uri.equals(CLIENT_TOKEN_API_PATH)) return INTERNAL_ERROR_RESPONSE;
|
||||||
|
|
||||||
|
Logger.printInfo(() -> "Serving request for URI: " + uri);
|
||||||
|
|
||||||
|
ClientTokenResponse response = serveClientTokenRequest(getInputStream(session));
|
||||||
|
if (response != null) return newResponse(Response.Status.OK, response);
|
||||||
|
|
||||||
|
Logger.printException(() -> "Failed to serve client token request");
|
||||||
|
return INTERNAL_ERROR_RESPONSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
private static InputStream newLimitedInputStream(InputStream inputStream, long contentLength) {
|
||||||
|
return new FilterInputStream(inputStream) {
|
||||||
|
private long remaining = contentLength;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int read() throws IOException {
|
||||||
|
if (remaining <= 0) return -1;
|
||||||
|
int result = super.read();
|
||||||
|
if (result != -1) remaining--;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int read(byte[] b, int off, int len) throws IOException {
|
||||||
|
if (remaining <= 0) return -1;
|
||||||
|
len = (int) Math.min(len, remaining);
|
||||||
|
int result = super.read(b, off, len);
|
||||||
|
if (result != -1) remaining -= result;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
private static InputStream getInputStream(@NonNull IHTTPSession session) {
|
||||||
|
long requestContentLength = Long.parseLong(Objects.requireNonNull(session.getHeaders().get("content-length")));
|
||||||
|
return newLimitedInputStream(session.getInputStream(), requestContentLength);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final Response INTERNAL_ERROR_RESPONSE = newResponse(INTERNAL_ERROR);
|
||||||
|
|
||||||
|
@SuppressWarnings("SameParameterValue")
|
||||||
|
@NonNull
|
||||||
|
private static Response newResponse(Response.Status status) {
|
||||||
|
return newResponse(status, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
private static Response newResponse(Response.IStatus status, MessageLite messageLite) {
|
||||||
|
if (messageLite == null) {
|
||||||
|
return newFixedLengthResponse(status, "application/x-protobuf", null);
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] messageBytes = messageLite.toByteArray();
|
||||||
|
InputStream stream = new ByteArrayInputStream(messageBytes);
|
||||||
|
return newFixedLengthResponse(status, "application/x-protobuf", stream, messageBytes.length);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
package app.revanced.extension.spotify.misc.fix;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class SpoofClientPatch {
|
||||||
|
private static RequestListener listener;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point. Launch requests listener server.
|
||||||
|
*/
|
||||||
|
public synchronized static void launchListener(int port) {
|
||||||
|
if (listener != null) {
|
||||||
|
Logger.printInfo(() -> "Listener already running on port " + port);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
Logger.printInfo(() -> "Launching listener on port " + port);
|
||||||
|
listener = new RequestListener(port);
|
||||||
|
} catch (Exception ex) {
|
||||||
|
Logger.printException(() -> "launchListener failure", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
syntax = "proto3";
|
||||||
|
|
||||||
|
package spotify.clienttoken.data.v0;
|
||||||
|
|
||||||
|
option optimize_for = LITE_RUNTIME;
|
||||||
|
option java_package = "app.revanced.extension.spotify.misc.fix.clienttoken.data.v0";
|
||||||
|
|
||||||
|
message ClientTokenRequest {
|
||||||
|
ClientTokenRequestType request_type = 1;
|
||||||
|
|
||||||
|
oneof request {
|
||||||
|
ClientDataRequest client_data = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum ClientTokenRequestType {
|
||||||
|
REQUEST_UNKNOWN = 0;
|
||||||
|
REQUEST_CLIENT_DATA_REQUEST = 1;
|
||||||
|
REQUEST_CHALLENGE_ANSWERS_REQUEST = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
message ClientDataRequest {
|
||||||
|
string client_version = 1;
|
||||||
|
string client_id = 2;
|
||||||
|
|
||||||
|
oneof data {
|
||||||
|
ConnectivitySdkData connectivity_sdk_data = 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
message ConnectivitySdkData {
|
||||||
|
PlatformSpecificData platform_specific_data = 1;
|
||||||
|
string device_id = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
message PlatformSpecificData {
|
||||||
|
oneof data {
|
||||||
|
NativeIOSData ios = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
message NativeIOSData {
|
||||||
|
int32 user_interface_idiom = 1;
|
||||||
|
bool target_iphone_simulator = 2;
|
||||||
|
string hw_machine = 3;
|
||||||
|
string system_version = 4;
|
||||||
|
string simulator_model_identifier = 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
message ClientTokenResponse {
|
||||||
|
ClientTokenResponseType response_type = 1;
|
||||||
|
|
||||||
|
oneof response {
|
||||||
|
GrantedTokenResponse granted_token = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum ClientTokenResponseType {
|
||||||
|
RESPONSE_UNKNOWN = 0;
|
||||||
|
RESPONSE_GRANTED_TOKEN_RESPONSE = 1;
|
||||||
|
RESPONSE_CHALLENGES_RESPONSE = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
message GrantedTokenResponse {
|
||||||
|
string token = 1;
|
||||||
|
int32 expires_after_seconds = 2;
|
||||||
|
int32 refresh_after_seconds = 3;
|
||||||
|
repeated TokenDomain domains = 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
message TokenDomain {
|
||||||
|
string domain = 1;
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
@@ -0,0 +1,5 @@
|
|||||||
|
package app.revanced;
|
||||||
|
|
||||||
|
public interface ContextMenuItemPlaceholder {
|
||||||
|
Object getViewModel();
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
package com.spotify.browsita.v1.resolved;
|
||||||
|
|
||||||
|
public final class Section {
|
||||||
|
public static final int BRAND_ADS_FIELD_NUMBER = 6;
|
||||||
|
public int sectionTypeCase_;
|
||||||
|
}
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
package com.spotify.useraccount.v1;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Used for target 8.6.98.900. Class is still present in newer app targets.
|
|
||||||
*/
|
|
||||||
public class AccountAttribute {
|
|
||||||
public Object value_;
|
|
||||||
}
|
|
||||||
@@ -2,4 +2,5 @@ dependencies {
|
|||||||
compileOnly(project(":extensions:shared:library"))
|
compileOnly(project(":extensions:shared:library"))
|
||||||
compileOnly(project(":extensions:syncforreddit:stub"))
|
compileOnly(project(":extensions:syncforreddit:stub"))
|
||||||
compileOnly(libs.annotation)
|
compileOnly(libs.annotation)
|
||||||
|
compileOnly(libs.okhttp)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
package app.revanced.extension.syncforreddit;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.fixes.redgifs.BaseFixRedgifsApiPatch;
|
||||||
|
import okhttp3.OkHttpClient;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @noinspection unused
|
||||||
|
*/
|
||||||
|
public class FixRedgifsApiPatch extends BaseFixRedgifsApiPatch {
|
||||||
|
static {
|
||||||
|
INSTANCE = new FixRedgifsApiPatch();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getDefaultUserAgent() {
|
||||||
|
// To be filled in by patch
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
public static OkHttpClient install(OkHttpClient.Builder builder) {
|
||||||
|
return builder.addInterceptor(INSTANCE).build();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
android.namespace = "app.revanced.extension"
|
android.namespace = "app.revanced.extension"
|
||||||
|
|
||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
plugins {
|
plugins {
|
||||||
id(libs.plugins.android.library.get().pluginId)
|
alias(libs.plugins.android.library)
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
|
|||||||
@@ -1,16 +1,20 @@
|
|||||||
package app.revanced.extension.youtube
|
package app.revanced.extension.youtube
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.Logger
|
||||||
|
import java.util.Collections
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* generic event provider class
|
* generic event provider class
|
||||||
*/
|
*/
|
||||||
class Event<T> {
|
class Event<T> {
|
||||||
private val eventListeners = mutableSetOf<(T) -> Unit>()
|
private val eventListeners = Collections.synchronizedSet(mutableSetOf<(T) -> Unit>())
|
||||||
|
|
||||||
operator fun plusAssign(observer: (T) -> Unit) {
|
operator fun plusAssign(observer: (T) -> Unit) {
|
||||||
addObserver(observer)
|
addObserver(observer)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun addObserver(observer: (T) -> Unit) {
|
fun addObserver(observer: (T) -> Unit) {
|
||||||
|
Logger.printDebug { "Adding observer: $observer" }
|
||||||
eventListeners.add(observer)
|
eventListeners.add(observer)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -23,7 +27,8 @@ class Event<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
operator fun invoke(value: T) {
|
operator fun invoke(value: T) {
|
||||||
for (observer in eventListeners)
|
for (observer in eventListeners) {
|
||||||
observer.invoke(value)
|
observer.invoke(value)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,101 @@
|
|||||||
|
package app.revanced.extension.youtube.patches;
|
||||||
|
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
|
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
|
import app.revanced.extension.shared.Logger;
|
||||||
|
import app.revanced.extension.shared.Utils;
|
||||||
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class ChangeHeaderPatch {
|
||||||
|
|
||||||
|
public enum HeaderLogo {
|
||||||
|
DEFAULT(null, null),
|
||||||
|
REGULAR("ytWordmarkHeader", "yt_ringo2_wordmark_header"),
|
||||||
|
PREMIUM("ytPremiumWordmarkHeader", "yt_ringo2_premium_wordmark_header"),
|
||||||
|
REVANCED("revanced_header_logo", "revanced_header_logo"),
|
||||||
|
REVANCED_MINIMAL("revanced_header_logo_minimal", "revanced_header_logo_minimal"),
|
||||||
|
CUSTOM("custom_header", "custom_header");
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
private final String attributeName;
|
||||||
|
@Nullable
|
||||||
|
private final String drawableName;
|
||||||
|
|
||||||
|
HeaderLogo(@Nullable String attributeName, @Nullable String drawableName) {
|
||||||
|
this.attributeName = attributeName;
|
||||||
|
this.drawableName = drawableName;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return The attribute id of this header logo, or NULL if the logo should not be replaced.
|
||||||
|
*/
|
||||||
|
@Nullable
|
||||||
|
private Integer getAttributeId() {
|
||||||
|
if (attributeName == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final int identifier = Utils.getResourceIdentifier(attributeName, "attr");
|
||||||
|
if (identifier == 0) {
|
||||||
|
// Identifier is zero if custom header setting was included in imported settings
|
||||||
|
// and a custom image was not included during patching.
|
||||||
|
Logger.printDebug(() -> "Could not find attribute: " + drawableName);
|
||||||
|
Settings.HEADER_LOGO.resetToDefault();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return identifier;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public Drawable getDrawable() {
|
||||||
|
if (drawableName == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
String drawableFullName = drawableName + (Utils.isDarkModeEnabled()
|
||||||
|
? "_dark"
|
||||||
|
: "_light");
|
||||||
|
|
||||||
|
final int identifier = Utils.getResourceIdentifier(drawableFullName, "drawable");
|
||||||
|
if (identifier == 0) {
|
||||||
|
Logger.printDebug(() -> "Could not find drawable: " + drawableFullName);
|
||||||
|
Settings.HEADER_LOGO.resetToDefault();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return Utils.getContext().getDrawable(identifier);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*/
|
||||||
|
public static int getHeaderAttributeId(int original) {
|
||||||
|
return Objects.requireNonNullElse(Settings.HEADER_LOGO.get().getAttributeId(), original);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Drawable getDrawable(Drawable original) {
|
||||||
|
Drawable logo = Settings.HEADER_LOGO.get().getDrawable();
|
||||||
|
if (logo != null) {
|
||||||
|
return logo;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: If 'Hide Doodles' is enabled, this will force the regular logo regardless
|
||||||
|
// what account the user has. This can be improved the next time a Doodle is
|
||||||
|
// active and the attribute id is passed to this method so the correct
|
||||||
|
// regular/premium logo is returned.
|
||||||
|
logo = HeaderLogo.REGULAR.getDrawable();
|
||||||
|
if (logo != null) {
|
||||||
|
return logo;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Should never happen.
|
||||||
|
Logger.printException(() -> "Could not find regular header logo resource");
|
||||||
|
return original;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
package app.revanced.extension.youtube.patches;
|
||||||
|
|
||||||
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public final class DisableDoubleTapActionsPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*
|
||||||
|
* @return If "should skip to chapter start" flag is set.
|
||||||
|
*/
|
||||||
|
public static boolean disableDoubleTapChapters(boolean original) {
|
||||||
|
return original && !Settings.DISABLE_CHAPTER_SKIP_DOUBLE_TAP.get();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,7 @@
|
|||||||
package app.revanced.extension.youtube.patches;
|
package app.revanced.extension.youtube.patches;
|
||||||
|
|
||||||
|
import android.view.Display;
|
||||||
|
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
@@ -8,8 +10,10 @@ public class DisableHdrPatch {
|
|||||||
/**
|
/**
|
||||||
* Injection point.
|
* Injection point.
|
||||||
*/
|
*/
|
||||||
public static boolean disableHDRVideo() {
|
public static int[] disableHdrVideo(Display.HdrCapabilities capabilities) {
|
||||||
return !Settings.DISABLE_HDR_VIDEO.get();
|
return Settings.DISABLE_HDR_VIDEO.get()
|
||||||
|
? new int[0]
|
||||||
|
: capabilities.getSupportedHdrTypes();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,14 @@
|
|||||||
|
package app.revanced.extension.youtube.patches;
|
||||||
|
|
||||||
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class DisableSignInToTvPopupPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*/
|
||||||
|
public static boolean disableSignInToTvPopup() {
|
||||||
|
return Settings.DISABLE_SIGNIN_TO_TV_POPUP.get();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,17 +1,15 @@
|
|||||||
package app.revanced.extension.youtube.patches;
|
package app.revanced.extension.youtube.patches;
|
||||||
|
|
||||||
|
import static app.revanced.extension.youtube.settings.preference.ExternalDownloaderPreference.showDialogIfAppIsNotInstalled;
|
||||||
|
|
||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.pm.PackageManager;
|
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
|
||||||
|
|
||||||
import java.lang.ref.WeakReference;
|
import java.lang.ref.WeakReference;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.StringRef;
|
|
||||||
import app.revanced.extension.shared.Utils;
|
import app.revanced.extension.shared.Utils;
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
@@ -36,7 +34,7 @@ public final class DownloadsPatch {
|
|||||||
*
|
*
|
||||||
* Appears to always be called from the main thread.
|
* Appears to always be called from the main thread.
|
||||||
*/
|
*/
|
||||||
public static boolean inAppDownloadButtonOnClick(@NonNull String videoId) {
|
public static boolean inAppDownloadButtonOnClick(String videoId) {
|
||||||
try {
|
try {
|
||||||
if (!Settings.EXTERNAL_DOWNLOADER_ACTION_BUTTON.get()) {
|
if (!Settings.EXTERNAL_DOWNLOADER_ACTION_BUTTON.get()) {
|
||||||
return false;
|
return false;
|
||||||
@@ -48,6 +46,9 @@ public final class DownloadsPatch {
|
|||||||
boolean isActivityContext = true;
|
boolean isActivityContext = true;
|
||||||
if (context == null) {
|
if (context == null) {
|
||||||
// Utils context is the application context, and not an activity context.
|
// Utils context is the application context, and not an activity context.
|
||||||
|
//
|
||||||
|
// Edit: This check may no longer be needed since YT can now
|
||||||
|
// only be launched from the main Activity (embedded usage in other apps no longer works).
|
||||||
context = Utils.getContext();
|
context = Utils.getContext();
|
||||||
isActivityContext = false;
|
isActivityContext = false;
|
||||||
}
|
}
|
||||||
@@ -64,8 +65,7 @@ public final class DownloadsPatch {
|
|||||||
* @param isActivityContext If the context parameter is for an Activity. If this is false, then
|
* @param isActivityContext If the context parameter is for an Activity. If this is false, then
|
||||||
* the downloader is opened as a new task (which forces YT to minimize).
|
* the downloader is opened as a new task (which forces YT to minimize).
|
||||||
*/
|
*/
|
||||||
public static void launchExternalDownloader(@NonNull String videoId,
|
public static void launchExternalDownloader(String videoId, Context context, boolean isActivityContext) {
|
||||||
@NonNull Context context, boolean isActivityContext) {
|
|
||||||
try {
|
try {
|
||||||
Objects.requireNonNull(videoId);
|
Objects.requireNonNull(videoId);
|
||||||
Logger.printDebug(() -> "Launching external downloader with context: " + context);
|
Logger.printDebug(() -> "Launching external downloader with context: " + context);
|
||||||
@@ -73,16 +73,8 @@ public final class DownloadsPatch {
|
|||||||
// Trim string to avoid any accidental whitespace.
|
// Trim string to avoid any accidental whitespace.
|
||||||
var downloaderPackageName = Settings.EXTERNAL_DOWNLOADER_PACKAGE_NAME.get().trim();
|
var downloaderPackageName = Settings.EXTERNAL_DOWNLOADER_PACKAGE_NAME.get().trim();
|
||||||
|
|
||||||
boolean packageEnabled = false;
|
// If the package is not installed, show a dialog.
|
||||||
try {
|
if (showDialogIfAppIsNotInstalled(context, downloaderPackageName)) {
|
||||||
packageEnabled = context.getPackageManager().getApplicationInfo(downloaderPackageName, 0).enabled;
|
|
||||||
} catch (PackageManager.NameNotFoundException error) {
|
|
||||||
Logger.printDebug(() -> "External downloader could not be found: " + error);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the package is not installed, show the toast
|
|
||||||
if (!packageEnabled) {
|
|
||||||
Utils.showToastLong(StringRef.str("revanced_external_downloader_not_installed_warning", downloaderPackageName));
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
package app.revanced.extension.youtube.patches;
|
package app.revanced.extension.youtube.patches;
|
||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.settings.Setting;
|
import app.revanced.extension.shared.settings.AppLanguage;
|
||||||
import app.revanced.extension.shared.spoof.SpoofVideoStreamsPatch;
|
import app.revanced.extension.shared.spoof.SpoofVideoStreamsPatch;
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
@@ -11,19 +11,34 @@ public class ForceOriginalAudioPatch {
|
|||||||
private static final String DEFAULT_AUDIO_TRACKS_SUFFIX = ".4";
|
private static final String DEFAULT_AUDIO_TRACKS_SUFFIX = ".4";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If the conditions to use this patch were present when the app launched.
|
* Injection point.
|
||||||
*/
|
*/
|
||||||
public static boolean PATCH_AVAILABLE = SpoofVideoStreamsPatch.notSpoofingToAndroid();
|
public static void setPreferredLanguage() {
|
||||||
|
if (Settings.FORCE_ORIGINAL_AUDIO.get()
|
||||||
public static final class ForceOriginalAudioAvailability implements Setting.Availability {
|
&& SpoofVideoStreamsPatch.spoofingToClientWithNoMultiAudioStreams()) {
|
||||||
@Override
|
// If client spoofing does not use authentication and lacks multi-audio streams,
|
||||||
public boolean isAvailable() {
|
// then can use any language code for the request and if that requested language is
|
||||||
// Check conditions of launch and now. Otherwise if spoofing is changed
|
// not available YT uses the original audio language. Authenticated requests ignore
|
||||||
// without a restart the setting will show as available when it's not.
|
// the language code and always use the account language. Use a language that is
|
||||||
return PATCH_AVAILABLE && SpoofVideoStreamsPatch.notSpoofingToAndroid();
|
// not auto-dubbed by YouTube: https://support.google.com/youtube/answer/15569972
|
||||||
|
// but the language is also supported natively by the Meta Quest device that
|
||||||
|
// Android VR is spoofing.
|
||||||
|
AppLanguage override = AppLanguage.SV;
|
||||||
|
Logger.printDebug(() -> "Setting language override: " + override);
|
||||||
|
SpoofVideoStreamsPatch.setLanguageOverride(override);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*/
|
||||||
|
public static boolean ignoreDefaultAudioStream(boolean original) {
|
||||||
|
if (Settings.FORCE_ORIGINAL_AUDIO.get()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return original;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Injection point.
|
* Injection point.
|
||||||
*/
|
*/
|
||||||
@@ -50,7 +65,6 @@ public class ForceOriginalAudioPatch {
|
|||||||
return isOriginal;
|
return isOriginal;
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.printException(() -> "isDefaultAudioStream failure", ex);
|
Logger.printException(() -> "isDefaultAudioStream failure", ex);
|
||||||
|
|
||||||
return isDefault;
|
return isDefault;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,6 +8,6 @@ public final class HideRelatedVideoOverlayPatch {
|
|||||||
* Injection point.
|
* Injection point.
|
||||||
*/
|
*/
|
||||||
public static boolean hideRelatedVideoOverlay() {
|
public static boolean hideRelatedVideoOverlay() {
|
||||||
return Settings.HIDE_RELATED_VIDEO_OVERLAY.get();
|
return Settings.HIDE_RELATED_VIDEOS_OVERLAY.get();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -57,11 +57,4 @@ public class PlayerControlsPatch {
|
|||||||
private static void fullscreenButtonVisibilityChanged(boolean isVisible) {
|
private static void fullscreenButtonVisibilityChanged(boolean isVisible) {
|
||||||
// Code added during patching.
|
// Code added during patching.
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Injection point.
|
|
||||||
*/
|
|
||||||
public static String getPlayerTopControlsLayoutResourceName(String original) {
|
|
||||||
return "default";
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,18 @@
|
|||||||
|
package app.revanced.extension.youtube.patches;
|
||||||
|
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
|
import app.revanced.extension.youtube.shared.PlayerControlsVisibility;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public class PlayerControlsVisibilityHookPatch {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*/
|
||||||
|
public static void setPlayerControlsVisibility(@Nullable Enum<?> youTubePlayerControlsVisibility) {
|
||||||
|
if (youTubePlayerControlsVisibility == null) return;
|
||||||
|
|
||||||
|
PlayerControlsVisibility.setFromString(youTubePlayerControlsVisibility.name());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -16,7 +16,7 @@ import java.util.Objects;
|
|||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.Utils;
|
import app.revanced.extension.shared.Utils;
|
||||||
import app.revanced.extension.youtube.patches.components.ReturnYouTubeDislikeFilterPatch;
|
import app.revanced.extension.youtube.patches.components.ReturnYouTubeDislikeFilter;
|
||||||
import app.revanced.extension.youtube.returnyoutubedislike.ReturnYouTubeDislike;
|
import app.revanced.extension.youtube.returnyoutubedislike.ReturnYouTubeDislike;
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
import app.revanced.extension.youtube.shared.PlayerType;
|
import app.revanced.extension.youtube.shared.PlayerType;
|
||||||
@@ -55,7 +55,7 @@ public class ReturnYouTubeDislikePatch {
|
|||||||
private static volatile ReturnYouTubeDislike lastLithoShortsVideoData;
|
private static volatile ReturnYouTubeDislike lastLithoShortsVideoData;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Because litho Shorts spans are created offscreen after {@link ReturnYouTubeDislikeFilterPatch}
|
* Because litho Shorts spans are created offscreen after {@link ReturnYouTubeDislikeFilter}
|
||||||
* detects the video ids, but the current Short can arbitrarily reload the same span,
|
* detects the video ids, but the current Short can arbitrarily reload the same span,
|
||||||
* then use the {@link #lastLithoShortsVideoData} if this value is greater than zero.
|
* then use the {@link #lastLithoShortsVideoData} if this value is greater than zero.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -1,12 +1,18 @@
|
|||||||
package app.revanced.extension.youtube.patches;
|
package app.revanced.extension.youtube.patches;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
|
import com.google.android.libraries.youtube.innertube.model.media.VideoQuality;
|
||||||
|
|
||||||
import java.lang.ref.WeakReference;
|
import java.lang.ref.WeakReference;
|
||||||
|
import java.util.Arrays;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.Utils;
|
import app.revanced.extension.shared.Utils;
|
||||||
|
import app.revanced.extension.youtube.Event;
|
||||||
|
import app.revanced.extension.youtube.shared.ShortsPlayerState;
|
||||||
import app.revanced.extension.youtube.shared.VideoState;
|
import app.revanced.extension.youtube.shared.VideoState;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -16,11 +22,30 @@ import app.revanced.extension.youtube.shared.VideoState;
|
|||||||
public final class VideoInformation {
|
public final class VideoInformation {
|
||||||
|
|
||||||
public interface PlaybackController {
|
public interface PlaybackController {
|
||||||
// Methods are added to YT classes during patching.
|
// Methods are added during patching.
|
||||||
boolean seekTo(long videoTime);
|
boolean patch_seekTo(long videoTime);
|
||||||
void seekToRelative(long videoTimeOffset);
|
void patch_seekToRelative(long videoTimeOffset);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface to use obfuscated methods.
|
||||||
|
*/
|
||||||
|
public interface VideoQualityMenuInterface {
|
||||||
|
// Method is added during patching.
|
||||||
|
void patch_setQuality(VideoQuality quality);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Video resolution of the automatic quality option..
|
||||||
|
*/
|
||||||
|
public static final int AUTOMATIC_VIDEO_QUALITY_VALUE = -2;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Video quality names are the same text for all languages.
|
||||||
|
* Premium can be "1080p Premium" or "1080p60 Premium"
|
||||||
|
*/
|
||||||
|
public static final String VIDEO_QUALITY_PREMIUM_NAME = "Premium";
|
||||||
|
|
||||||
private static final float DEFAULT_YOUTUBE_PLAYBACK_SPEED = 1.0f;
|
private static final float DEFAULT_YOUTUBE_PLAYBACK_SPEED = 1.0f;
|
||||||
/**
|
/**
|
||||||
* Prefix present in all Short player parameters signature.
|
* Prefix present in all Short player parameters signature.
|
||||||
@@ -30,12 +55,10 @@ public final class VideoInformation {
|
|||||||
private static WeakReference<PlaybackController> playerControllerRef = new WeakReference<>(null);
|
private static WeakReference<PlaybackController> playerControllerRef = new WeakReference<>(null);
|
||||||
private static WeakReference<PlaybackController> mdxPlayerDirectorRef = new WeakReference<>(null);
|
private static WeakReference<PlaybackController> mdxPlayerDirectorRef = new WeakReference<>(null);
|
||||||
|
|
||||||
@NonNull
|
|
||||||
private static String videoId = "";
|
private static String videoId = "";
|
||||||
private static long videoLength = 0;
|
private static long videoLength = 0;
|
||||||
private static long videoTime = -1;
|
private static long videoTime = -1;
|
||||||
|
|
||||||
@NonNull
|
|
||||||
private static volatile String playerResponseVideoId = "";
|
private static volatile String playerResponseVideoId = "";
|
||||||
private static volatile boolean playerResponseVideoIdIsShort;
|
private static volatile boolean playerResponseVideoIdIsShort;
|
||||||
private static volatile boolean videoIdIsShort;
|
private static volatile boolean videoIdIsShort;
|
||||||
@@ -45,6 +68,44 @@ public final class VideoInformation {
|
|||||||
*/
|
*/
|
||||||
private static float playbackSpeed = DEFAULT_YOUTUBE_PLAYBACK_SPEED;
|
private static float playbackSpeed = DEFAULT_YOUTUBE_PLAYBACK_SPEED;
|
||||||
|
|
||||||
|
private static int desiredVideoResolution = AUTOMATIC_VIDEO_QUALITY_VALUE;
|
||||||
|
|
||||||
|
private static boolean qualityNeedsUpdating;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The available qualities of the current video.
|
||||||
|
*/
|
||||||
|
@Nullable
|
||||||
|
private static VideoQuality[] currentQualities;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The current quality of the video playing.
|
||||||
|
* This is always the actual quality even if Automatic quality is active.
|
||||||
|
*/
|
||||||
|
@Nullable
|
||||||
|
private static VideoQuality currentQuality;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The current VideoQualityMenuInterface, set during setVideoQuality.
|
||||||
|
*/
|
||||||
|
@Nullable
|
||||||
|
private static VideoQualityMenuInterface currentMenuInterface;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callback for when the current quality changes.
|
||||||
|
*/
|
||||||
|
public static final Event<VideoQuality> onQualityChange = new Event<>();
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public static VideoQuality[] getCurrentQualities() {
|
||||||
|
return currentQualities;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public static VideoQuality getCurrentQuality() {
|
||||||
|
return currentQuality;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Injection point.
|
* Injection point.
|
||||||
*
|
*
|
||||||
@@ -52,12 +113,18 @@ public final class VideoInformation {
|
|||||||
*/
|
*/
|
||||||
public static void initialize(@NonNull PlaybackController playerController) {
|
public static void initialize(@NonNull PlaybackController playerController) {
|
||||||
try {
|
try {
|
||||||
|
Logger.printDebug(() -> "newVideoStarted");
|
||||||
|
|
||||||
playerControllerRef = new WeakReference<>(Objects.requireNonNull(playerController));
|
playerControllerRef = new WeakReference<>(Objects.requireNonNull(playerController));
|
||||||
videoTime = -1;
|
videoTime = -1;
|
||||||
videoLength = 0;
|
videoLength = 0;
|
||||||
playbackSpeed = DEFAULT_YOUTUBE_PLAYBACK_SPEED;
|
playbackSpeed = DEFAULT_YOUTUBE_PLAYBACK_SPEED;
|
||||||
|
desiredVideoResolution = AUTOMATIC_VIDEO_QUALITY_VALUE;
|
||||||
|
currentQualities = null;
|
||||||
|
currentMenuInterface = null;
|
||||||
|
setCurrentQuality(null);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.printException(() -> "Failed to initialize", ex);
|
Logger.printException(() -> "initialize failure", ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -197,14 +264,14 @@ public final class VideoInformation {
|
|||||||
if (controller == null) {
|
if (controller == null) {
|
||||||
Logger.printDebug(() -> "Cannot seekTo because player controller is null");
|
Logger.printDebug(() -> "Cannot seekTo because player controller is null");
|
||||||
} else {
|
} else {
|
||||||
if (controller.seekTo(adjustedSeekTime)) return true;
|
if (controller.patch_seekTo(adjustedSeekTime)) return true;
|
||||||
Logger.printDebug(() -> "seekTo did not succeeded. Trying MXD.");
|
Logger.printDebug(() -> "seekTo did not succeeded. Trying MXD.");
|
||||||
// Else the video is loading or changing videos, or video is casting to a different device.
|
// Else the video is loading or changing videos, or video is casting to a different device.
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try calling the seekTo method of the MDX player director (called when casting).
|
// Try calling the seekTo method of the MDX player director (called when casting).
|
||||||
// The difference has to be a different second mark in order to avoid infinite skip loops
|
// The difference has to be a different second mark in order to avoid infinite skip loops
|
||||||
// as the Lounge API only supports seconds.
|
// as the Lounge API only supports whole seconds.
|
||||||
if (adjustedSeekTime / 1000 == videoTime / 1000) {
|
if (adjustedSeekTime / 1000 == videoTime / 1000) {
|
||||||
Logger.printDebug(() -> "Skipping seekTo for MDX because seek time is too small "
|
Logger.printDebug(() -> "Skipping seekTo for MDX because seek time is too small "
|
||||||
+ "(" + (adjustedSeekTime - videoTime) + "ms)");
|
+ "(" + (adjustedSeekTime - videoTime) + "ms)");
|
||||||
@@ -217,9 +284,9 @@ public final class VideoInformation {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return controller.seekTo(adjustedSeekTime);
|
return controller.patch_seekTo(adjustedSeekTime);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.printException(() -> "Failed to seek", ex);
|
Logger.printException(() -> "seekTo failure", ex);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -239,7 +306,7 @@ public final class VideoInformation {
|
|||||||
if (controller == null) {
|
if (controller == null) {
|
||||||
Logger.printDebug(() -> "Cannot seek relative as player controller is null");
|
Logger.printDebug(() -> "Cannot seek relative as player controller is null");
|
||||||
} else {
|
} else {
|
||||||
controller.seekToRelative(seekTime);
|
controller.patch_seekToRelative(seekTime);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Adjust the fine adjustment function so it's at least 1 second before/after.
|
// Adjust the fine adjustment function so it's at least 1 second before/after.
|
||||||
@@ -255,10 +322,10 @@ public final class VideoInformation {
|
|||||||
if (controller == null) {
|
if (controller == null) {
|
||||||
Logger.printDebug(() -> "Cannot seek relative as MXD player controller is null");
|
Logger.printDebug(() -> "Cannot seek relative as MXD player controller is null");
|
||||||
} else {
|
} else {
|
||||||
controller.seekToRelative(adjustedSeekTime);
|
controller.patch_seekToRelative(adjustedSeekTime);
|
||||||
}
|
}
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
Logger.printException(() -> "Failed to seek relative", ex);
|
Logger.printException(() -> "seekToRelative failure", ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -339,14 +406,13 @@ public final class VideoInformation {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return If the playback is at the end of the video.
|
|
||||||
* <p>
|
|
||||||
* If video is playing in the background with no video visible,
|
* If video is playing in the background with no video visible,
|
||||||
* this always returns false (even if the video is actually at the end).
|
* this always returns false (even if the video is actually at the end).
|
||||||
* <p>
|
* <p>
|
||||||
* This is equivalent to checking for {@link VideoState#ENDED},
|
* This is equivalent to checking for {@link VideoState#ENDED},
|
||||||
* but can give a more up-to-date result for code calling from some hooks.
|
* but can give a more up-to-date result for code calling from some hooks.
|
||||||
*
|
*
|
||||||
|
* @return If the playback is at the end of the video.
|
||||||
* @see VideoState
|
* @see VideoState
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("BooleanMethodIsAlwaysInverted")
|
@SuppressWarnings("BooleanMethodIsAlwaysInverted")
|
||||||
@@ -373,4 +439,137 @@ public final class VideoInformation {
|
|||||||
playbackSpeed = newlyLoadedPlaybackSpeed;
|
playbackSpeed = newlyLoadedPlaybackSpeed;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param resolution The desired video quality resolution to use.
|
||||||
|
*/
|
||||||
|
public static void setDesiredVideoResolution(int resolution) {
|
||||||
|
Utils.verifyOnMainThread();
|
||||||
|
Logger.printDebug(() -> "Setting desired video resolution: " + resolution);
|
||||||
|
desiredVideoResolution = resolution;
|
||||||
|
qualityNeedsUpdating = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void setCurrentQuality(@Nullable VideoQuality quality) {
|
||||||
|
Utils.verifyOnMainThread();
|
||||||
|
if (currentQuality != quality) {
|
||||||
|
Logger.printDebug(() -> "Current quality changed to: " + quality);
|
||||||
|
currentQuality = quality;
|
||||||
|
onQualityChange.invoke(quality);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Forcefully changes the video quality of the currently playing video.
|
||||||
|
*/
|
||||||
|
public static void changeQuality(VideoQuality quality) {
|
||||||
|
Utils.verifyOnMainThread();
|
||||||
|
|
||||||
|
if (currentMenuInterface == null) {
|
||||||
|
Logger.printException(() -> "Cannot change quality, menu interface is null");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
currentMenuInterface.patch_setQuality(quality);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point. Fixes bad data used by YouTube.
|
||||||
|
* Issue can be reproduced by selecting 480p quality on any Short,
|
||||||
|
* and occasionally with random regular videos.
|
||||||
|
*/
|
||||||
|
public static int fixVideoQualityResolution(String name, int quality) {
|
||||||
|
try {
|
||||||
|
if (!name.startsWith(Integer.toString(quality))) {
|
||||||
|
final int suffixIndex = name.indexOf('p');
|
||||||
|
if (suffixIndex > 0) {
|
||||||
|
final int fixedQuality = Integer.parseInt(name.substring(0, suffixIndex));
|
||||||
|
Logger.printDebug(() -> "Fixing wrong quality resolution from: " +
|
||||||
|
name + "(" + quality + ") to: " + name + ")" + fixedQuality + ")");
|
||||||
|
return fixedQuality;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception ex) {
|
||||||
|
Logger.printException(() -> "fixVideoQualityResolution failed", ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
return quality;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Injection point.
|
||||||
|
*
|
||||||
|
* @param qualities Video qualities available, ordered from largest to smallest, with index 0 being the 'automatic' value of -2
|
||||||
|
* @param originalQualityIndex quality index to use, as chosen by YouTube
|
||||||
|
*/
|
||||||
|
public static int setVideoQuality(VideoQuality[] qualities, VideoQualityMenuInterface menu, int originalQualityIndex) {
|
||||||
|
try {
|
||||||
|
Utils.verifyOnMainThread();
|
||||||
|
currentMenuInterface = menu;
|
||||||
|
|
||||||
|
final boolean availableQualitiesChanged = (currentQualities == null)
|
||||||
|
|| !Arrays.equals(currentQualities, qualities);
|
||||||
|
if (availableQualitiesChanged) {
|
||||||
|
currentQualities = qualities;
|
||||||
|
Logger.printDebug(() -> "VideoQualities: " + Arrays.toString(currentQualities));
|
||||||
|
}
|
||||||
|
|
||||||
|
// On extremely slow internet connections the index can initially be -1
|
||||||
|
originalQualityIndex = Math.max(0, originalQualityIndex);
|
||||||
|
|
||||||
|
VideoQuality updatedCurrentQuality = qualities[originalQualityIndex];
|
||||||
|
if (updatedCurrentQuality.patch_getResolution() != AUTOMATIC_VIDEO_QUALITY_VALUE
|
||||||
|
&& (currentQuality == null || currentQuality != updatedCurrentQuality)) {
|
||||||
|
setCurrentQuality(updatedCurrentQuality);
|
||||||
|
}
|
||||||
|
|
||||||
|
final int preferredQuality = desiredVideoResolution;
|
||||||
|
if (preferredQuality == AUTOMATIC_VIDEO_QUALITY_VALUE) {
|
||||||
|
return originalQualityIndex; // Nothing to do.
|
||||||
|
}
|
||||||
|
|
||||||
|
// After changing videos the qualities can initially be for the prior video.
|
||||||
|
// If the qualities have changed and the default is not auto then an update is needed.
|
||||||
|
if (qualityNeedsUpdating) {
|
||||||
|
qualityNeedsUpdating = false;
|
||||||
|
} else if (!availableQualitiesChanged) {
|
||||||
|
return originalQualityIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find the highest quality that is equal to or less than the preferred.
|
||||||
|
int i = 0;
|
||||||
|
final int lastQualityIndex = qualities.length - 1;
|
||||||
|
for (VideoQuality quality : qualities) {
|
||||||
|
final int qualityResolution = quality.patch_getResolution();
|
||||||
|
if ((qualityResolution != AUTOMATIC_VIDEO_QUALITY_VALUE && qualityResolution <= preferredQuality)
|
||||||
|
// Use the lowest video quality if the default is lower than all available.
|
||||||
|
|| i == lastQualityIndex) {
|
||||||
|
final boolean qualityNeedsChange = (i != originalQualityIndex);
|
||||||
|
Logger.printDebug(() -> qualityNeedsChange
|
||||||
|
? "Changing video quality from: " + updatedCurrentQuality + " to: " + quality
|
||||||
|
: "Video is already the preferred quality: " + quality
|
||||||
|
);
|
||||||
|
|
||||||
|
// On first load of a new regular video, if the video is already the
|
||||||
|
// desired quality then the quality flyout will show 'Auto' (ie: Auto (720p)).
|
||||||
|
//
|
||||||
|
// To prevent user confusion, set the video index even if the
|
||||||
|
// quality is already correct so the UI picker will not display "Auto".
|
||||||
|
//
|
||||||
|
// Only change Shorts quality if the quality actually needs to change,
|
||||||
|
// because the "auto" option is not shown in the flyout
|
||||||
|
// and setting the same quality again can cause the Short to restart.
|
||||||
|
if (qualityNeedsChange || !ShortsPlayerState.isOpen()) {
|
||||||
|
changeQuality(quality);
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
return originalQualityIndex;
|
||||||
|
}
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
} catch (Exception ex) {
|
||||||
|
Logger.printException(() -> "setVideoQuality failure", ex);
|
||||||
|
}
|
||||||
|
return originalQualityIndex;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,10 +59,11 @@ public final class AnnouncementsPatch {
|
|||||||
int id = Settings.ANNOUNCEMENT_LAST_ID.defaultValue;
|
int id = Settings.ANNOUNCEMENT_LAST_ID.defaultValue;
|
||||||
try {
|
try {
|
||||||
final var announcementIds = new JSONArray(jsonString);
|
final var announcementIds = new JSONArray(jsonString);
|
||||||
|
if (announcementIds.length() == 0) return true;
|
||||||
|
|
||||||
id = announcementIds.getJSONObject(0).getInt("id");
|
id = announcementIds.getJSONObject(0).getInt("id");
|
||||||
|
|
||||||
} catch (Throwable ex) {
|
} catch (Throwable ex) {
|
||||||
Logger.printException(() -> "Failed to parse announcement IDs", ex);
|
Logger.printException(() -> "Failed to parse announcement ID", ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Do not show the announcement, if the last announcement id is the same as the current one.
|
// Do not show the announcement, if the last announcement id is the same as the current one.
|
||||||
|
|||||||
@@ -10,8 +10,8 @@ import static app.revanced.extension.shared.requests.Route.Method.GET;
|
|||||||
|
|
||||||
public class AnnouncementsRoutes {
|
public class AnnouncementsRoutes {
|
||||||
private static final String ANNOUNCEMENTS_PROVIDER = "https://api.revanced.app/v4";
|
private static final String ANNOUNCEMENTS_PROVIDER = "https://api.revanced.app/v4";
|
||||||
public static final Route GET_LATEST_ANNOUNCEMENT_IDS = new Route(GET, "/announcements/latest/id?tag=youtube");
|
public static final Route GET_LATEST_ANNOUNCEMENT_IDS = new Route(GET, "/announcements/latest/id?tag=\uD83C\uDF9E\uFE0F%20YouTube");
|
||||||
public static final Route GET_LATEST_ANNOUNCEMENTS = new Route(GET, "/announcements/latest?tag=youtube");
|
public static final Route GET_LATEST_ANNOUNCEMENTS = new Route(GET, "/announcements/latest?tag=\uD83C\uDF9E\uFE0F%20YouTube");
|
||||||
|
|
||||||
private AnnouncementsRoutes() {
|
private AnnouncementsRoutes() {
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,13 +6,11 @@ import android.app.Instrumentation;
|
|||||||
import android.view.KeyEvent;
|
import android.view.KeyEvent;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.Utils;
|
import app.revanced.extension.shared.Utils;
|
||||||
import app.revanced.extension.youtube.StringTrieSearch;
|
import app.revanced.extension.shared.StringTrieSearch;
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
@@ -34,10 +32,6 @@ public final class AdsFilter extends Filter {
|
|||||||
private final StringFilterGroup playerShoppingShelf;
|
private final StringFilterGroup playerShoppingShelf;
|
||||||
private final ByteArrayFilterGroup playerShoppingShelfBuffer;
|
private final ByteArrayFilterGroup playerShoppingShelfBuffer;
|
||||||
|
|
||||||
private final StringFilterGroup channelProfile;
|
|
||||||
private final ByteArrayFilterGroup visitStoreButton;
|
|
||||||
|
|
||||||
private final StringFilterGroup shoppingLinks;
|
|
||||||
|
|
||||||
public AdsFilter() {
|
public AdsFilter() {
|
||||||
exceptions.addPatterns(
|
exceptions.addPatterns(
|
||||||
@@ -91,6 +85,7 @@ public final class AdsFilter extends Filter {
|
|||||||
"text_image_no_button_layout", // Tablet layout search results.
|
"text_image_no_button_layout", // Tablet layout search results.
|
||||||
"video_display_button_group_layout",
|
"video_display_button_group_layout",
|
||||||
"video_display_carousel_button_group_layout",
|
"video_display_carousel_button_group_layout",
|
||||||
|
"video_display_carousel_buttoned_short_dr_layout",
|
||||||
"video_display_full_buttoned_short_dr_layout",
|
"video_display_full_buttoned_short_dr_layout",
|
||||||
"video_display_full_layout",
|
"video_display_full_layout",
|
||||||
"watch_metadata_app_promo"
|
"watch_metadata_app_promo"
|
||||||
@@ -107,41 +102,27 @@ public final class AdsFilter extends Filter {
|
|||||||
);
|
);
|
||||||
|
|
||||||
final var viewProducts = new StringFilterGroup(
|
final var viewProducts = new StringFilterGroup(
|
||||||
Settings.HIDE_PRODUCTS_BANNER,
|
Settings.HIDE_VIEW_PRODUCTS_BANNER,
|
||||||
"product_item",
|
"product_item",
|
||||||
"products_in_video",
|
"products_in_video",
|
||||||
"shopping_overlay.eml", // Video player overlay shopping links.
|
"shopping_overlay.eml" // Video player overlay shopping links.
|
||||||
"shopping_carousel.eml" // Channel profile shopping shelf.
|
|
||||||
);
|
);
|
||||||
|
|
||||||
shoppingLinks = new StringFilterGroup(
|
final var shoppingLinks = new StringFilterGroup(
|
||||||
Settings.HIDE_SHOPPING_LINKS,
|
Settings.HIDE_SHOPPING_LINKS,
|
||||||
"expandable_list"
|
"shopping_description_shelf.eml"
|
||||||
);
|
);
|
||||||
|
|
||||||
playerShoppingShelf = new StringFilterGroup(
|
playerShoppingShelf = new StringFilterGroup(
|
||||||
Settings.HIDE_PLAYER_STORE_SHELF,
|
Settings.HIDE_CREATOR_STORE_SHELF,
|
||||||
"expandable_list.eml",
|
|
||||||
"horizontal_shelf.eml"
|
"horizontal_shelf.eml"
|
||||||
);
|
);
|
||||||
|
|
||||||
playerShoppingShelfBuffer = new ByteArrayFilterGroup(
|
playerShoppingShelfBuffer = new ByteArrayFilterGroup(
|
||||||
null,
|
null,
|
||||||
"shopping_link_item",
|
|
||||||
"shopping_item_card_list"
|
"shopping_item_card_list"
|
||||||
);
|
);
|
||||||
|
|
||||||
channelProfile = new StringFilterGroup(
|
|
||||||
Settings.HIDE_VISIT_STORE_BUTTON,
|
|
||||||
"channel_profile.eml",
|
|
||||||
"page_header.eml"
|
|
||||||
);
|
|
||||||
|
|
||||||
visitStoreButton = new ByteArrayFilterGroup(
|
|
||||||
null,
|
|
||||||
"header_store_button"
|
|
||||||
);
|
|
||||||
|
|
||||||
final var webLinkPanel = new StringFilterGroup(
|
final var webLinkPanel = new StringFilterGroup(
|
||||||
Settings.HIDE_WEB_SEARCH_RESULTS,
|
Settings.HIDE_WEB_SEARCH_RESULTS,
|
||||||
"web_link_panel"
|
"web_link_panel"
|
||||||
@@ -149,7 +130,8 @@ public final class AdsFilter extends Filter {
|
|||||||
|
|
||||||
final var merchandise = new StringFilterGroup(
|
final var merchandise = new StringFilterGroup(
|
||||||
Settings.HIDE_MERCHANDISE_BANNERS,
|
Settings.HIDE_MERCHANDISE_BANNERS,
|
||||||
"product_carousel"
|
"product_carousel",
|
||||||
|
"shopping_carousel.eml" // Channel profile shopping shelf.
|
||||||
);
|
);
|
||||||
|
|
||||||
final var selfSponsor = new StringFilterGroup(
|
final var selfSponsor = new StringFilterGroup(
|
||||||
@@ -158,29 +140,23 @@ public final class AdsFilter extends Filter {
|
|||||||
);
|
);
|
||||||
|
|
||||||
addPathCallbacks(
|
addPathCallbacks(
|
||||||
|
fullscreenAd,
|
||||||
generalAds,
|
generalAds,
|
||||||
merchandise,
|
merchandise,
|
||||||
viewProducts,
|
movieAds,
|
||||||
selfSponsor,
|
|
||||||
fullscreenAd,
|
|
||||||
channelProfile,
|
|
||||||
webLinkPanel,
|
|
||||||
shoppingLinks,
|
|
||||||
playerShoppingShelf,
|
playerShoppingShelf,
|
||||||
movieAds
|
selfSponsor,
|
||||||
|
shoppingLinks,
|
||||||
|
viewProducts,
|
||||||
|
webLinkPanel
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
boolean isFiltered(@Nullable String identifier, String path, byte[] protobufBufferArray,
|
boolean isFiltered(String identifier, String path, byte[] buffer,
|
||||||
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
||||||
if (matchedGroup == playerShoppingShelf) {
|
if (matchedGroup == playerShoppingShelf) {
|
||||||
return contentIndex == 0 && playerShoppingShelfBuffer.check(protobufBufferArray).isFiltered();
|
return contentIndex == 0 && playerShoppingShelfBuffer.check(buffer).isFiltered();
|
||||||
}
|
|
||||||
|
|
||||||
// Check for the index because of likelihood of false positives.
|
|
||||||
if (contentIndex != 0 && matchedGroup == shoppingLinks) {
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (exceptions.matches(path)) {
|
if (exceptions.matches(path)) {
|
||||||
@@ -194,10 +170,6 @@ public final class AdsFilter extends Filter {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (matchedGroup == channelProfile) {
|
|
||||||
return visitStoreButton.check(protobufBufferArray).isFiltered();
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
package app.revanced.extension.youtube.patches.components;
|
package app.revanced.extension.youtube.patches.components;
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
|
|
||||||
import app.revanced.extension.youtube.patches.playback.quality.AdvancedVideoQualityMenuPatch;
|
import app.revanced.extension.youtube.patches.playback.quality.AdvancedVideoQualityMenuPatch;
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
@@ -21,7 +19,7 @@ public final class AdvancedVideoQualityMenuFilter extends Filter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
boolean isFiltered(@Nullable String identifier, String path, byte[] protobufBufferArray,
|
boolean isFiltered(String identifier, String path, byte[] buffer,
|
||||||
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
||||||
isVideoQualityMenuVisible = true;
|
isVideoQualityMenuVisible = true;
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
package app.revanced.extension.youtube.patches.components;
|
package app.revanced.extension.youtube.patches.components;
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
|
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
@@ -46,7 +44,7 @@ final class ButtonsFilter extends Filter {
|
|||||||
"|download_button.eml"
|
"|download_button.eml"
|
||||||
),
|
),
|
||||||
new StringFilterGroup(
|
new StringFilterGroup(
|
||||||
Settings.HIDE_PLAYLIST_BUTTON,
|
Settings.HIDE_SAVE_BUTTON,
|
||||||
"|save_to_playlist_button"
|
"|save_to_playlist_button"
|
||||||
),
|
),
|
||||||
new StringFilterGroup(
|
new StringFilterGroup(
|
||||||
@@ -76,11 +74,27 @@ final class ButtonsFilter extends Filter {
|
|||||||
Settings.HIDE_ASK_BUTTON,
|
Settings.HIDE_ASK_BUTTON,
|
||||||
"yt_fill_spark"
|
"yt_fill_spark"
|
||||||
),
|
),
|
||||||
|
new ByteArrayFilterGroup(
|
||||||
|
Settings.HIDE_SHOP_BUTTON,
|
||||||
|
"yt_outline_bag"
|
||||||
|
),
|
||||||
|
new ByteArrayFilterGroup(
|
||||||
|
Settings.HIDE_STOP_ADS_BUTTON,
|
||||||
|
"yt_outline_slash_circle_left"
|
||||||
|
),
|
||||||
// Check for clip button both here and using a path filter,
|
// Check for clip button both here and using a path filter,
|
||||||
// as there's a chance the path is a generic action button and won't contain 'clip_button'
|
// as there's a chance the path is a generic action button and won't contain 'clip_button'
|
||||||
new ByteArrayFilterGroup(
|
new ByteArrayFilterGroup(
|
||||||
Settings.HIDE_CLIP_BUTTON,
|
Settings.HIDE_CLIP_BUTTON,
|
||||||
"yt_outline_scissors"
|
"yt_outline_scissors"
|
||||||
|
),
|
||||||
|
new ByteArrayFilterGroup(
|
||||||
|
Settings.HIDE_HYPE_BUTTON,
|
||||||
|
"yt_outline_star_shooting"
|
||||||
|
),
|
||||||
|
new ByteArrayFilterGroup(
|
||||||
|
Settings.HIDE_PROMOTE_BUTTON,
|
||||||
|
"yt_outline_megaphone"
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -96,7 +110,7 @@ final class ButtonsFilter extends Filter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
boolean isFiltered(@Nullable String identifier, String path, byte[] protobufBufferArray,
|
boolean isFiltered(String identifier, String path, byte[] buffer,
|
||||||
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
||||||
if (matchedGroup == likeSubscribeGlow) {
|
if (matchedGroup == likeSubscribeGlow) {
|
||||||
return (path.startsWith(VIDEO_ACTION_BAR_PATH_PREFIX) || path.startsWith(COMPACT_CHANNEL_BAR_PATH_PREFIX))
|
return (path.startsWith(VIDEO_ACTION_BAR_PATH_PREFIX) || path.startsWith(COMPACT_CHANNEL_BAR_PATH_PREFIX))
|
||||||
@@ -113,7 +127,7 @@ final class ButtonsFilter extends Filter {
|
|||||||
// Make sure the current path is the right one
|
// Make sure the current path is the right one
|
||||||
// to avoid false positives.
|
// to avoid false positives.
|
||||||
return path.startsWith(VIDEO_ACTION_BAR_PATH)
|
return path.startsWith(VIDEO_ACTION_BAR_PATH)
|
||||||
&& bufferButtonsGroupList.check(protobufBufferArray).isFiltered();
|
&& bufferButtonsGroupList.check(buffer).isFiltered();
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
@@ -1,13 +1,12 @@
|
|||||||
package app.revanced.extension.youtube.patches.components;
|
package app.revanced.extension.youtube.patches.components;
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
|
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
import app.revanced.extension.youtube.shared.PlayerType;
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
final class CommentsFilter extends Filter {
|
final class CommentsFilter extends Filter {
|
||||||
|
|
||||||
private final StringFilterGroup filterChipBar;
|
private final StringFilterGroup chipBar;
|
||||||
private final ByteArrayFilterGroup aiCommentsSummary;
|
private final ByteArrayFilterGroup aiCommentsSummary;
|
||||||
|
|
||||||
public CommentsFilter() {
|
public CommentsFilter() {
|
||||||
@@ -16,6 +15,21 @@ final class CommentsFilter extends Filter {
|
|||||||
"live_chat_summary_banner.eml"
|
"live_chat_summary_banner.eml"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
chipBar = new StringFilterGroup(
|
||||||
|
Settings.HIDE_COMMENTS_AI_SUMMARY,
|
||||||
|
"chip_bar.eml"
|
||||||
|
);
|
||||||
|
|
||||||
|
aiCommentsSummary = new ByteArrayFilterGroup(
|
||||||
|
null,
|
||||||
|
"yt_fill_spark_"
|
||||||
|
);
|
||||||
|
|
||||||
|
var channelGuidelines = new StringFilterGroup(
|
||||||
|
Settings.HIDE_COMMENTS_CHANNEL_GUIDELINES,
|
||||||
|
"channel_guidelines_entry_banner"
|
||||||
|
);
|
||||||
|
|
||||||
var commentsByMembers = new StringFilterGroup(
|
var commentsByMembers = new StringFilterGroup(
|
||||||
Settings.HIDE_COMMENTS_BY_MEMBERS_HEADER,
|
Settings.HIDE_COMMENTS_BY_MEMBERS_HEADER,
|
||||||
"sponsorships_comments_header.eml",
|
"sponsorships_comments_header.eml",
|
||||||
@@ -28,6 +42,11 @@ final class CommentsFilter extends Filter {
|
|||||||
"_comments"
|
"_comments"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
var communityGuidelines = new StringFilterGroup(
|
||||||
|
Settings.HIDE_COMMENTS_COMMUNITY_GUIDELINES,
|
||||||
|
"community_guidelines"
|
||||||
|
);
|
||||||
|
|
||||||
var createAShort = new StringFilterGroup(
|
var createAShort = new StringFilterGroup(
|
||||||
Settings.HIDE_COMMENTS_CREATE_A_SHORT_BUTTON,
|
Settings.HIDE_COMMENTS_CREATE_A_SHORT_BUTTON,
|
||||||
"composer_short_creation_button.eml"
|
"composer_short_creation_button.eml"
|
||||||
@@ -50,33 +69,28 @@ final class CommentsFilter extends Filter {
|
|||||||
"composer_timestamp_button.eml"
|
"composer_timestamp_button.eml"
|
||||||
);
|
);
|
||||||
|
|
||||||
filterChipBar = new StringFilterGroup(
|
|
||||||
Settings.HIDE_COMMENTS_AI_SUMMARY,
|
|
||||||
"filter_chip_bar.eml"
|
|
||||||
);
|
|
||||||
|
|
||||||
aiCommentsSummary = new ByteArrayFilterGroup(
|
|
||||||
null,
|
|
||||||
"yt_fill_spark_"
|
|
||||||
);
|
|
||||||
|
|
||||||
addPathCallbacks(
|
addPathCallbacks(
|
||||||
|
channelGuidelines,
|
||||||
chatSummary,
|
chatSummary,
|
||||||
|
chipBar,
|
||||||
commentsByMembers,
|
commentsByMembers,
|
||||||
comments,
|
comments,
|
||||||
|
communityGuidelines,
|
||||||
createAShort,
|
createAShort,
|
||||||
previewComment,
|
previewComment,
|
||||||
thanksButton,
|
thanksButton,
|
||||||
timestampButton,
|
timestampButton
|
||||||
filterChipBar
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
boolean isFiltered(@Nullable String identifier, String path, byte[] protobufBufferArray,
|
boolean isFiltered(String identifier, String path, byte[] buffer,
|
||||||
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
||||||
if (matchedGroup == filterChipBar) {
|
if (matchedGroup == chipBar) {
|
||||||
return aiCommentsSummary.check(protobufBufferArray).isFiltered();
|
// Playlist sort button uses same components and must only filter if the player is opened.
|
||||||
|
return PlayerType.getCurrent().isMaximizedOrFullscreen()
|
||||||
|
&& aiCommentsSummary.check(buffer).isFiltered();
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ package app.revanced.extension.youtube.patches.components;
|
|||||||
import static app.revanced.extension.shared.StringRef.str;
|
import static app.revanced.extension.shared.StringRef.str;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
|
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
@@ -15,7 +14,7 @@ import java.util.regex.Pattern;
|
|||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.Utils;
|
import app.revanced.extension.shared.Utils;
|
||||||
import app.revanced.extension.youtube.ByteTrieSearch;
|
import app.revanced.extension.shared.ByteTrieSearch;
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -146,7 +145,7 @@ final class CustomFilter extends Filter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
boolean isFiltered(@Nullable String identifier, String path, byte[] protobufBufferArray,
|
boolean isFiltered(String identifier, String path, byte[] buffer,
|
||||||
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
||||||
// All callbacks are custom filter groups.
|
// All callbacks are custom filter groups.
|
||||||
CustomFilterGroup custom = (CustomFilterGroup) matchedGroup;
|
CustomFilterGroup custom = (CustomFilterGroup) matchedGroup;
|
||||||
@@ -158,6 +157,6 @@ final class CustomFilter extends Filter {
|
|||||||
return true; // No buffer filter, only path filtering.
|
return true; // No buffer filter, only path filtering.
|
||||||
}
|
}
|
||||||
|
|
||||||
return custom.bufferSearch.matches(protobufBufferArray);
|
return custom.bufferSearch.matches(buffer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,9 +1,8 @@
|
|||||||
package app.revanced.extension.youtube.patches.components;
|
package app.revanced.extension.youtube.patches.components;
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
import app.revanced.extension.shared.StringTrieSearch;
|
||||||
|
|
||||||
import app.revanced.extension.youtube.StringTrieSearch;
|
|
||||||
import app.revanced.extension.youtube.settings.Settings;
|
import app.revanced.extension.youtube.settings.Settings;
|
||||||
|
import app.revanced.extension.youtube.shared.PlayerType;
|
||||||
|
|
||||||
@SuppressWarnings("unused")
|
@SuppressWarnings("unused")
|
||||||
final class DescriptionComponentsFilter extends Filter {
|
final class DescriptionComponentsFilter extends Filter {
|
||||||
@@ -14,6 +13,11 @@ final class DescriptionComponentsFilter extends Filter {
|
|||||||
|
|
||||||
private final StringFilterGroup macroMarkersCarousel;
|
private final StringFilterGroup macroMarkersCarousel;
|
||||||
|
|
||||||
|
private final StringFilterGroup horizontalShelf;
|
||||||
|
private final ByteArrayFilterGroup cellVideoAttribute;
|
||||||
|
|
||||||
|
private final StringFilterGroup aiGeneratedVideoSummarySection;
|
||||||
|
|
||||||
public DescriptionComponentsFilter() {
|
public DescriptionComponentsFilter() {
|
||||||
exceptions.addPatterns(
|
exceptions.addPatterns(
|
||||||
"compact_channel",
|
"compact_channel",
|
||||||
@@ -23,7 +27,7 @@ final class DescriptionComponentsFilter extends Filter {
|
|||||||
"metadata"
|
"metadata"
|
||||||
);
|
);
|
||||||
|
|
||||||
final StringFilterGroup aiGeneratedVideoSummarySection = new StringFilterGroup(
|
aiGeneratedVideoSummarySection = new StringFilterGroup(
|
||||||
Settings.HIDE_AI_GENERATED_VIDEO_SUMMARY_SECTION,
|
Settings.HIDE_AI_GENERATED_VIDEO_SUMMARY_SECTION,
|
||||||
"cell_expandable_metadata.eml"
|
"cell_expandable_metadata.eml"
|
||||||
);
|
);
|
||||||
@@ -35,8 +39,7 @@ final class DescriptionComponentsFilter extends Filter {
|
|||||||
|
|
||||||
final StringFilterGroup attributesSection = new StringFilterGroup(
|
final StringFilterGroup attributesSection = new StringFilterGroup(
|
||||||
Settings.HIDE_ATTRIBUTES_SECTION,
|
Settings.HIDE_ATTRIBUTES_SECTION,
|
||||||
"gaming_section",
|
// "gaming_section", "music_section"
|
||||||
"music_section",
|
|
||||||
"video_attributes_section"
|
"video_attributes_section"
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -76,25 +79,46 @@ final class DescriptionComponentsFilter extends Filter {
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
horizontalShelf = new StringFilterGroup(
|
||||||
|
Settings.HIDE_ATTRIBUTES_SECTION,
|
||||||
|
"horizontal_shelf.eml"
|
||||||
|
);
|
||||||
|
|
||||||
|
cellVideoAttribute = new ByteArrayFilterGroup(
|
||||||
|
null,
|
||||||
|
"cell_video_attribute"
|
||||||
|
);
|
||||||
|
|
||||||
addPathCallbacks(
|
addPathCallbacks(
|
||||||
aiGeneratedVideoSummarySection,
|
aiGeneratedVideoSummarySection,
|
||||||
askSection,
|
askSection,
|
||||||
attributesSection,
|
attributesSection,
|
||||||
infoCardsSection,
|
infoCardsSection,
|
||||||
|
horizontalShelf,
|
||||||
howThisWasMadeSection,
|
howThisWasMadeSection,
|
||||||
|
macroMarkersCarousel,
|
||||||
podcastSection,
|
podcastSection,
|
||||||
transcriptSection,
|
transcriptSection
|
||||||
macroMarkersCarousel
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
boolean isFiltered(@Nullable String identifier, String path, byte[] protobufBufferArray,
|
boolean isFiltered(String identifier, String path, byte[] buffer,
|
||||||
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
||||||
|
|
||||||
|
if (matchedGroup == aiGeneratedVideoSummarySection) {
|
||||||
|
// Only hide if player is open, in case this component is used somewhere else.
|
||||||
|
return PlayerType.getCurrent().isMaximizedOrFullscreen();
|
||||||
|
}
|
||||||
|
|
||||||
if (exceptions.matches(path)) return false;
|
if (exceptions.matches(path)) return false;
|
||||||
|
|
||||||
if (matchedGroup == macroMarkersCarousel) {
|
if (matchedGroup == macroMarkersCarousel) {
|
||||||
return contentIndex == 0 && macroMarkersCarouselGroupList.check(protobufBufferArray).isFiltered();
|
return contentIndex == 0 && macroMarkersCarouselGroupList.check(buffer).isFiltered();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (matchedGroup == horizontalShelf) {
|
||||||
|
return cellVideoAttribute.check(buffer).isFiltered();
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
package app.revanced.extension.youtube.patches.components;
|
package app.revanced.extension.youtube.patches.components;
|
||||||
|
|
||||||
import androidx.annotation.Nullable;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@@ -59,7 +57,6 @@ abstract class Filter {
|
|||||||
* Called after an enabled filter has been matched.
|
* Called after an enabled filter has been matched.
|
||||||
* Default implementation is to always filter the matched component and log the action.
|
* Default implementation is to always filter the matched component and log the action.
|
||||||
* Subclasses can perform additional or different checks if needed.
|
* Subclasses can perform additional or different checks if needed.
|
||||||
*
|
|
||||||
* <p>
|
* <p>
|
||||||
* Method is called off the main thread.
|
* Method is called off the main thread.
|
||||||
*
|
*
|
||||||
@@ -68,7 +65,7 @@ abstract class Filter {
|
|||||||
* @param contentIndex Matched index of the identifier or path.
|
* @param contentIndex Matched index of the identifier or path.
|
||||||
* @return True if the litho component should be filtered out.
|
* @return True if the litho component should be filtered out.
|
||||||
*/
|
*/
|
||||||
boolean isFiltered(@Nullable String identifier, String path, byte[] protobufBufferArray,
|
boolean isFiltered(String identifier, String path, byte[] buffer,
|
||||||
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
StringFilterGroup matchedGroup, FilterContentType contentType, int contentIndex) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import androidx.annotation.NonNull;
|
|||||||
|
|
||||||
import app.revanced.extension.shared.Logger;
|
import app.revanced.extension.shared.Logger;
|
||||||
import app.revanced.extension.shared.settings.BooleanSetting;
|
import app.revanced.extension.shared.settings.BooleanSetting;
|
||||||
import app.revanced.extension.youtube.ByteTrieSearch;
|
import app.revanced.extension.shared.ByteTrieSearch;
|
||||||
|
|
||||||
abstract class FilterGroup<T> {
|
abstract class FilterGroup<T> {
|
||||||
final static class FilterGroupResult {
|
final static class FilterGroupResult {
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user