mirror of
https://github.com/revanced/revanced-integrations.git
synced 2024-06-14 09:00:04 +02:00
dbf1f3f0ce
Co-authored-by: oSumAtrIX <johan.melkonyan1@web.de>
191 lines
6.5 KiB
Java
191 lines
6.5 KiB
Java
package app.revanced.integrations.utils;
|
|
|
|
import android.annotation.SuppressLint;
|
|
import android.content.Context;
|
|
import android.content.res.Resources;
|
|
import android.os.Handler;
|
|
import android.os.Looper;
|
|
|
|
import androidx.annotation.NonNull;
|
|
|
|
import java.text.Bidi;
|
|
import java.util.Locale;
|
|
import java.util.concurrent.Callable;
|
|
import java.util.concurrent.Future;
|
|
import java.util.concurrent.LinkedBlockingQueue;
|
|
import java.util.concurrent.ThreadFactory;
|
|
import java.util.concurrent.ThreadPoolExecutor;
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
import app.revanced.integrations.sponsorblock.player.PlayerType;
|
|
|
|
public class ReVancedUtils {
|
|
|
|
private static PlayerType env;
|
|
private static boolean newVideo = false;
|
|
|
|
@SuppressLint("StaticFieldLeak")
|
|
public static Context context;
|
|
|
|
private ReVancedUtils() {
|
|
} // utility class
|
|
|
|
/**
|
|
* Maximum number of background threads run concurrently
|
|
*/
|
|
private static final int SHARED_THREAD_POOL_MAXIMUM_BACKGROUND_THREADS = 20;
|
|
|
|
/**
|
|
* General purpose pool for network calls and other background tasks.
|
|
* All tasks run at max thread priority.
|
|
*/
|
|
private static final ThreadPoolExecutor backgroundThreadPool = new ThreadPoolExecutor(
|
|
1, // minimum 1 thread always ready to be used
|
|
10, // For any threads over the minimum, keep them alive 10 seconds after they go idle
|
|
SHARED_THREAD_POOL_MAXIMUM_BACKGROUND_THREADS,
|
|
TimeUnit.SECONDS,
|
|
new LinkedBlockingQueue<Runnable>(),
|
|
new ThreadFactory() {
|
|
@Override
|
|
public Thread newThread(Runnable r) {
|
|
Thread t = new Thread(r);
|
|
t.setPriority(Thread.MAX_PRIORITY); // run at max priority
|
|
return t;
|
|
}
|
|
});
|
|
|
|
private static void checkIfPoolHasReachedLimit() {
|
|
if (backgroundThreadPool.getActiveCount() >= SHARED_THREAD_POOL_MAXIMUM_BACKGROUND_THREADS) {
|
|
// Something is wrong. Background threads are piling up and not completing as expected,
|
|
// or some ReVanced code is submitting an unexpected number of background tasks.
|
|
LogHelper.printException(() -> "Reached maximum background thread count of "
|
|
+ SHARED_THREAD_POOL_MAXIMUM_BACKGROUND_THREADS + " threads");
|
|
}
|
|
}
|
|
|
|
public static void runOnBackgroundThread(Runnable task) {
|
|
backgroundThreadPool.execute(task);
|
|
checkIfPoolHasReachedLimit();
|
|
}
|
|
|
|
public static <T> Future<T> submitOnBackgroundThread(Callable<T> call) {
|
|
Future<T> future = backgroundThreadPool.submit(call);
|
|
checkIfPoolHasReachedLimit();
|
|
return future;
|
|
}
|
|
|
|
public static boolean containsAny(final String value, final String... targets) {
|
|
for (String string : targets)
|
|
if (!string.isEmpty() && value.contains(string)) return true;
|
|
return false;
|
|
}
|
|
|
|
public static void setNewVideo(boolean started) {
|
|
LogHelper.printDebug(() -> "New video started: " + started);
|
|
newVideo = started;
|
|
}
|
|
|
|
public static boolean isNewVideoStarted() {
|
|
return newVideo;
|
|
}
|
|
|
|
public static Integer getResourceIdByName(Context context, String type, String name) {
|
|
try {
|
|
Resources res = context.getResources();
|
|
return res.getIdentifier(name, type, context.getPackageName());
|
|
} catch (Throwable exception) {
|
|
LogHelper.printException(() -> "Resource not found.", exception);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public static void setPlayerType(PlayerType type) {
|
|
env = type;
|
|
}
|
|
|
|
public static PlayerType getPlayerType() {
|
|
return env;
|
|
}
|
|
|
|
public static int getIdentifier(String name, String defType) {
|
|
Context context = getContext();
|
|
return context.getResources().getIdentifier(name, defType, context.getPackageName());
|
|
}
|
|
|
|
public static Context getContext() {
|
|
if (context != null) {
|
|
return context;
|
|
} else {
|
|
LogHelper.printException(() -> "Context is null, returning null!");
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public static void setClipboard(String text) {
|
|
android.content.ClipboardManager clipboard = (android.content.ClipboardManager) context.getSystemService(Context.CLIPBOARD_SERVICE);
|
|
android.content.ClipData clip = android.content.ClipData.newPlainText("ReVanced", text);
|
|
clipboard.setPrimaryClip(clip);
|
|
}
|
|
|
|
public static boolean isTablet(Context context) {
|
|
return context.getResources().getConfiguration().smallestScreenWidthDp >= 600;
|
|
}
|
|
|
|
private static Boolean isRightToLeftTextLayout;
|
|
/**
|
|
* If the device language uses right to left text layout (hebrew, arabic, etc)
|
|
*/
|
|
public static boolean isRightToLeftTextLayout() {
|
|
if (isRightToLeftTextLayout == null) {
|
|
String displayLanguage = Locale.getDefault().getDisplayLanguage();
|
|
isRightToLeftTextLayout = new Bidi(displayLanguage, Bidi.DIRECTION_DEFAULT_LEFT_TO_RIGHT).isRightToLeft();
|
|
}
|
|
return isRightToLeftTextLayout;
|
|
}
|
|
|
|
/**
|
|
* Automatically logs any exceptions the runnable throws
|
|
*/
|
|
public static void runOnMainThread(Runnable runnable) {
|
|
runOnMainThreadDelayed(runnable, 0);
|
|
}
|
|
|
|
/**
|
|
* Automatically logs any exceptions the runnable throws
|
|
*/
|
|
public static void runOnMainThreadDelayed(Runnable runnable, long delayMillis) {
|
|
Runnable loggingRunnable = () -> {
|
|
try {
|
|
runnable.run();
|
|
} catch (Exception ex) {
|
|
LogHelper.printException(() -> runnable.getClass() + ": " + ex.getMessage(), ex);
|
|
}
|
|
};
|
|
new Handler(Looper.getMainLooper()).postDelayed(loggingRunnable, delayMillis);
|
|
}
|
|
|
|
/**
|
|
* @return if the calling thread is on the main thread
|
|
*/
|
|
public static boolean currentlyIsOnMainThread() {
|
|
return Looper.getMainLooper().isCurrentThread();
|
|
}
|
|
|
|
/**
|
|
* @throws IllegalStateException if the calling thread is _off_ the main thread
|
|
*/
|
|
public static void verifyOnMainThread() throws IllegalStateException {
|
|
if (!currentlyIsOnMainThread()) {
|
|
throw new IllegalStateException("Must call _on_ the main thread");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @throws IllegalStateException if the calling thread is _on_ the main thread
|
|
*/
|
|
public static void verifyOffMainThread() throws IllegalStateException {
|
|
if (currentlyIsOnMainThread()) {
|
|
throw new IllegalStateException("Must call _off_ the main thread");
|
|
}
|
|
}
|
|
} |