JobService: Transition to the Binder Interprocess Communication System
With this transition, we are deprecating the implicit intent
broadcasting system in preparation of Android O.
For more information regarding this API change:
https://commonsware.com/blog/2017/04/11/android-o-implicit-broadcast-ban.html
A brief explanation on Binder (thanks to @Surge1223):
Binder is generally a shorthand for IPC binder, which means interprocess
communication.
This usually involves a client and a service, and communicate through
the kernel via the /dev/binder of the kernel. In order for them to
communicate, the client and remote service needs a way to interface,
which is where the Android Interface Description Language is used.
The client and remote service must both implement an AIDL file,
which generates a stub and proxy that can be used to communicate with
each other.The client usually sends a parcelized stub proxy to
/dev/binder and the service unparcelizes the message via its stub
interface.
Change-Id: I23057a091890614c6d132be52cc6beca43ede200
diff --git a/Android.mk b/Android.mk
index e36c8bf..9ce44f8 100644
--- a/Android.mk
+++ b/Android.mk
@@ -12,7 +12,7 @@
LOCAL_MODULE_TAGS := optional
LOCAL_MANIFEST_FILE := app/src/main/AndroidManifest.xml
-LOCAL_SRC_FILES := $(call all-java-files-under, app/src/main)
+LOCAL_SRC_FILES := $(call all-java-files-under, app/src/main) $(call all-Iaidl-files-under, app/src/main)
LOCAL_REQUIRED_MODULES := projekt.substratum.theme.xml
LOCAL_RESOURCE_DIR := $(LOCAL_PATH)/app/src/main/res
diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml
index e2709b6..007e530 100644
--- a/app/src/main/AndroidManifest.xml
+++ b/app/src/main/AndroidManifest.xml
@@ -4,8 +4,8 @@
coreApp="true"
package="projekt.interfacer"
android:sharedUserId="android.uid.system"
- android:versionCode="56"
- android:versionName="fifty six"
+ android:versionCode="60"
+ android:versionName="sixty"
tools:ignore="GradleOverrides">
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
@@ -37,6 +37,10 @@
<service
android:name=".services.JobService"
android:exported="true"
- android:permission="android.permission.MODIFY_OVERLAYS"/>
+ android:permission="android.permission.MODIFY_OVERLAYS">
+ <intent-filter>
+ <action android:name="projekt.interfacer.INITIALIZE" />
+ </intent-filter>
+ </service>
</application>
</manifest>
diff --git a/app/src/main/aidl/projekt/substratum/IInterfacerInterface.aidl b/app/src/main/aidl/projekt/substratum/IInterfacerInterface.aidl
new file mode 100644
index 0000000..db7f0e9
--- /dev/null
+++ b/app/src/main/aidl/projekt/substratum/IInterfacerInterface.aidl
@@ -0,0 +1,134 @@
+/*
+ * Copyright (c) 2016-2017 Projekt Substratum
+ * This file is part of Substratum.
+ *
+ * Substratum is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Substratum is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with Substratum. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package projekt.substratum;
+
+interface IInterfacerInterface {
+
+ /**
+ * Install a list of specified applications
+ *
+ * @param paths Filled in with a list of path names for packages to be installed from.
+ */
+ void installPackage(in List<String> paths);
+
+ /**
+ * Uninstall a list of specified applications
+ *
+ * @param packages Filled in with a list of path names for packages to be installed from.
+ * @param restartUi Flag to automatically restart the SystemUI.
+ */
+ void uninstallPackage(in List<String> packages, boolean restartUi);
+
+ /**
+ * Restart SystemUI
+ */
+ void restartSystemUI();
+
+ /**
+ * Perform a low-level configuration change
+ */
+ void configurationShim();
+
+ /**
+ * Apply a specified bootanimation
+ *
+ * @param name Path to extract the bootanimation archive from.
+ */
+ void applyBootanimation(String name);
+
+ /**
+ * Apply a specified font pack
+ *
+ * @param name Path to extract the font archive from.
+ */
+ void applyFonts(String pid, String fileName);
+
+ /**
+ * Apply a specified sound pack
+ *
+ * @param name Path to extract the font archive from.
+ */
+ void applyAudio(String pid, String fileName);
+
+ /**
+ * Enable a specified list of overlays
+ *
+ * @param packages Filled in with a list of package names to be enabled.
+ * @param restartUi Flag to automatically restart the SystemUI.
+ */
+ void enableOverlay(in List<String> packages, boolean restartUi);
+
+ /**
+ * Disable a specified list of overlays
+ *
+ * @param packages Filled in with a list of package names to be disabled.
+ * @param restartUi Flag to automatically restart the SystemUI.
+ */
+ void disableOverlay(in List<String> packages, boolean restartUi);
+
+ /**
+ * Change the priority of a specified list of overlays
+ *
+ * @param packages Filled in with a list of package names to be reordered.
+ * @param restartUi Flag to automatically restart the SystemUI.
+ */
+ void changePriority(in List<String> packages, boolean restartUi);
+
+ /**
+ * Copy Method
+ *
+ * @param source Path of the source file.
+ * @param destination Path of the source file to be copied to.
+ */
+ void copy(String source, String destination);
+
+ /**
+ * Move Method
+ *
+ * @param source Path of the source file.
+ * @param destination Path of the source file to be moved to.
+ */
+ void move(String source, String destination);
+
+ /**
+ * Create Directory Method
+ *
+ * @param destination Path of the created destination folder.
+ */
+ void mkdir(String destination);
+
+ /**
+ * Delete Directory Method
+ *
+ * @param destination Path of the directory to be deleted.
+ * @param withParent Flag to automatically delete the folder encompassing the folder.
+ */
+ void deleteDirectory(String directory, boolean withParent);
+
+ /**
+ * Profile Applicator
+ *
+ * @param enable Filled in with a list of package names to be enabled.
+ * @param disable Filled in with a list of package names to be disabled.
+ * @param name Name of the profile to be applied.
+ * @param restartUi Flag to automatically restart the SystemUI.
+ */
+ void applyProfile(in List<String> enable, in List<String> disable, String name,
+ boolean restartUi);
+}
\ No newline at end of file
diff --git a/app/src/main/java/projekt/interfacer/services/JobService.java b/app/src/main/java/projekt/interfacer/services/JobService.java
index 83f2dc0..3c0e038 100644
--- a/app/src/main/java/projekt/interfacer/services/JobService.java
+++ b/app/src/main/java/projekt/interfacer/services/JobService.java
@@ -20,7 +20,6 @@
import android.app.ActivityManager;
import android.app.ActivityManagerNative;
-import android.app.PendingIntent;
import android.app.Service;
import android.content.BroadcastReceiver;
import android.content.ComponentName;
@@ -40,16 +39,13 @@
import android.content.res.Configuration;
import android.graphics.Typeface;
import android.media.RingtoneManager;
+import android.os.Binder;
import android.os.Build;
import android.os.Bundle;
import android.os.Environment;
import android.os.FileUtils;
import android.os.Handler;
-import android.os.HandlerThread;
import android.os.IBinder;
-import android.os.Looper;
-import android.os.Message;
-import android.os.Process;
import android.os.RemoteException;
import android.os.ServiceManager;
import android.os.SystemProperties;
@@ -63,77 +59,346 @@
import java.io.InputStream;
import java.io.OutputStream;
import java.lang.reflect.Method;
-import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Locale;
-import java.util.stream.Collectors;
import projekt.interfacer.utils.IOUtils;
import projekt.interfacer.utils.SoundUtils;
+import projekt.substratum.IInterfacerInterface;
public class JobService extends Service {
- public static final String INTENT_STATUS_CHANGED = "projekt.interfacer.STATUS_CHANGED";
- public static final String PRIMARY_COMMAND_KEY = "primary_command_key";
- public static final String JOB_TIME_KEY = "job_time_key";
- public static final String INSTALL_LIST_KEY = "install_list";
- public static final String UNINSTALL_LIST_KEY = "uninstall_list";
- public static final String WITH_RESTART_UI_KEY = "with_restart_ui";
- public static final String BOOTANIMATION_FILE_NAME = "bootanimation_file_name";
- public static final String FONTS_PID = "fonts_pid";
- public static final String FONTS_FILENAME = "fonts_filename";
- public static final String AUDIO_PID = "audio_pid";
- public static final String AUDIO_FILENAME = "audio_filename";
- public static final String ENABLE_LIST_KEY = "enable_list";
- public static final String DISABLE_LIST_KEY = "disable_list";
- public static final String PRIORITY_LIST_KEY = "priority_list";
- public static final String SOURCE_FILE_KEY = "source_file";
- public static final String DESTINATION_FILE_KEY = "destination_file";
- public static final String WITH_DELETE_PARENT_KEY = "delete_parent";
- public static final String PROFILE_NAME_KEY = "profile_name";
- public static final String COMMAND_VALUE_JOB_COMPLETE = "job_complete";
- public static final String COMMAND_VALUE_INSTALL = "install";
- public static final String COMMAND_VALUE_UNINSTALL = "uninstall";
- public static final String COMMAND_VALUE_RESTART_UI = "restart_ui";
- public static final String COMMAND_VALUE_FORCE_STOP_SERVICE = "force_stop_service";
- public static final String COMMAND_VALUE_CONFIGURATION_SHIM = "configuration_shim";
- public static final String COMMAND_VALUE_BOOTANIMATION = "bootanimation";
- public static final String COMMAND_VALUE_FONTS = "fonts";
- public static final String COMMAND_VALUE_AUDIO = "audio";
- public static final String COMMAND_VALUE_ENABLE = "enable";
- public static final String COMMAND_VALUE_DISABLE = "disable";
- public static final String COMMAND_VALUE_PRIORITY = "priority";
- public static final String COMMAND_VALUE_COPY = "copy";
- public static final String COMMAND_VALUE_MOVE = "move";
- public static final String COMMAND_VALUE_DELETE = "delete";
- public static final String COMMAND_VALUE_PROFILE = "profile";
- public static final String COMMAND_VALUE_MKDIR = "mkdir";
private static final String TAG = JobService.class.getSimpleName();
private static final boolean DEBUG = true;
- private static final String INTERFACER_TOKEN = "interfacer_token";
+ private static final String INTENT_STATUS_CHANGED = "projekt.interfacer.STATUS_CHANGED";
+ private static final String PRIMARY_COMMAND_KEY = "primary_command_key";
+ private static final String COMMAND_VALUE_JOB_COMPLETE = "job_complete";
+ private static final String COMMAND_VALUE_BOOTANIMATION = "bootanimation";
+ private static final String COMMAND_VALUE_FONTS = "fonts";
+ private static final String COMMAND_VALUE_AUDIO = "audio";
private static final String INTERFACER_PACKAGE = "projekt.interfacer";
private static final String SUBSTRATUM_PACKAGE = "projekt.substratum";
private static final String[] AUTHORIZED_CALLERS = new String[]{
INTERFACER_PACKAGE,
SUBSTRATUM_PACKAGE,
};
- private static List<Sound> SOUNDS = Arrays.asList(
- new Sound(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH, "/SoundsCache/ui/", "Effect_Tick", "Effect_Tick", RingtoneManager.TYPE_RINGTONE),
- new Sound(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH, "/SoundsCache/ui/", "lock_sound", "Lock"),
- new Sound(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH, "/SoundsCache/ui/", "unlock_sound", "Unlock"),
- new Sound(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH, "/SoundsCache/ui/", "low_battery_sound", "LowBattery"),
- new Sound(IOUtils.SYSTEM_THEME_ALARM_PATH, "/SoundsCache/alarms/", "alarm", "alarm", RingtoneManager.TYPE_ALARM),
- new Sound(IOUtils.SYSTEM_THEME_NOTIFICATION_PATH, "/SoundsCache/notifications/", "notification", "notification", RingtoneManager.TYPE_NOTIFICATION),
- new Sound(IOUtils.SYSTEM_THEME_RINGTONE_PATH, "/SoundsCache/ringtones/", "ringtone", "ringtone", RingtoneManager.TYPE_RINGTONE)
- );
private static final String INTENT_CALLER_AUTHORIZED = "projekt.interfacer.CALLER_AUTHORIZED";
+ private static final Signature SUBSTRATUM_SIGNATURE = new Signature(""
+ + "308202eb308201d3a003020102020411c02f2f300d06092a864886f70d01010b050030263124302206"
+ + "03550403131b5375627374726174756d20446576656c6f706d656e74205465616d301e170d31363037"
+ + "30333032333335385a170d3431303632373032333335385a3026312430220603550403131b53756273"
+ + "74726174756d20446576656c6f706d656e74205465616d30820122300d06092a864886f70d01010105"
+ + "000382010f003082010a02820101008855626336f645a335aa5d40938f15db911556385f72f72b5f8b"
+ + "ad01339aaf82ae2d30302d3f2bba26126e8da8e76a834e9da200cdf66d1d5977c90a4e4172ce455704"
+ + "a22bbe4a01b08478673b37d23c34c8ade3ec040a704da8570d0a17fce3c7397ea63ebcde3a2a3c7c5f"
+ + "983a163e4cd5a1fc80c735808d014df54120e2e5708874739e22e5a22d50e1c454b2ae310b480825ab"
+ + "3d877f675d6ac1293222602a53080f94e4a7f0692b627905f69d4f0bb1dfd647e281cc0695e0733fa3"
+ + "efc57d88706d4426c4969aff7a177ac2d9634401913bb20a93b6efe60e790e06dad3493776c2c0878c"
+ + "e82caababa183b494120edde3d823333efd464c8aea1f51f330203010001a321301f301d0603551d0e"
+ + "04160414203ec8b075d1c9eb9d600100281c3924a831a46c300d06092a864886f70d01010b05000382"
+ + "01010042d4bd26d535ce2bf0375446615ef5bf25973f61ecf955bdb543e4b6e6b5d026fdcab09fec09"
+ + "c747fb26633c221df8e3d3d0fe39ce30ca0a31547e9ec693a0f2d83e26d231386ff45f8e4fd5c06095"
+ + "8681f9d3bd6db5e940b1e4a0b424f5c463c79c5748a14a3a38da4dd7a5499dcc14a70ba82a50be5fe0"
+ + "82890c89a27e56067d2eae952e0bcba4d6beb5359520845f1fdb7df99868786055555187ba46c69ee6"
+ + "7fa2d2c79e74a364a8b3544997dc29cc625395e2f45bf8bdb2c9d8df0d5af1a59a58ad08b32cdbec38"
+ + "19fa49201bb5b5aadeee8f2f096ac029055713b77054e8af07cd61fe97f7365d0aa92d570be98acb89"
+ + "41b8a2b0053b54f18bfde092eb");
+ private static final Signature[] AUTHORIZED_SIGNATURES = new Signature[]{
+ SUBSTRATUM_SIGNATURE,
+ };
+ private static List<Sound> SOUNDS = Arrays.asList(
+ new Sound(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH, "/SoundsCache/ui/", "Effect_Tick",
+ "Effect_Tick", RingtoneManager.TYPE_RINGTONE),
+ new Sound(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH, "/SoundsCache/ui/", "lock_sound",
+ "Lock"),
+ new Sound(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH, "/SoundsCache/ui/", "unlock_sound",
+ "Unlock"),
+ new Sound(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH, "/SoundsCache/ui/",
+ "low_battery_sound", "LowBattery"),
+ new Sound(IOUtils.SYSTEM_THEME_ALARM_PATH, "/SoundsCache/alarms/", "alarm", "alarm",
+ RingtoneManager.TYPE_ALARM),
+ new Sound(IOUtils.SYSTEM_THEME_NOTIFICATION_PATH, "/SoundsCache/notifications/",
+ "notification", "notification", RingtoneManager.TYPE_NOTIFICATION),
+ new Sound(IOUtils.SYSTEM_THEME_RINGTONE_PATH, "/SoundsCache/ringtones/", "ringtone",
+ "ringtone", RingtoneManager.TYPE_RINGTONE)
+ );
private static IOverlayManager mOMS;
private static IPackageManager mPM;
- private final List<Runnable> mJobQueue = new ArrayList<>(0);
- private JobHandler mJobHandler;
- private MainHandler mMainHandler;
- private long mLastJobTime;
- private boolean mIsRunning;
+ private static boolean isWaiting = false;
+ private final IInterfacerInterface.Stub mBinder = new IInterfacerInterface.Stub() {
+ @Override
+ public void installPackage(List<String> paths) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ for (String path : paths) {
+ log("Installer - installing \'" + path + "\'...");
+ PackageInstallObserver observer = new PackageInstallObserver();
+ install(path, observer);
+ }
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void uninstallPackage(List<String> packages, boolean restartUi) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ for (String p : packages) {
+ // TODO: Fix isOverlayEnabled function, for now it's causing NPE
+ if (isOverlayEnabled(p)) {
+ log("Remover - disabling overlay for \'" + p + "\'...");
+ switchOverlay(p, false);
+ }
+
+ log("Remover - uninstalling \'" + p + "\'...");
+ PackageDeleteObserver observer = new PackageDeleteObserver();
+ uninstall(p, observer);
+ }
+ if (restartUi) restartUi();
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void restartSystemUI() {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ log("Restarting SystemUI...");
+ restartUi();
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void configurationShim() {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ new LocaleChanger(getSubsContext()).run();
+ }
+
+ @Override
+ public void applyBootanimation(String name) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ if (name == null) {
+ log("Restoring system boot animation...");
+ clearBootAnimation();
+ } else {
+ log("Configuring themed boot animation...");
+ copyBootAnimation(name);
+ }
+ informCompletion(COMMAND_VALUE_BOOTANIMATION);
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void applyFonts(String pid, String fileName) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ if (pid == null) {
+ log("Restoring system font...");
+ clearFonts();
+ } else {
+ log("Configuring theme font...");
+ copyFonts(pid, fileName);
+ }
+ restartUi();
+ informCompletion(COMMAND_VALUE_FONTS);
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void applyAudio(String pid, String fileName) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ if (pid == null) {
+ log("Restoring system sounds...");
+ clearSounds(JobService.this);
+ } else {
+ log("Configuring theme sounds...");
+ applyThemedSounds(pid, fileName);
+ }
+ restartUi();
+ informCompletion(COMMAND_VALUE_AUDIO);
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void enableOverlay(List<String> packages, boolean restartUi) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ for (String p : packages) {
+ log("Enabler - enabling overlay for \'" + p + "\'...");
+ switchOverlay(p, true);
+ }
+ if (restartUi) restartUi();
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void disableOverlay(List<String> packages, boolean restartUi) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ for (String p : packages) {
+ log("Disabler - disabling overlay for \'" + p + "\'...");
+ switchOverlay(p, false);
+ }
+ if (restartUi) restartUi();
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void changePriority(List<String> packages, boolean restartUi) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ log("PriorityJob - processing priority changes...");
+ try {
+ for (int i = 0; i < packages.size() - 1; i++) {
+ String parentName = packages.get(i);
+ String packageName = packages.get(i + 1);
+
+ getOMS().setPriority(packageName, parentName,
+ UserHandle.USER_SYSTEM);
+ }
+ } catch (RemoteException e) {
+ // Exception
+ }
+ if (restartUi) restartUi();
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void copy(String source, String destination) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ log("CopyJob - copying \'" + source + "\' to \'" + destination +
+ "\'...");
+
+ File sourceFile = new File(source);
+ if (sourceFile.exists()) {
+ if (sourceFile.isFile()) {
+ IOUtils.bufferedCopy(source, destination);
+ } else {
+ IOUtils.copyFolder(source, destination);
+ }
+ } else {
+ log("CopyJob - \'" + source + "\' does not exist, aborting...");
+ }
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void move(String source, String destination) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ log("MoveJob - moving \'" + source + "\' to \'" + destination +
+ "\'...");
+
+ File sourceFile = new File(source);
+ if (sourceFile.exists()) {
+ if (sourceFile.isFile()) {
+ IOUtils.bufferedCopy(source, destination);
+ } else {
+ IOUtils.copyFolder(source, destination);
+ }
+ IOUtils.deleteRecursive(sourceFile);
+ } else {
+ log("MoveJob - \'" + source + "\' does not exist, aborting...");
+ }
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void mkdir(String destination) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ log("MkdirJob - creating \'" + destination + "\'...");
+ IOUtils.createDirIfNotExists(destination);
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void deleteDirectory(String directory, boolean withParent) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ if (withParent) {
+ delete(directory);
+ } else {
+ for (File child : new File(directory).listFiles()) {
+ delete(child.getAbsolutePath());
+ }
+ }
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+
+ @Override
+ public void applyProfile(List<String> enable, List<String> disable,
+ String name, boolean restartUi) {
+ // Verify caller identity
+ if (!isCallerAuthorized(Binder.getCallingUid())) return;
+
+ log("Applying profile...");
+ boolean mRestartUi = restartUi;
+
+ // Clear system theme folder content
+ File themeDir = new File(IOUtils.SYSTEM_THEME_PATH);
+ for (File f : themeDir.listFiles()) {
+ IOUtils.deleteRecursive(f);
+ }
+
+ // Process theme folder
+ File profileDir = new File(Environment.getExternalStorageDirectory()
+ .getAbsolutePath() + "/substratum/profiles/" +
+ name + "/theme");
+
+ if (profileDir.exists()) {
+ File profileFonts = new File(profileDir, "fonts");
+ if (profileFonts.exists()) {
+ IOUtils.copyFolder(profileFonts, new File(IOUtils.SYSTEM_THEME_FONT_PATH));
+ refreshFonts();
+ mRestartUi = true;
+ } else {
+ clearFonts();
+ }
+
+ File profileSounds = new File(profileDir, "audio");
+ if (profileSounds.exists()) {
+ IOUtils.copyFolder(profileSounds, new File(IOUtils.SYSTEM_THEME_AUDIO_PATH));
+ refreshSounds();
+ mRestartUi = true;
+ } else {
+ clearSounds(JobService.this);
+ }
+ }
+
+ // Disable all overlays installed
+ for (String overlay : disable) {
+ switchOverlay(overlay, false);
+ }
+
+ // Enable provided overlays
+ for (String overlay : enable) {
+ switchOverlay(overlay, true);
+ }
+
+ // Restart SystemUI when needed
+ if (mRestartUi) {
+ restartUi();
+ }
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
+ }
+ };
private static IOverlayManager getOMS() {
if (mOMS == null) {
@@ -161,173 +426,51 @@
Log.d(TAG, msg);
}
+ private static boolean isDebugBuild() {
+ return Build.TYPE.equals("eng") || Build.TYPE.equals("userdebug");
+ }
+
+ private static boolean shouldCheckBuildType() {
+ return SystemProperties.getBoolean("ro.interfacer.buildtype.check", false);
+ }
+
@Override
public void onCreate() {
- HandlerThread mWorker = new HandlerThread("BackgroundWorker", Process
- .THREAD_PRIORITY_BACKGROUND);
- mWorker.start();
- mJobHandler = new JobHandler(mWorker.getLooper());
- mMainHandler = new MainHandler(Looper.getMainLooper());
-
// Needed here before any checks
IOUtils.createThemeDirIfNotExists();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
- // Verify identity
- boolean authorized = isCallerAuthorized(intent);
-
- // Broadcast authorization result
- informCaller(authorized);
-
- if (!authorized) {
- log("caller not authorized, aborting");
- return START_NOT_STICKY;
- }
-
- // Don't run job if there is another running job
- mIsRunning = isProcessing();
-
- // Filter out duplicate intents
- long jobTime = intent.getLongExtra(JOB_TIME_KEY, 1);
- if (jobTime == 1 || jobTime == mLastJobTime) {
- log("Received empty job time or duplicate job time, aborting");
- return START_NOT_STICKY;
- }
- mLastJobTime = jobTime;
-
- // Must have a primary command
- String command = intent.getStringExtra(PRIMARY_COMMAND_KEY);
- if (TextUtils.isEmpty(command)) {
- log("Received empty primary command, aborting");
- return START_NOT_STICKY;
- }
-
- // Queue up the job
- List<Runnable> jobs_to_add = new ArrayList<>(0);
-
- log("Starting job with primary command \'" + command + "\', with job time: " + jobTime);
- if (TextUtils.equals(command, COMMAND_VALUE_INSTALL)) {
- List<String> paths = intent.getStringArrayListExtra(INSTALL_LIST_KEY);
- jobs_to_add.addAll(paths.stream().map(Installer::new).collect(Collectors.toList()));
- } else if (TextUtils.equals(command, COMMAND_VALUE_UNINSTALL)) {
- List<String> packages = intent.getStringArrayListExtra(UNINSTALL_LIST_KEY);
- jobs_to_add.addAll(packages.stream().map(Remover::new).collect(Collectors.toList()));
- if (intent.getBooleanExtra(WITH_RESTART_UI_KEY, false)) {
- jobs_to_add.add(new UiResetJob());
- }
- } else if (TextUtils.equals(command, COMMAND_VALUE_RESTART_UI)) {
- jobs_to_add.add(new UiResetJob());
- } else if (TextUtils.equals(command, COMMAND_VALUE_FORCE_STOP_SERVICE)) {
- log("Force stopping JobService...");
- forceStopService();
- } else if (TextUtils.equals(command, COMMAND_VALUE_CONFIGURATION_SHIM)) {
- jobs_to_add.add(new LocaleChanger(getApplicationContext(), mMainHandler));
- } else if (TextUtils.equals(command, COMMAND_VALUE_BOOTANIMATION)) {
- String fileName = intent.getStringExtra(BOOTANIMATION_FILE_NAME);
- if (TextUtils.isEmpty(fileName)) {
- jobs_to_add.add(new BootAnimationJob(true));
- } else {
- jobs_to_add.add(new BootAnimationJob(fileName));
- }
- } else if (TextUtils.equals(command, COMMAND_VALUE_FONTS)) {
- String pid = intent.getStringExtra(FONTS_PID);
- String fileName = intent.getStringExtra(FONTS_FILENAME);
-
- jobs_to_add.add(new FontsJob(pid, fileName));
- jobs_to_add.add(new UiResetJob());
- } else if (TextUtils.equals(command, COMMAND_VALUE_AUDIO)) {
- String pid = intent.getStringExtra(AUDIO_PID);
- String fileName = intent.getStringExtra(AUDIO_FILENAME);
-
- jobs_to_add.add(new SoundsJob(pid, fileName));
- jobs_to_add.add(new UiResetJob());
- } else if (TextUtils.equals(command, COMMAND_VALUE_ENABLE)) {
- List<String> packages = intent.getStringArrayListExtra(ENABLE_LIST_KEY);
- jobs_to_add.addAll(packages.stream().map(Enabler::new).collect(Collectors.toList()));
-
- if (intent.getBooleanExtra(WITH_RESTART_UI_KEY, false)) {
- jobs_to_add.add(new UiResetJob());
- }
- } else if (TextUtils.equals(command, COMMAND_VALUE_DISABLE)) {
- List<String> packages = intent.getStringArrayListExtra(DISABLE_LIST_KEY);
- jobs_to_add.addAll(packages.stream().map(Disabler::new).collect(Collectors.toList()));
-
- if (intent.getBooleanExtra(WITH_RESTART_UI_KEY, false)) {
- jobs_to_add.add(new UiResetJob());
- }
- } else if (TextUtils.equals(command, COMMAND_VALUE_PRIORITY)) {
- List<String> packages = intent.getStringArrayListExtra(PRIORITY_LIST_KEY);
- jobs_to_add.add(new PriorityJob(packages));
-
- if (intent.getBooleanExtra(WITH_RESTART_UI_KEY, false)) {
- jobs_to_add.add(new UiResetJob());
- }
- } else if (TextUtils.equals(command, COMMAND_VALUE_COPY)) {
- String source = intent.getStringExtra(SOURCE_FILE_KEY);
- String destination = intent.getStringExtra(DESTINATION_FILE_KEY);
-
- jobs_to_add.add(new CopyJob(source, destination));
- } else if (TextUtils.equals(command, COMMAND_VALUE_MOVE)) {
- String source = intent.getStringExtra(SOURCE_FILE_KEY);
- String destination = intent.getStringExtra(DESTINATION_FILE_KEY);
-
- jobs_to_add.add(new MoveJob(source, destination));
- } else if (TextUtils.equals(command, COMMAND_VALUE_MKDIR)) {
- String destination = intent.getStringExtra(DESTINATION_FILE_KEY);
-
- jobs_to_add.add(new MkdirJob(destination));
- } else if (TextUtils.equals(command, COMMAND_VALUE_DELETE)) {
- String dir = intent.getStringExtra(SOURCE_FILE_KEY);
-
- if (intent.getBooleanExtra(WITH_DELETE_PARENT_KEY, true)) {
- jobs_to_add.add(new DeleteJob(dir));
- } else {
- for (File child : new File(dir).listFiles()) {
- jobs_to_add.add(new DeleteJob(child.getAbsolutePath()));
- }
- }
- } else if (TextUtils.equals(command, COMMAND_VALUE_PROFILE)) {
- List<String> enable = intent.getStringArrayListExtra(ENABLE_LIST_KEY);
- List<String> disable = intent.getStringArrayListExtra(DISABLE_LIST_KEY);
- boolean restartUi = intent.getBooleanExtra(WITH_RESTART_UI_KEY, false);
- String profile = intent.getStringExtra(PROFILE_NAME_KEY);
-
- jobs_to_add.add(new ProfileJob(profile, disable, enable, restartUi));
- }
-
- if (jobs_to_add.size() > 0) {
- log("Adding new jobs to job queue");
-
- synchronized (mJobQueue) {
- mJobQueue.addAll(jobs_to_add);
- }
-
- if (!mJobHandler.hasMessages(JobHandler.MESSAGE_CHECK_QUEUE)) {
- mJobHandler.sendEmptyMessage(JobHandler.MESSAGE_CHECK_QUEUE);
- }
- }
-
return START_NOT_STICKY;
}
@Override
public IBinder onBind(Intent intent) {
- return null;
+ return mBinder;
}
- private boolean isProcessing() {
- return mJobQueue.size() > 0;
+ @Override
+ public void onDestroy() {
+ }
+
+ private void informCompletion(String extra) {
+ Intent intent = new Intent(INTENT_STATUS_CHANGED);
+ intent.putExtra(PRIMARY_COMMAND_KEY, extra);
+ sendBroadcastAsUser(intent, UserHandle.ALL);
}
@SuppressWarnings("deprecation")
private void install(String path, IPackageInstallObserver2 observer) {
try {
+ isWaiting = true;
getPM().installPackageAsUser(path, observer,
PackageManager.INSTALL_REPLACE_EXISTING,
null,
UserHandle.USER_SYSTEM);
+ while (isWaiting) {
+ Thread.sleep(500);
+ }
} catch (Exception e) {
Log.e(TAG, "", e);
}
@@ -336,7 +479,11 @@
@SuppressWarnings("deprecation")
private void uninstall(String packageName, IPackageDeleteObserver observer) {
try {
+ isWaiting = true;
getPM().deletePackageAsUser(packageName, observer, 0, UserHandle.USER_SYSTEM);
+ while (isWaiting) {
+ Thread.sleep(500);
+ }
} catch (Exception e) {
Log.e(TAG, "", e);
}
@@ -454,7 +601,7 @@
SystemProperties.set("sys.refresh_theme", "1");
Typeface.recreateDefaults();
float fontSize = Settings.System.getFloatForUser(getContentResolver(),
- Settings.System.FONT_SCALE, 1.0f, UserHandle.USER_CURRENT);
+ Settings.System.FONT_SCALE, 1.0f, UserHandle.USER_CURRENT);
Settings.System.putFloatForUser(getContentResolver(),
Settings.System.FONT_SCALE, (fontSize + 0.0000001f), UserHandle.USER_CURRENT);
}
@@ -566,24 +713,31 @@
File ogg = new File(themePath, sound.soundPath + ".ogg");
if (ogg.exists()) {
- if (sound.themePath.equals(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH) && sound.type != 0) {
+ if (sound.themePath.equals(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH) && sound.type !=
+ 0) {
SoundUtils.setUIAudible(this, ogg, ogg, sound.type, sound.soundName);
} else if (sound.themePath.equals(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH)) {
- SoundUtils.setUISounds(getContentResolver(), sound.soundName, ogg.getAbsolutePath());
+ SoundUtils.setUISounds(getContentResolver(), sound.soundName, ogg
+ .getAbsolutePath());
} else {
- SoundUtils.setAudible(this, ogg, ogg, sound.type, getSubsContext().getString(metaDataId));
+ SoundUtils.setAudible(this, ogg, ogg, sound.type, getSubsContext().getString
+ (metaDataId));
}
} else if (mp3.exists()) {
- if (sound.themePath.equals(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH) && sound.type != 0) {
+ if (sound.themePath.equals(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH) && sound.type !=
+ 0) {
SoundUtils.setUIAudible(this, mp3, mp3, sound.type, sound.soundName);
} else if (sound.themePath.equals(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH)) {
- SoundUtils.setUISounds(getContentResolver(), sound.soundName, mp3.getAbsolutePath());
+ SoundUtils.setUISounds(getContentResolver(), sound.soundName,
+ mp3.getAbsolutePath());
} else {
- SoundUtils.setAudible(this, mp3, mp3, sound.type, getSubsContext().getString(metaDataId));
+ SoundUtils.setAudible(this, mp3, mp3, sound.type, getSubsContext().getString
+ (metaDataId));
}
} else {
if (sound.themePath.equals(IOUtils.SYSTEM_THEME_UI_SOUNDS_PATH)) {
- SoundUtils.setDefaultUISounds(getContentResolver(), sound.soundName, sound.soundPath + ".ogg");
+ SoundUtils.setDefaultUISounds(getContentResolver(), sound.soundName, sound
+ .soundPath + ".ogg");
} else {
SoundUtils.setDefaultAudible(this, sound.type);
}
@@ -652,10 +806,6 @@
}
}
- private void forceStopService() {
- stopService(new Intent(this, JobService.class));
- }
-
private Context getSubsContext() {
return getAppContext(SUBSTRATUM_PACKAGE);
}
@@ -673,79 +823,60 @@
return ctx;
}
- private static boolean isDebugBuild() {
- return Build.TYPE.equals("eng") || Build.TYPE.equals("userdebug");
- }
-
- private static boolean shouldCheckBuildType() {
- return SystemProperties.getBoolean("ro.interfacer.buildtype.check", false);
- }
-
private boolean forceAuthorizePackages() {
return Settings.Secure.getIntForUser(getContentResolver(),
- Settings.Secure.FORCE_AUTHORIZE_SUBSTRATUM_PACKAGES, 0, UserHandle.USER_CURRENT) == 1;
+ Settings.Secure.FORCE_AUTHORIZE_SUBSTRATUM_PACKAGES, 0, UserHandle.USER_CURRENT)
+ == 1;
}
private boolean doSignaturesMatch(String packageName, Signature signature) {
if (packageName != null) {
- try {
- PackageInfo pi = getPM().getPackageInfo(packageName,
+ try {
+ PackageInfo pi = getPM().getPackageInfo(packageName,
PackageManager.GET_SIGNATURES, UserHandle.USER_SYSTEM);
- if (pi.signatures != null
- && pi.signatures.length == 1
- && signature.equals(pi.signatures[0])) {
- return true;
- }
- } catch (RemoteException ignored) {
- return false;
- }
- }
- return false;
+ if (pi.signatures != null
+ && pi.signatures.length == 1
+ && signature.equals(pi.signatures[0])) {
+ return true;
+ }
+ } catch (RemoteException ignored) {
+ return false;
+ }
+ }
+ return false;
}
- private boolean isCallerAuthorized(Intent intent) {
- PendingIntent token = null;
+ @SuppressWarnings("ConstantConditions")
+ private boolean isCallerAuthorized(int uid) {
+ String callingPackage = getPackageManager().getPackagesForUid(uid)[0];
- try {
- token = intent.getParcelableExtra(INTERFACER_TOKEN);
- } catch (Exception e) {
- log("Attempting to start service without a token - unauthorized!");
- }
-
- if (token == null) {
- return false;
- }
-
- // SECOND: We got a token, validate originating package
- // if not in our white list, return null
- String callingPackage = token.getCreatorPackage();
for (String AUTHORIZED_CALLER : AUTHORIZED_CALLERS) {
if (TextUtils.equals(callingPackage, AUTHORIZED_CALLER)) {
for (Signature AUTHORIZED_SIGNATURE : AUTHORIZED_SIGNATURES) {
if (doSignaturesMatch(callingPackage, AUTHORIZED_SIGNATURE)) {
log("\'" + callingPackage
- + "\' is an authorized calling package, validating calling package " +
- "permissions...");
+ + "\' is an authorized calling package...");
return true;
}
- }
+ }
}
}
if (shouldCheckBuildType() && isDebugBuild()) {
- log("the ROM is a userdebug or eng build configured to allow all packages," +
- "validating calling package permissions...");
+ log("the ROM is a userdebug or eng build configured to allow all " +
+ "packages, validating calling package permissions...");
return true;
}
- if (forceAuthorizePackages()) {
- log("\'" + callingPackage + "\' is not an authorized calling package, but the user " +
- "has explicitly allowed all calling packages, " +
- "validating calling package permissions...");
+ if (forceAuthorizePackages()) {
+ log("\'" + callingPackage + "\' is not an authorized calling " +
+ "package, but the user has explicitly allowed all calling" +
+ " packages, validating calling package permissions...");
return true;
}
log("\'" + callingPackage + "\' is not an authorized calling package.");
+ informCaller(false);
return false;
}
@@ -755,114 +886,14 @@
sendBroadcastAsUser(intent, UserHandle.ALL);
}
- private class MainHandler extends Handler {
- static final int MSG_JOB_QUEUE_EMPTY = 1;
+ private void delete(String mFileOrDirectory) {
+ log("DeleteJob - deleting \'" + mFileOrDirectory + "\'...");
- MainHandler(Looper looper) {
- super(looper);
- }
-
- @Override
- public void handleMessage(Message msg) {
- switch (msg.what) {
- case MSG_JOB_QUEUE_EMPTY:
- Intent intent = new Intent(INTENT_STATUS_CHANGED);
- intent.putExtra(PRIMARY_COMMAND_KEY, COMMAND_VALUE_JOB_COMPLETE);
- sendBroadcastAsUser(intent, UserHandle.ALL);
- break;
- }
- }
- }
-
- private class JobHandler extends Handler {
- private static final int MESSAGE_CHECK_QUEUE = 1;
- private static final int MESSAGE_DEQUEUE = 2;
-
- JobHandler(Looper looper) {
- super(looper);
- }
-
- @Override
- public void handleMessage(Message msg) {
- switch (msg.what) {
- case MESSAGE_CHECK_QUEUE:
- Runnable job;
-
- synchronized (mJobQueue) {
- job = mJobQueue.get(0);
- }
-
- if (job != null && !mIsRunning) {
- job.run();
- }
-
- break;
- case MESSAGE_DEQUEUE:
- Runnable toRemove = (Runnable) msg.obj;
-
- synchronized (mJobQueue) {
- mJobQueue.remove(toRemove);
-
- if (mJobQueue.size() > 0) {
- mIsRunning = false;
- this.sendEmptyMessage(MESSAGE_CHECK_QUEUE);
- } else {
- log("Job queue is empty. All done!");
-
- mIsRunning = false;
- mMainHandler.sendEmptyMessage(MainHandler.MSG_JOB_QUEUE_EMPTY);
- }
- }
-
- break;
- default:
- log("Unknown message " + msg.what);
- break;
- }
- }
- }
-
- private class UiResetJob implements Runnable {
- @Override
- public void run() {
- restartUi();
- log("Restarting SystemUI...");
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- UiResetJob.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class FontsJob implements Runnable {
- boolean mClear;
- String mPid;
- String mFileName;
-
- FontsJob(String pid, String fileName) {
- if (pid == null) {
- mClear = true;
- } else {
- mPid = pid;
- mFileName = fileName;
- }
- }
-
- @Override
- public void run() {
- if (mClear) {
- log("Restoring system font...");
- clearFonts();
- } else {
- log("Configuring theme font...");
- copyFonts(mPid, mFileName);
- }
-
- Intent intent = new Intent(INTENT_STATUS_CHANGED);
- intent.putExtra(PRIMARY_COMMAND_KEY, COMMAND_VALUE_FONTS);
- sendBroadcastAsUser(intent, UserHandle.ALL);
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- FontsJob.this);
- mJobHandler.sendMessage(message);
+ File file = new File(mFileOrDirectory);
+ if (file.exists()) {
+ IOUtils.deleteRecursive(file);
+ } else {
+ log("DeleteJob - \'" + mFileOrDirectory + "\' is already deleted.");
}
}
@@ -889,374 +920,23 @@
}
}
- private class SoundsJob implements Runnable {
- boolean mClear;
- String mPid;
- String mFileName;
-
- SoundsJob(String pid, String fileName) {
- if (pid == null) {
- mClear = true;
- } else {
- mPid = pid;
- mFileName = fileName;
- }
- }
-
- @Override
- public void run() {
- if (mClear) {
- log("Restoring system sounds...");
- clearSounds(JobService.this);
- } else {
- log("Configuring theme sounds...");
- applyThemedSounds(mPid, mFileName);
- }
-
- Intent intent = new Intent(INTENT_STATUS_CHANGED);
- intent.putExtra(PRIMARY_COMMAND_KEY, COMMAND_VALUE_AUDIO);
- sendBroadcastAsUser(intent, UserHandle.ALL);
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- SoundsJob.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class BootAnimationJob implements Runnable {
- final boolean mClear;
- String mFileName;
-
- BootAnimationJob(boolean clear) {
- mClear = clear;
- }
-
- BootAnimationJob(String fileName) {
- mFileName = fileName;
- mClear = false;
- }
-
- @Override
- public void run() {
- if (mClear) {
- log("Restoring system boot animation...");
- clearBootAnimation();
- } else {
- log("Configuring themed boot animation...");
- copyBootAnimation(mFileName);
- }
-
- Intent intent = new Intent(INTENT_STATUS_CHANGED);
- intent.putExtra(PRIMARY_COMMAND_KEY, COMMAND_VALUE_BOOTANIMATION);
- sendBroadcastAsUser(intent, UserHandle.ALL);
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- BootAnimationJob.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class Installer implements Runnable {
- String mPath;
-
- Installer(String path) {
- mPath = path;
- }
-
- @Override
- public void run() {
- log("Installer - installing \'" + mPath + "\'...");
- PackageInstallObserver observer = new PackageInstallObserver(Installer.this);
- install(mPath, observer);
- }
- }
-
private class PackageInstallObserver extends IPackageInstallObserver2.Stub {
- Object mObject;
-
- PackageInstallObserver(Object _object) {
- mObject = _object;
- }
-
public void onUserActionRequired(Intent intent) throws RemoteException {
log("Installer - user action required callback");
+ isWaiting = false;
}
- public void onPackageInstalled(String packageName, int returnCode, String msg, Bundle
- extras) {
+ public void onPackageInstalled(String packageName, int returnCode,
+ String msg, Bundle extras) {
log("Installer - successfully installed \'" + packageName + "\'!");
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE, mObject);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class Remover implements Runnable {
- String mPackage;
-
- Remover(String _package) {
- mPackage = _package;
- }
-
- @Override
- public void run() {
- // TODO: Fix isOverlayEnabled function, for now it's causing NPE
- if (isOverlayEnabled(mPackage)) {
- log("Remover - disabling overlay for \'" + mPackage + "\'...");
- switchOverlay(mPackage, false);
- }
-
- log("Remover - uninstalling \'" + mPackage + "\'...");
- PackageDeleteObserver observer = new PackageDeleteObserver(Remover.this);
- uninstall(mPackage, observer);
+ isWaiting = false;
}
}
private class PackageDeleteObserver extends IPackageDeleteObserver.Stub {
- Object mObject;
-
- PackageDeleteObserver(Object _object) {
- mObject = _object;
- }
-
public void packageDeleted(String packageName, int returnCode) {
log("Remover - successfully removed \'" + packageName + "\'");
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE, mObject);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class Enabler implements Runnable {
- String mPackage;
-
- Enabler(String _package) {
- mPackage = _package;
- }
-
- @Override
- public void run() {
- log("Enabler - enabling overlay for \'" + mPackage + "\'...");
- switchOverlay(mPackage, true);
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- Enabler.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class Disabler implements Runnable {
- String mPackage;
-
- Disabler(String _package) {
- mPackage = _package;
- }
-
- @Override
- public void run() {
- log("Disabler - disabling overlay for \'" + mPackage + "\'...");
- switchOverlay(mPackage, false);
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- Disabler.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class PriorityJob implements Runnable {
- List<String> mPackages;
-
- PriorityJob(List<String> _packages) {
- mPackages = _packages;
- }
-
- @Override
- public void run() {
- log("PriorityJob - processing priority changes...");
-
- try {
- for (int i = 0; i < mPackages.size() - 1; i++) {
- String parentName = mPackages.get(i);
- String packageName = mPackages.get(i + 1);
-
- getOMS().setPriority(packageName, parentName, UserHandle.USER_SYSTEM);
- }
- } catch (RemoteException e) {
- Log.e(TAG, "", e);
- }
-
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- PriorityJob.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class CopyJob implements Runnable {
- String mSource;
- String mDestination;
-
- CopyJob(String _source, String _destination) {
- mSource = _source;
- mDestination = _destination;
- }
-
- @Override
- public void run() {
- log("CopyJob - copying \'" + mSource + "\' to \'" + mDestination + "\'...");
-
- File sourceFile = new File(mSource);
- if (sourceFile.exists()) {
- if (sourceFile.isFile()) {
- IOUtils.bufferedCopy(mSource, mDestination);
- } else {
- IOUtils.copyFolder(mSource, mDestination);
- }
- } else {
- log("CopyJob - \'" + mSource + "\' does not exist, aborting...");
- }
-
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- CopyJob.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class MoveJob implements Runnable {
- String mSource;
- String mDestination;
-
- MoveJob(String _source, String _destination) {
- mSource = _source;
- mDestination = _destination;
- }
-
- @Override
- public void run() {
- log("MoveJob - moving \'" + mSource + "\' to \'" + mDestination + "\'...");
-
- File sourceFile = new File(mSource);
- if (sourceFile.exists()) {
- if (sourceFile.isFile()) {
- IOUtils.bufferedCopy(mSource, mDestination);
- } else {
- IOUtils.copyFolder(mSource, mDestination);
- }
- IOUtils.deleteRecursive(sourceFile);
- } else {
- log("MoveJob - \'" + mSource + "\' does not exist, aborting...");
- }
-
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- MoveJob.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class MkdirJob implements Runnable {
- String mDestination;
-
- MkdirJob(String destination) {
- mDestination = destination;
- }
-
- @Override
- public void run() {
- log("MkdirJob - creating \'" + mDestination + "\'...");
- IOUtils.createDirIfNotExists(mDestination);
-
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- MkdirJob.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class DeleteJob implements Runnable {
- String mFileOrDirectory;
-
- DeleteJob(String _directory) {
- mFileOrDirectory = _directory;
- }
-
- @Override
- public void run() {
- log("DeleteJob - deleting \'" + mFileOrDirectory + "\'...");
-
- File file = new File(mFileOrDirectory);
- if (file.exists()) {
- IOUtils.deleteRecursive(file);
- } else {
- log("DeleteJob - \'" + mFileOrDirectory + "\' is already deleted.");
- }
-
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- DeleteJob.this);
- mJobHandler.sendMessage(message);
- }
- }
-
- private class ProfileJob implements Runnable {
- String mProfileName;
- List<String> mToBeDisabled;
- List<String> mToBeEnabled;
- boolean mRestartUi;
-
- ProfileJob(String _name, List<String> _toBeDisabled,
- List<String> _toBeEnabled, boolean _restartUi) {
- mProfileName = _name;
- mToBeDisabled = _toBeDisabled;
- mToBeEnabled = _toBeEnabled;
- mRestartUi = _restartUi;
- }
-
- @Override
- public void run() {
- log("Applying profile...");
-
- // Clear system theme folder content
- File themeDir = new File(IOUtils.SYSTEM_THEME_PATH);
- for (File f : themeDir.listFiles()) {
- IOUtils.deleteRecursive(f);
- }
-
- // Process theme folder
- File profileDir = new File(Environment.getExternalStorageDirectory()
- .getAbsolutePath() + "/substratum/profiles/" +
- mProfileName + "/theme");
-
- if (profileDir.exists()) {
- File profileFonts = new File(profileDir, "fonts");
- if (profileFonts.exists()) {
- IOUtils.copyFolder(profileFonts, new File(IOUtils.SYSTEM_THEME_FONT_PATH));
- refreshFonts();
- mRestartUi = true;
- } else {
- clearFonts();
- }
-
- File profileSounds = new File(profileDir, "audio");
- if (profileSounds.exists()) {
- IOUtils.copyFolder(profileSounds, new File(IOUtils.SYSTEM_THEME_AUDIO_PATH));
- refreshSounds();
- mRestartUi = true;
- } else {
- clearSounds(JobService.this);
- }
- }
-
- // Disable all overlays installed
- for (String overlay : mToBeDisabled) {
- switchOverlay(overlay, false);
- }
-
- // Enable provided overlays
- for (String overlay : mToBeEnabled) {
- switchOverlay(overlay, true);
- }
-
- // Restart SystemUI when needed
- if (mRestartUi) {
- synchronized (mJobQueue) {
- mJobQueue.add(new UiResetJob());
- }
- }
-
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- ProfileJob.this);
- mJobHandler.sendMessage(message);
+ isWaiting = false;
}
}
@@ -1266,9 +946,9 @@
private Handler mHandler;
private Locale mCurrentLocale;
- public LocaleChanger(Context context, Handler mainHandler) {
+ public LocaleChanger(Context context) {
mContext = context;
- mHandler = mainHandler;
+ mHandler = new Handler();
}
@Override
@@ -1342,10 +1022,7 @@
Log.e(TAG, "", e);
return;
}
-
- Message message = mJobHandler.obtainMessage(JobHandler.MESSAGE_DEQUEUE,
- LocaleChanger.this);
- mJobHandler.sendMessage(message);
+ informCompletion(COMMAND_VALUE_JOB_COMPLETE);
}
@Override
@@ -1353,29 +1030,4 @@
mHandler.postDelayed(this::restoreLocale, 500);
}
}
-
- private static final Signature SUBSTRATUM_SIGNATURE = new Signature(""
- + "308202eb308201d3a003020102020411c02f2f300d06092a864886f70d01010b050030263124302206"
- + "03550403131b5375627374726174756d20446576656c6f706d656e74205465616d301e170d31363037"
- + "30333032333335385a170d3431303632373032333335385a3026312430220603550403131b53756273"
- + "74726174756d20446576656c6f706d656e74205465616d30820122300d06092a864886f70d01010105"
- + "000382010f003082010a02820101008855626336f645a335aa5d40938f15db911556385f72f72b5f8b"
- + "ad01339aaf82ae2d30302d3f2bba26126e8da8e76a834e9da200cdf66d1d5977c90a4e4172ce455704"
- + "a22bbe4a01b08478673b37d23c34c8ade3ec040a704da8570d0a17fce3c7397ea63ebcde3a2a3c7c5f"
- + "983a163e4cd5a1fc80c735808d014df54120e2e5708874739e22e5a22d50e1c454b2ae310b480825ab"
- + "3d877f675d6ac1293222602a53080f94e4a7f0692b627905f69d4f0bb1dfd647e281cc0695e0733fa3"
- + "efc57d88706d4426c4969aff7a177ac2d9634401913bb20a93b6efe60e790e06dad3493776c2c0878c"
- + "e82caababa183b494120edde3d823333efd464c8aea1f51f330203010001a321301f301d0603551d0e"
- + "04160414203ec8b075d1c9eb9d600100281c3924a831a46c300d06092a864886f70d01010b05000382"
- + "01010042d4bd26d535ce2bf0375446615ef5bf25973f61ecf955bdb543e4b6e6b5d026fdcab09fec09"
- + "c747fb26633c221df8e3d3d0fe39ce30ca0a31547e9ec693a0f2d83e26d231386ff45f8e4fd5c06095"
- + "8681f9d3bd6db5e940b1e4a0b424f5c463c79c5748a14a3a38da4dd7a5499dcc14a70ba82a50be5fe0"
- + "82890c89a27e56067d2eae952e0bcba4d6beb5359520845f1fdb7df99868786055555187ba46c69ee6"
- + "7fa2d2c79e74a364a8b3544997dc29cc625395e2f45bf8bdb2c9d8df0d5af1a59a58ad08b32cdbec38"
- + "19fa49201bb5b5aadeee8f2f096ac029055713b77054e8af07cd61fe97f7365d0aa92d570be98acb89"
- + "41b8a2b0053b54f18bfde092eb");
-
- private static final Signature[] AUTHORIZED_SIGNATURES = new Signature[]{
- SUBSTRATUM_SIGNATURE,
- };
}
diff --git a/app/src/main/java/projekt/interfacer/utils/IOUtils.java b/app/src/main/java/projekt/interfacer/utils/IOUtils.java
index b04a343..1104647 100644
--- a/app/src/main/java/projekt/interfacer/utils/IOUtils.java
+++ b/app/src/main/java/projekt/interfacer/utils/IOUtils.java
@@ -33,7 +33,6 @@
import java.util.zip.ZipInputStream;
public class IOUtils {
- private static final String TAG = IOUtils.class.getSimpleName();
public static final String SYSTEM_THEME_PATH = "/data/system/theme";
public static final String SYSTEM_THEME_FONT_PATH = SYSTEM_THEME_PATH + File.separator
+ "fonts";
@@ -49,6 +48,7 @@
+ File.separator + "ui";
public static final String SYSTEM_THEME_BOOTANIMATION_PATH = SYSTEM_THEME_PATH + File.separator
+ "bootanimation.zip";
+ private static final String TAG = IOUtils.class.getSimpleName();
private static boolean dirExists(String dirPath) {
final File dir = new File(dirPath);