From 12ea26b10ddea5ad39da1d35e2b8fd0b48c15d88 Mon Sep 17 00:00:00 2001
From: LisoUseInAIKyrios <118716522+LisoUseInAIKyrios@users.noreply.github.com>
Date: Wed, 27 Nov 2024 17:30:55 +0400
Subject: [PATCH] feat(TikTok): Add ReVanced settings about screen (#4009)
---
.../preference/ReVancedAboutPreference.java | 100 ++++++++++++------
.../ReVancedTikTokAboutPreference.java | 56 ++++++++++
.../ExtensionPreferenceCategory.java | 5 +-
.../resources/addresources/values/strings.xml | 3 +-
4 files changed, 130 insertions(+), 34 deletions(-)
create mode 100644 extensions/shared/src/main/java/app/revanced/extension/tiktok/settings/preference/ReVancedTikTokAboutPreference.java
diff --git a/extensions/shared/src/main/java/app/revanced/extension/shared/settings/preference/ReVancedAboutPreference.java b/extensions/shared/src/main/java/app/revanced/extension/shared/settings/preference/ReVancedAboutPreference.java
index 89fbe80e90..c3db3b4b08 100644
--- a/extensions/shared/src/main/java/app/revanced/extension/shared/settings/preference/ReVancedAboutPreference.java
+++ b/extensions/shared/src/main/java/app/revanced/extension/shared/settings/preference/ReVancedAboutPreference.java
@@ -1,6 +1,5 @@
package app.revanced.extension.shared.settings.preference;
-import static app.revanced.extension.shared.StringRef.sf;
import static app.revanced.extension.shared.StringRef.str;
import static app.revanced.extension.youtube.requests.Route.Method.GET;
@@ -13,6 +12,8 @@
import android.graphics.Color;
import android.net.Uri;
import android.os.Bundle;
+import android.os.Handler;
+import android.os.Looper;
import android.preference.Preference;
import android.util.AttributeSet;
import android.view.Window;
@@ -37,7 +38,7 @@
import app.revanced.extension.youtube.requests.Route;
/**
- * Opens a dialog showing the links from {@link SocialLinksRoutes}.
+ * Opens a dialog showing official links.
*/
@SuppressWarnings({"unused", "deprecation"})
public class ReVancedAboutPreference extends Preference {
@@ -72,7 +73,16 @@ protected int getDarkColor() {
return Color.BLACK;
}
- private String createDialogHtml(WebLink[] socialLinks) {
+ /**
+ * Apps that do not support bundling resources must override this.
+ *
+ * @return A localized string to display for the key.
+ */
+ protected String getString(String key, Object ... args) {
+ return str(key, args);
+ }
+
+ private String createDialogHtml(WebLink[] aboutLinks) {
final boolean isNetworkConnected = Utils.isNetworkConnected();
StringBuilder builder = new StringBuilder();
@@ -91,7 +101,7 @@ private String createDialogHtml(WebLink[] socialLinks) {
builder.append("");
+ + "src=\"").append(AboutLinksRoutes.aboutLogoUrl).append("\" />");
}
String patchesVersion = Utils.getPatchesReleaseVersion();
@@ -103,29 +113,29 @@ private String createDialogHtml(WebLink[] socialLinks) {
builder.append("
")
// Replace hyphens with non breaking dashes so the version number does not break lines.
- .append(useNonBreakingHyphens(str("revanced_settings_about_links_body", patchesVersion)))
+ .append(useNonBreakingHyphens(getString("revanced_settings_about_links_body", patchesVersion)))
.append("
");
// Add a disclaimer if using a dev release.
if (patchesVersion.contains("dev")) {
builder.append("")
// English text 'Pre-release' can break lines.
- .append(useNonBreakingHyphens(str("revanced_settings_about_links_dev_header")))
+ .append(useNonBreakingHyphens(getString("revanced_settings_about_links_dev_header")))
.append("
");
builder.append("")
- .append(str("revanced_settings_about_links_dev_body"))
+ .append(getString("revanced_settings_about_links_dev_body"))
.append("
");
}
builder.append("")
- .append(str("revanced_settings_about_links_header"))
+ .append(getString("revanced_settings_about_links_header"))
.append("
");
builder.append("");
- for (WebLink social : socialLinks) {
+ for (WebLink link : aboutLinks) {
builder.append("
");
- builder.append(String.format("
%s", social.url, social.name));
+ builder.append(String.format("
%s", link.url, link.name));
builder.append("
");
}
builder.append("
");
@@ -137,25 +147,44 @@ private String createDialogHtml(WebLink[] socialLinks) {
{
setOnPreferenceClickListener(pref -> {
// Show a progress spinner if the social links are not fetched yet.
- if (!SocialLinksRoutes.hasFetchedLinks() && Utils.isNetworkConnected()) {
+ if (!AboutLinksRoutes.hasFetchedLinks() && Utils.isNetworkConnected()) {
+ // Show a progress spinner, but only if the api fetch takes more than a half a second.
+ final long delayToShowProgressSpinner = 500;
ProgressDialog progress = new ProgressDialog(getContext());
progress.setProgressStyle(ProgressDialog.STYLE_SPINNER);
- progress.show();
- Utils.runOnBackgroundThread(() -> fetchLinksAndShowDialog(progress));
+
+ Handler handler = new Handler(Looper.getMainLooper());
+ Runnable showDialogRunnable = progress::show;
+ handler.postDelayed(showDialogRunnable, delayToShowProgressSpinner);
+
+ Utils.runOnBackgroundThread(() ->
+ fetchLinksAndShowDialog(handler, showDialogRunnable, progress));
} else {
// No network call required and can run now.
- fetchLinksAndShowDialog(null);
+ fetchLinksAndShowDialog(null, null, null);
}
return false;
});
}
- private void fetchLinksAndShowDialog(@Nullable ProgressDialog progress) {
- WebLink[] socialLinks = SocialLinksRoutes.fetchSocialLinks();
- String htmlDialog = createDialogHtml(socialLinks);
+ private void fetchLinksAndShowDialog(@Nullable Handler handler,
+ Runnable showDialogRunnable,
+ @Nullable ProgressDialog progress) {
+ WebLink[] links = AboutLinksRoutes.fetchAboutLinks();
+ String htmlDialog = createDialogHtml(links);
+
+ // Enable to randomly force a delay to debug the spinner logic.
+ final boolean debugSpinnerDelayLogic = false;
+ //noinspection ConstantConditions
+ if (debugSpinnerDelayLogic && handler != null && Math.random() < 0.5f) {
+ Utils.doNothingForDuration((long) (Math.random() * 4000));
+ }
Utils.runOnMainThreadNowOrLater(() -> {
+ if (handler != null) {
+ handler.removeCallbacks(showDialogRunnable);
+ }
if (progress != null) {
progress.dismiss();
}
@@ -224,7 +253,7 @@ public boolean shouldOverrideUrlLoading(WebView view, String url) {
class WebLink {
final boolean preferred;
- final String name;
+ String name;
final String url;
WebLink(JSONObject json) throws JSONException {
@@ -243,7 +272,7 @@ class WebLink {
@NonNull
@Override
public String toString() {
- return "ReVancedSocialLink{" +
+ return "WebLink{" +
"preferred=" + preferred +
", name='" + name + '\'' +
", url='" + url + '\'' +
@@ -251,25 +280,21 @@ public String toString() {
}
}
-class SocialLinksRoutes {
+class AboutLinksRoutes {
/**
- * Simple link to the website donate page,
- * rather than fetching and parsing the donation links using the API.
+ * Backup icon url if the API call fails.
*/
- public static final WebLink DONATE_LINK = new WebLink(true,
- sf("revanced_settings_about_links_donate").toString(),
- "https://revanced.app/donate");
+ public static volatile String aboutLogoUrl = "https://revanced.app/favicon.ico";
/**
* Links to use if fetch links api call fails.
*/
private static final WebLink[] NO_CONNECTION_STATIC_LINKS = {
- new WebLink(true, "ReVanced.app", "https://revanced.app"),
- DONATE_LINK,
+ new WebLink(true, "ReVanced.app", "https://revanced.app")
};
- private static final String SOCIAL_LINKS_PROVIDER = "https://api.revanced.app/v2";
- private static final Route.CompiledRoute GET_SOCIAL = new Route(GET, "/socials").compile();
+ private static final String SOCIAL_LINKS_PROVIDER = "https://api.revanced.app/v4";
+ private static final Route.CompiledRoute GET_SOCIAL = new Route(GET, "/about").compile();
@Nullable
private static volatile WebLink[] fetchedLinks;
@@ -278,7 +303,7 @@ static boolean hasFetchedLinks() {
return fetchedLinks != null;
}
- static WebLink[] fetchSocialLinks() {
+ static WebLink[] fetchAboutLinks() {
try {
if (hasFetchedLinks()) return fetchedLinks;
@@ -298,11 +323,22 @@ static WebLink[] fetchSocialLinks() {
}
JSONObject json = Requester.parseJSONObjectAndDisconnect(connection);
- JSONArray socials = json.getJSONArray("socials");
+ aboutLogoUrl = json.getJSONObject("branding").getString("logo");
List links = new ArrayList<>();
- links.add(DONATE_LINK); // Show donate link first.
+ JSONArray donations = json.getJSONObject("donations").getJSONArray("links");
+ for (int i = 0, length = donations.length(); i < length; i++) {
+ WebLink link = new WebLink(donations.getJSONObject(i));
+ if (link.preferred) {
+ // This could be localized, but TikTok does not support localized resources.
+ // All link names returned by the api are also non localized.
+ link.name = "Donate";
+ links.add(link);
+ }
+ }
+
+ JSONArray socials = json.getJSONArray("socials");
for (int i = 0, length = socials.length(); i < length; i++) {
WebLink link = new WebLink(socials.getJSONObject(i));
links.add(link);
diff --git a/extensions/shared/src/main/java/app/revanced/extension/tiktok/settings/preference/ReVancedTikTokAboutPreference.java b/extensions/shared/src/main/java/app/revanced/extension/tiktok/settings/preference/ReVancedTikTokAboutPreference.java
new file mode 100644
index 0000000000..f20a0b6350
--- /dev/null
+++ b/extensions/shared/src/main/java/app/revanced/extension/tiktok/settings/preference/ReVancedTikTokAboutPreference.java
@@ -0,0 +1,56 @@
+package app.revanced.extension.tiktok.settings.preference;
+
+import android.content.Context;
+import android.util.AttributeSet;
+
+import java.util.Map;
+
+import app.revanced.extension.shared.Logger;
+import app.revanced.extension.shared.settings.preference.ReVancedAboutPreference;
+
+@SuppressWarnings("unused")
+public class ReVancedTikTokAboutPreference extends ReVancedAboutPreference {
+
+ /**
+ * Because resources cannot be added to TikTok,
+ * these strings are copied from the shared strings.xml file.
+ *
+ * Changes here must also be made in strings.xml
+ */
+ private final Map aboutStrings = Map.of(
+ "revanced_settings_about_links_body", "You are using ReVanced Patches version %s",
+ "revanced_settings_about_links_dev_header", "Note",
+ "revanced_settings_about_links_dev_body", "This version is a pre-release and you may experience unexpected issues",
+ "revanced_settings_about_links_header", "Official links"
+ );
+
+ {
+ //noinspection deprecation
+ setTitle("About");
+ }
+
+ public ReVancedTikTokAboutPreference(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
+ super(context, attrs, defStyleAttr, defStyleRes);
+ }
+ public ReVancedTikTokAboutPreference(Context context, AttributeSet attrs, int defStyleAttr) {
+ super(context, attrs, defStyleAttr);
+ }
+ public ReVancedTikTokAboutPreference(Context context, AttributeSet attrs) {
+ super(context, attrs);
+ }
+ public ReVancedTikTokAboutPreference(Context context) {
+ super(context);
+ }
+
+ @Override
+ protected String getString(String key, Object ... args) {
+ String format = aboutStrings.get(key);
+
+ if (format == null) {
+ Logger.printException(() -> "Unknown key: " + key);
+ return "";
+ }
+
+ return String.format(format, args);
+ }
+}
diff --git a/extensions/shared/src/main/java/app/revanced/extension/tiktok/settings/preference/categories/ExtensionPreferenceCategory.java b/extensions/shared/src/main/java/app/revanced/extension/tiktok/settings/preference/categories/ExtensionPreferenceCategory.java
index ad49df688b..60d7983ea0 100644
--- a/extensions/shared/src/main/java/app/revanced/extension/tiktok/settings/preference/categories/ExtensionPreferenceCategory.java
+++ b/extensions/shared/src/main/java/app/revanced/extension/tiktok/settings/preference/categories/ExtensionPreferenceCategory.java
@@ -4,13 +4,14 @@
import android.preference.PreferenceScreen;
import app.revanced.extension.shared.settings.BaseSettings;
+import app.revanced.extension.tiktok.settings.preference.ReVancedTikTokAboutPreference;
import app.revanced.extension.tiktok.settings.preference.TogglePreference;
@SuppressWarnings("deprecation")
public class ExtensionPreferenceCategory extends ConditionalPreferenceCategory {
public ExtensionPreferenceCategory(Context context, PreferenceScreen screen) {
super(context, screen);
- setTitle("Extension");
+ setTitle("Miscellaneous");
}
@Override
@@ -20,6 +21,8 @@ public boolean getSettingsStatus() {
@Override
public void addPreferences(Context context) {
+ addPreference(new ReVancedTikTokAboutPreference(context));
+
addPreference(new TogglePreference(context,
"Enable debug log",
"Show extension debug log.",
diff --git a/patches/src/main/resources/addresources/values/strings.xml b/patches/src/main/resources/addresources/values/strings.xml
index 86fb2405cc..c8bee767d8 100644
--- a/patches/src/main/resources/addresources/values/strings.xml
+++ b/patches/src/main/resources/addresources/values/strings.xml
@@ -60,7 +60,8 @@ This is because Crowdin requires temporarily flattening this file and removing t
Note
This version is a pre-release and you may experience unexpected issues
Official links
- Donate
+