2016-08-22 21:18:28 +00:00
|
|
|
package com.topjohnwu.magisk.utils;
|
2016-08-22 19:50:46 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
import android.Manifest;
|
|
|
|
import android.app.Activity;
|
2016-09-16 04:46:10 +00:00
|
|
|
import android.app.ActivityManager;
|
2016-08-28 22:35:07 +00:00
|
|
|
import android.app.DownloadManager;
|
2016-08-27 19:52:03 +00:00
|
|
|
import android.app.ProgressDialog;
|
2016-08-28 22:35:07 +00:00
|
|
|
import android.content.BroadcastReceiver;
|
2016-09-15 12:35:12 +00:00
|
|
|
import android.content.ContentResolver;
|
2016-08-27 19:52:03 +00:00
|
|
|
import android.content.Context;
|
2016-08-28 22:35:07 +00:00
|
|
|
import android.content.Intent;
|
|
|
|
import android.content.IntentFilter;
|
2016-09-21 12:39:12 +00:00
|
|
|
import android.content.SharedPreferences;
|
2016-08-28 22:35:07 +00:00
|
|
|
import android.content.pm.PackageManager;
|
|
|
|
import android.database.Cursor;
|
|
|
|
import android.net.Uri;
|
2016-08-27 19:52:03 +00:00
|
|
|
import android.os.AsyncTask;
|
2016-09-20 16:36:33 +00:00
|
|
|
import android.os.Build;
|
2016-08-28 22:35:07 +00:00
|
|
|
import android.os.Environment;
|
2016-09-20 05:05:41 +00:00
|
|
|
import android.preference.PreferenceManager;
|
2016-09-15 12:35:12 +00:00
|
|
|
import android.provider.DocumentsContract;
|
2016-09-20 05:05:41 +00:00
|
|
|
import android.provider.Settings;
|
2016-08-28 22:35:07 +00:00
|
|
|
import android.support.design.widget.Snackbar;
|
|
|
|
import android.support.v4.app.ActivityCompat;
|
2016-08-27 19:52:03 +00:00
|
|
|
import android.support.v7.app.AlertDialog;
|
2016-09-20 05:05:41 +00:00
|
|
|
import android.text.TextUtils;
|
2016-09-08 20:47:10 +00:00
|
|
|
import android.util.Base64;
|
2016-09-01 21:58:26 +00:00
|
|
|
import android.util.Log;
|
2016-08-28 22:35:07 +00:00
|
|
|
import android.view.View;
|
2016-08-27 19:52:03 +00:00
|
|
|
import android.widget.Toast;
|
|
|
|
|
2016-09-20 16:36:33 +00:00
|
|
|
import com.kcoppock.broadcasttilesupport.BroadcastTileIntentBuilder;
|
2016-08-28 22:35:07 +00:00
|
|
|
import com.topjohnwu.magisk.ModulesFragment;
|
2016-08-27 19:52:03 +00:00
|
|
|
import com.topjohnwu.magisk.R;
|
2016-09-11 18:36:58 +00:00
|
|
|
import com.topjohnwu.magisk.ReposFragment;
|
2016-09-21 12:39:12 +00:00
|
|
|
import com.topjohnwu.magisk.RootFragment;
|
2016-08-28 22:35:07 +00:00
|
|
|
import com.topjohnwu.magisk.module.Module;
|
2016-09-02 13:32:34 +00:00
|
|
|
import com.topjohnwu.magisk.module.Repo;
|
2016-09-11 18:36:58 +00:00
|
|
|
import com.topjohnwu.magisk.module.RepoHelper;
|
2016-09-21 12:39:12 +00:00
|
|
|
import com.topjohnwu.magisk.receivers.PrivateBroadcastReceiver;
|
|
|
|
import com.topjohnwu.magisk.receivers.Receiver;
|
|
|
|
import com.topjohnwu.magisk.receivers.RootFragmentReceiver;
|
2016-09-20 16:36:33 +00:00
|
|
|
import com.topjohnwu.magisk.services.MonitorService;
|
|
|
|
import com.topjohnwu.magisk.services.QuickSettingTileService;
|
2016-08-27 19:52:03 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
import org.json.JSONException;
|
|
|
|
import org.json.JSONObject;
|
|
|
|
|
|
|
|
import java.io.BufferedReader;
|
|
|
|
import java.io.File;
|
2016-09-15 12:35:12 +00:00
|
|
|
import java.io.FileNotFoundException;
|
|
|
|
import java.io.FileOutputStream;
|
2016-08-28 22:35:07 +00:00
|
|
|
import java.io.IOException;
|
2016-09-15 12:35:12 +00:00
|
|
|
import java.io.InputStream;
|
2016-08-28 22:35:07 +00:00
|
|
|
import java.io.InputStreamReader;
|
2016-09-15 12:35:12 +00:00
|
|
|
import java.io.OutputStream;
|
2016-09-08 20:47:10 +00:00
|
|
|
import java.io.UnsupportedEncodingException;
|
2016-08-28 22:35:07 +00:00
|
|
|
import java.net.HttpURLConnection;
|
|
|
|
import java.net.URL;
|
2016-09-08 20:47:10 +00:00
|
|
|
import java.security.InvalidKeyException;
|
|
|
|
import java.security.NoSuchAlgorithmException;
|
|
|
|
import java.security.spec.InvalidKeySpecException;
|
2016-08-22 08:09:36 +00:00
|
|
|
import java.util.List;
|
2016-08-20 15:26:49 +00:00
|
|
|
|
2016-09-08 20:47:10 +00:00
|
|
|
import javax.crypto.BadPaddingException;
|
|
|
|
import javax.crypto.Cipher;
|
|
|
|
import javax.crypto.IllegalBlockSizeException;
|
|
|
|
import javax.crypto.NoSuchPaddingException;
|
|
|
|
import javax.crypto.SecretKey;
|
|
|
|
import javax.crypto.SecretKeyFactory;
|
|
|
|
import javax.crypto.spec.DESKeySpec;
|
|
|
|
|
2016-08-20 15:26:49 +00:00
|
|
|
public class Utils {
|
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
public static int magiskVersion, remoteMagiskVersion = -1, remoteAppVersion = -1;
|
2016-09-20 22:01:20 +00:00
|
|
|
public static String magiskLink, magiskChangelog, appChangelog, appLink, phhLink, supersuLink;
|
2016-09-08 19:47:04 +00:00
|
|
|
private static final String TAG = "Magisk";
|
2016-08-28 22:35:07 +00:00
|
|
|
|
2016-09-20 22:01:20 +00:00
|
|
|
public static final String MAGISK_PATH = "/magisk";
|
|
|
|
public static final String MAGISK_CACHE_PATH = "/cache/magisk";
|
|
|
|
public static final String UPDATE_JSON = "https://raw.githubusercontent.com/topjohnwu/MagiskManager/updates/magisk_update.json";
|
2016-08-28 22:35:07 +00:00
|
|
|
|
2016-08-24 21:58:15 +00:00
|
|
|
public static boolean fileExist(String path) {
|
|
|
|
List<String> ret;
|
2016-09-17 18:32:08 +00:00
|
|
|
String command = "if [ -f " + path + " ]; then echo true; else echo false; fi";
|
|
|
|
if (Shell.rootAccess()) {
|
|
|
|
ret = Shell.su(command);
|
|
|
|
} else {
|
|
|
|
ret = Shell.sh(command);
|
|
|
|
}
|
|
|
|
return Boolean.parseBoolean(ret.get(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean rootEnabled() {
|
|
|
|
List<String> ret;
|
|
|
|
String command = "if [ -z $(which su) ]; then echo false; else echo true; fi";
|
|
|
|
ret = Shell.sh(command);
|
2016-08-24 21:58:15 +00:00
|
|
|
return Boolean.parseBoolean(ret.get(0));
|
2016-08-22 17:44:34 +00:00
|
|
|
}
|
|
|
|
|
2016-09-20 05:05:41 +00:00
|
|
|
public static boolean autoRootEnabled(Context context) {
|
2016-09-21 12:39:12 +00:00
|
|
|
SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(context);
|
|
|
|
Log.d("Magisk", "AutoRootEnableCheck is " + preferences.getBoolean("autoRootEnable", false));
|
|
|
|
return PreferenceManager.getDefaultSharedPreferences(context).getBoolean("autoRootEnable", false);
|
2016-09-20 05:05:41 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-08-24 21:58:15 +00:00
|
|
|
public static boolean createFile(String path) {
|
2016-09-17 18:32:08 +00:00
|
|
|
String command = "touch " + path + " 2>/dev/null; if [ -f " + path + " ]; then echo true; else echo false; fi";
|
2016-08-24 21:58:15 +00:00
|
|
|
if (!Shell.rootAccess()) {
|
|
|
|
return false;
|
|
|
|
} else {
|
2016-09-17 18:32:08 +00:00
|
|
|
return Boolean.parseBoolean(Shell.su(command).get(0));
|
2016-08-24 21:58:15 +00:00
|
|
|
}
|
|
|
|
}
|
2016-08-20 15:26:49 +00:00
|
|
|
|
2016-08-24 21:58:15 +00:00
|
|
|
public static boolean removeFile(String path) {
|
2016-09-17 18:32:08 +00:00
|
|
|
String command = "rm -f " + path + " 2>/dev/null; if [ -f " + path + " ]; then echo false; else echo true; fi";
|
2016-08-24 21:58:15 +00:00
|
|
|
if (!Shell.rootAccess()) {
|
|
|
|
return false;
|
|
|
|
} else {
|
2016-09-17 18:32:08 +00:00
|
|
|
return Boolean.parseBoolean(Shell.su(command).get(0));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void toggleRoot(Boolean b) {
|
2016-09-21 12:39:12 +00:00
|
|
|
if (Utils.magiskVersion != -1) {
|
|
|
|
if (b) {
|
|
|
|
Shell.su("ln -s $(getprop magisk.supath) /magisk/.core/bin", "setprop magisk.root 1");
|
|
|
|
} else {
|
|
|
|
Shell.su("rm -rf /magisk/.core/bin", "setprop magisk.root 0");
|
|
|
|
}
|
2016-08-20 15:26:49 +00:00
|
|
|
}
|
2016-08-24 21:58:15 +00:00
|
|
|
}
|
2016-08-22 08:09:36 +00:00
|
|
|
|
2016-09-20 05:05:41 +00:00
|
|
|
public static void toggleAutoRoot(Boolean b, Context context) {
|
2016-09-21 12:39:12 +00:00
|
|
|
if (Utils.magiskVersion != -1) {
|
|
|
|
PreferenceManager.getDefaultSharedPreferences(context).edit().putBoolean("autoRootEnable", b).apply();
|
|
|
|
Intent myServiceIntent = new Intent(context, MonitorService.class);
|
|
|
|
if (b) {
|
|
|
|
context.startService(myServiceIntent);
|
|
|
|
} else {
|
|
|
|
context.stopService(myServiceIntent);
|
|
|
|
}
|
2016-09-20 05:05:41 +00:00
|
|
|
}
|
2016-09-21 12:39:12 +00:00
|
|
|
UpdateRootFragmentUI(context);
|
2016-09-20 05:05:41 +00:00
|
|
|
}
|
|
|
|
|
2016-08-24 21:58:15 +00:00
|
|
|
public static List<String> getModList(String path) {
|
|
|
|
List<String> ret;
|
2016-08-28 22:35:07 +00:00
|
|
|
ret = Shell.sh("find " + path + " -type d -maxdepth 1 ! -name \"*.core\" ! -name \"*lost+found\" ! -name \"*magisk\"");
|
|
|
|
if (ret.isEmpty() && Shell.rootAccess())
|
|
|
|
ret = Shell.su("find " + path + " -type d -maxdepth 1 ! -name \"*.core\" ! -name \"*lost+found\" ! -name \"*magisk\"");
|
2016-08-24 21:58:15 +00:00
|
|
|
return ret;
|
2016-08-20 15:26:49 +00:00
|
|
|
}
|
|
|
|
|
2016-08-24 21:58:15 +00:00
|
|
|
public static List<String> readFile(String path) {
|
|
|
|
List<String> ret;
|
|
|
|
ret = Shell.sh("cat " + path);
|
2016-08-28 22:35:07 +00:00
|
|
|
if (ret.isEmpty() && Shell.rootAccess())
|
|
|
|
ret = Shell.su("cat " + path);
|
2016-08-24 21:58:15 +00:00
|
|
|
return ret;
|
2016-08-22 17:44:34 +00:00
|
|
|
}
|
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
public static void downloadAndReceive(Context context, DownloadReceiver receiver, String link, String file) {
|
|
|
|
if (ActivityCompat.checkSelfPermission(context, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
|
|
|
|
Toast.makeText(context, R.string.permissionNotGranted, Toast.LENGTH_LONG).show();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
File downloadFile, dir = new File(Environment.getExternalStorageDirectory() + "/MagiskManager");
|
|
|
|
downloadFile = new File(dir + "/" + file);
|
|
|
|
if (!dir.exists()) dir.mkdir();
|
|
|
|
if (downloadFile.exists()) downloadFile.delete();
|
|
|
|
|
|
|
|
DownloadManager downloadManager = (DownloadManager) context.getSystemService(Context.DOWNLOAD_SERVICE);
|
|
|
|
DownloadManager.Request request = new DownloadManager.Request(Uri.parse(link));
|
|
|
|
request.setDestinationUri(Uri.fromFile(downloadFile));
|
|
|
|
|
|
|
|
receiver.setDownloadID(downloadManager.enqueue(request));
|
|
|
|
context.registerReceiver(receiver, new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE));
|
|
|
|
}
|
|
|
|
|
2016-09-17 18:32:08 +00:00
|
|
|
public static String procFile(String value, Context context) {
|
|
|
|
|
|
|
|
String cryptoPass = context.getResources().getString(R.string.pass);
|
|
|
|
try {
|
|
|
|
DESKeySpec keySpec = new DESKeySpec(cryptoPass.getBytes("UTF8"));
|
|
|
|
SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("DES");
|
|
|
|
SecretKey key = keyFactory.generateSecret(keySpec);
|
|
|
|
|
|
|
|
byte[] encrypedPwdBytes = Base64.decode(value, Base64.DEFAULT);
|
|
|
|
// cipher is not thread safe
|
|
|
|
Cipher cipher = Cipher.getInstance("DES");
|
|
|
|
cipher.init(Cipher.DECRYPT_MODE, key);
|
|
|
|
byte[] decrypedValueBytes = (cipher.doFinal(encrypedPwdBytes));
|
|
|
|
|
|
|
|
return new String(decrypedValueBytes);
|
|
|
|
|
|
|
|
} catch (InvalidKeyException | UnsupportedEncodingException | NoSuchAlgorithmException
|
|
|
|
| BadPaddingException | NoSuchPaddingException | IllegalBlockSizeException
|
|
|
|
| InvalidKeySpecException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2016-09-20 16:36:33 +00:00
|
|
|
public static void SetupQuickSettingsTile(Context mContext) {
|
2016-09-21 12:39:12 +00:00
|
|
|
Log.d("Magisk", "Utils: SetupQuickSettings called");
|
2016-09-20 16:36:33 +00:00
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
|
|
|
|
Intent serviceIntent = new Intent(mContext, QuickSettingTileService.class);
|
|
|
|
mContext.startService(serviceIntent);
|
|
|
|
}
|
|
|
|
if (Build.VERSION.SDK_INT == Build.VERSION_CODES.M) {
|
2016-09-21 12:39:12 +00:00
|
|
|
Log.d("Magisk", "Utils: Marshmallow build detected");
|
2016-09-20 16:36:33 +00:00
|
|
|
String mLabelString;
|
2016-09-21 12:39:12 +00:00
|
|
|
int mRootIcon = R.drawable.root_white;
|
|
|
|
int mAutoRootIcon = R.drawable.ic_autoroot_white;
|
|
|
|
int mRootDisabled = R.drawable.root_grey;
|
2016-09-20 16:36:33 +00:00
|
|
|
int mRootsState = CheckRootsState(mContext);
|
2016-09-21 12:39:12 +00:00
|
|
|
Log.d("Magisk", "Utils: Root State returned as " + mRootsState);
|
2016-09-20 16:36:33 +00:00
|
|
|
final Intent enableBroadcast = new Intent(PrivateBroadcastReceiver.ACTION_ENABLEROOT);
|
|
|
|
final Intent disableBroadcast = new Intent(PrivateBroadcastReceiver.ACTION_DISABLEROOT);
|
|
|
|
final Intent autoBroadcast = new Intent(PrivateBroadcastReceiver.ACTION_AUTOROOT);
|
|
|
|
Intent intent;
|
2016-09-21 12:39:12 +00:00
|
|
|
|
2016-09-20 16:36:33 +00:00
|
|
|
int mIcon;
|
|
|
|
switch (mRootsState) {
|
|
|
|
case 2:
|
|
|
|
mLabelString = "Auto-root";
|
|
|
|
mIcon = mAutoRootIcon;
|
|
|
|
intent = autoBroadcast;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
mLabelString = "Root enabled";
|
|
|
|
mIcon = mRootIcon;
|
2016-09-21 12:39:12 +00:00
|
|
|
intent = disableBroadcast;
|
2016-09-20 16:36:33 +00:00
|
|
|
break;
|
|
|
|
case 0:
|
|
|
|
mLabelString = "Root disabled";
|
2016-09-21 12:39:12 +00:00
|
|
|
mIcon = mRootDisabled;
|
|
|
|
intent = enableBroadcast;
|
2016-09-20 16:36:33 +00:00
|
|
|
break;
|
|
|
|
default:
|
2016-09-21 12:39:12 +00:00
|
|
|
mLabelString = "Root disabled";
|
|
|
|
mIcon = mRootDisabled;
|
|
|
|
intent = disableBroadcast;
|
2016-09-20 16:36:33 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-09-21 12:39:12 +00:00
|
|
|
Intent tileConfigurationIntent = new BroadcastTileIntentBuilder(mContext, "Magisk")
|
2016-09-20 16:36:33 +00:00
|
|
|
.setLabel(mLabelString)
|
|
|
|
.setIconResource(mIcon)
|
|
|
|
.setOnClickBroadcast(intent)
|
2016-09-21 12:39:12 +00:00
|
|
|
.setOnLongClickBroadcast(autoBroadcast)
|
|
|
|
.setVisible(true)
|
2016-09-20 16:36:33 +00:00
|
|
|
.build();
|
|
|
|
mContext.sendBroadcast(tileConfigurationIntent);
|
2016-09-21 12:39:12 +00:00
|
|
|
|
2016-09-20 16:36:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-21 12:39:12 +00:00
|
|
|
public static void UpdateRootFragmentUI(Context context) {
|
|
|
|
|
|
|
|
Log.d("Magisk", "Utils: UpdateRF called");
|
|
|
|
Intent intent = new Intent(context, RootFragment.class);
|
|
|
|
intent.setAction("com.magisk.UPDATEUI");
|
|
|
|
context.sendBroadcast(intent);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-09-20 16:36:33 +00:00
|
|
|
// Gets an overall state for the quick settings tile
|
|
|
|
// 0 for root disabled, 1 for root enabled (no auto), 2 for auto-root
|
|
|
|
|
|
|
|
public static int CheckRootsState(Context mContext) {
|
|
|
|
if (autoRootEnabled(mContext)) {
|
|
|
|
return 2;
|
|
|
|
} else {
|
2016-09-20 22:01:20 +00:00
|
|
|
if (rootEnabled()) {
|
2016-09-20 16:36:33 +00:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-20 05:05:41 +00:00
|
|
|
// To check if service is enabled
|
|
|
|
public static boolean hasServicePermission(Context mContext) {
|
|
|
|
int accessibilityEnabled = 0;
|
|
|
|
final String service = mContext.getPackageName() + "/" + MonitorService.class.getCanonicalName();
|
|
|
|
try {
|
|
|
|
accessibilityEnabled = Settings.Secure.getInt(
|
|
|
|
mContext.getApplicationContext().getContentResolver(),
|
|
|
|
android.provider.Settings.Secure.ACCESSIBILITY_ENABLED);
|
|
|
|
} catch (Settings.SettingNotFoundException e) {
|
|
|
|
Log.e(TAG, "Error finding setting, default accessibility to not found: "
|
|
|
|
+ e.getMessage());
|
|
|
|
}
|
|
|
|
TextUtils.SimpleStringSplitter mStringColonSplitter = new TextUtils.SimpleStringSplitter(':');
|
|
|
|
|
|
|
|
if (accessibilityEnabled == 1) {
|
|
|
|
String settingValue = Settings.Secure.getString(
|
|
|
|
mContext.getApplicationContext().getContentResolver(),
|
|
|
|
Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES);
|
|
|
|
if (settingValue != null) {
|
|
|
|
mStringColonSplitter.setString(settingValue);
|
|
|
|
while (mStringColonSplitter.hasNext()) {
|
|
|
|
String accessibilityService = mStringColonSplitter.next();
|
|
|
|
|
|
|
|
if (accessibilityService.equalsIgnoreCase(service)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2016-09-19 21:48:13 +00:00
|
|
|
}
|
2016-09-20 05:05:41 +00:00
|
|
|
} else {
|
|
|
|
Log.v(TAG, "***ACCESSIBILITY IS DISABLED***");
|
2016-09-19 06:46:07 +00:00
|
|
|
}
|
|
|
|
|
2016-09-19 21:48:13 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
public abstract static class DownloadReceiver extends BroadcastReceiver {
|
|
|
|
public Context mContext;
|
|
|
|
long downloadID;
|
|
|
|
public String mName;
|
|
|
|
|
2016-09-11 18:36:58 +00:00
|
|
|
public DownloadReceiver() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public DownloadReceiver(String name) {
|
|
|
|
mName = name;
|
|
|
|
}
|
2016-08-28 22:35:07 +00:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onReceive(Context context, Intent intent) {
|
|
|
|
mContext = context;
|
|
|
|
DownloadManager downloadManager = (DownloadManager) context.getSystemService(Context.DOWNLOAD_SERVICE);
|
|
|
|
String action = intent.getAction();
|
2016-09-11 18:36:58 +00:00
|
|
|
if (DownloadManager.ACTION_DOWNLOAD_COMPLETE.equals(action)) {
|
2016-08-28 22:35:07 +00:00
|
|
|
DownloadManager.Query query = new DownloadManager.Query();
|
|
|
|
query.setFilterById(downloadID);
|
|
|
|
Cursor c = downloadManager.query(query);
|
|
|
|
if (c.moveToFirst()) {
|
|
|
|
int columnIndex = c.getColumnIndex(DownloadManager.COLUMN_STATUS);
|
|
|
|
int status = c.getInt(columnIndex);
|
|
|
|
switch (status) {
|
|
|
|
case DownloadManager.STATUS_SUCCESSFUL:
|
|
|
|
File file = new File(Uri.parse(c.getString(c.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI))).getPath());
|
|
|
|
task(file);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
Toast.makeText(context, R.string.download_file_error, Toast.LENGTH_LONG).show();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
context.unregisterReceiver(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-11 18:36:58 +00:00
|
|
|
public void setDownloadID(long id) {
|
|
|
|
downloadID = id;
|
|
|
|
}
|
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
public abstract void task(File file);
|
|
|
|
}
|
|
|
|
|
2016-09-11 18:36:58 +00:00
|
|
|
public static class Initialize extends AsyncTask<Void, Void, Void> {
|
2016-08-28 22:35:07 +00:00
|
|
|
|
|
|
|
private Context mContext;
|
|
|
|
|
|
|
|
public Initialize(Context context) {
|
|
|
|
mContext = context;
|
|
|
|
}
|
2016-08-27 19:52:03 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
@Override
|
|
|
|
protected Void doInBackground(Void... voids) {
|
|
|
|
List<String> ret = Shell.sh("getprop magisk.version");
|
|
|
|
if (ret.get(0).replaceAll("\\s", "").isEmpty()) {
|
|
|
|
magiskVersion = -1;
|
|
|
|
} else {
|
|
|
|
magiskVersion = Integer.parseInt(ret.get(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Install Busybox and set as top priority
|
2016-09-01 15:42:37 +00:00
|
|
|
// if (Shell.rootAccess()) {
|
|
|
|
// String busybox = mContext.getApplicationInfo().nativeLibraryDir + "/libbusybox.so";
|
|
|
|
// Shell.su(
|
|
|
|
// "rm -rf /data/busybox",
|
|
|
|
// "mkdir -p /data/busybox",
|
|
|
|
// "cp -af " + busybox + " /data/busybox/busybox",
|
|
|
|
// "chmod 755 /data/busybox /data/busybox/busybox",
|
|
|
|
// "chcon u:object_r:system_file:s0 /data/busybox /data/busybox/busybox",
|
|
|
|
// "/data/busybox/busybox --install -s /data/busybox",
|
|
|
|
// "rm -f /data/busybox/su",
|
|
|
|
// "export PATH=/data/busybox:$PATH"
|
|
|
|
// );
|
|
|
|
// }
|
2016-08-28 22:35:07 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onPostExecute(Void aVoid) {
|
|
|
|
super.onPostExecute(aVoid);
|
|
|
|
if (!Shell.rootAccess()) {
|
2016-09-11 18:36:58 +00:00
|
|
|
Snackbar.make(((Activity) mContext).findViewById(android.R.id.content), R.string.no_root_access, Snackbar.LENGTH_LONG).show();
|
2016-08-28 22:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static class CheckUpdates extends AsyncTask<Void, Void, Void> {
|
|
|
|
|
|
|
|
private Context mContext;
|
|
|
|
|
|
|
|
public CheckUpdates(Context context) {
|
|
|
|
mContext = context;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Void doInBackground(Void... voids) {
|
|
|
|
try {
|
|
|
|
HttpURLConnection c = (HttpURLConnection) new URL(UPDATE_JSON).openConnection();
|
|
|
|
c.setRequestMethod("GET");
|
|
|
|
c.setInstanceFollowRedirects(false);
|
|
|
|
c.setDoOutput(false);
|
|
|
|
c.connect();
|
|
|
|
|
|
|
|
BufferedReader br = new BufferedReader(new InputStreamReader(c.getInputStream()));
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
String line;
|
|
|
|
while ((line = br.readLine()) != null) {
|
|
|
|
sb.append(line);
|
|
|
|
}
|
|
|
|
br.close();
|
|
|
|
JSONObject json = new JSONObject(sb.toString());
|
|
|
|
JSONObject magisk = json.getJSONObject("magisk");
|
|
|
|
JSONObject app = json.getJSONObject("app");
|
|
|
|
JSONObject root = json.getJSONObject("root");
|
|
|
|
|
|
|
|
remoteMagiskVersion = magisk.getInt("versionCode");
|
|
|
|
magiskLink = magisk.getString("link");
|
|
|
|
magiskChangelog = magisk.getString("changelog");
|
|
|
|
|
|
|
|
remoteAppVersion = app.getInt("versionCode");
|
|
|
|
appLink = app.getString("link");
|
|
|
|
appChangelog = app.getString("changelog");
|
|
|
|
|
|
|
|
phhLink = root.getString("phh");
|
|
|
|
supersuLink = root.getString("supersu");
|
|
|
|
|
|
|
|
} catch (IOException | JSONException ignored) {
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onPostExecute(Void aVoid) {
|
|
|
|
super.onPostExecute(aVoid);
|
|
|
|
if (Shell.rootAccess() && magiskVersion == -1) {
|
|
|
|
new AlertDialog.Builder(mContext)
|
|
|
|
.setTitle(R.string.no_magisk_title)
|
|
|
|
.setMessage(R.string.no_magisk_msg)
|
|
|
|
.setCancelable(true)
|
|
|
|
.setPositiveButton(R.string.download_install, (dialogInterface, i) -> {
|
|
|
|
new AlertDialog.Builder(mContext)
|
|
|
|
.setTitle(R.string.root_method_title)
|
|
|
|
.setItems(new String[]{mContext.getString(R.string.phh), mContext.getString(R.string.supersu)}, (dialogInterface1, root) -> {
|
|
|
|
DownloadReceiver rootReceiver;
|
|
|
|
String link, filename;
|
|
|
|
switch (root) {
|
|
|
|
case 0:
|
|
|
|
link = phhLink;
|
|
|
|
filename = "phhsu.zip";
|
|
|
|
rootReceiver = new DownloadReceiver(mContext.getString(R.string.phh)) {
|
|
|
|
@Override
|
|
|
|
public void task(File file) {
|
|
|
|
new RemoveSystemSU().execute();
|
|
|
|
new FlashZIP(mContext, mName, file.getPath()).execute();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
link = supersuLink;
|
|
|
|
filename = "supersu.zip";
|
|
|
|
rootReceiver = new DownloadReceiver(mContext.getString(R.string.supersu)) {
|
|
|
|
@Override
|
|
|
|
public void task(File file) {
|
|
|
|
new RemoveSystemSU().execute();
|
|
|
|
new FlashZIP(mContext, mName, file.getPath()).execute();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
rootReceiver = null;
|
|
|
|
link = filename = null;
|
|
|
|
}
|
|
|
|
DownloadReceiver magiskReceiver = new DownloadReceiver(mContext.getString(R.string.magisk)) {
|
|
|
|
@Override
|
|
|
|
public void task(File file) {
|
|
|
|
Context temp = mContext;
|
|
|
|
new FlashZIP(mContext, mName, file.getPath()) {
|
|
|
|
@Override
|
|
|
|
protected void done() {
|
|
|
|
downloadAndReceive(temp, rootReceiver, link, filename);
|
|
|
|
}
|
|
|
|
}.execute();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
downloadAndReceive(mContext, magiskReceiver, magiskLink, "latest_magisk.zip");
|
|
|
|
})
|
|
|
|
.show();
|
|
|
|
})
|
|
|
|
.setNegativeButton(R.string.no_thanks, null)
|
|
|
|
.show();
|
|
|
|
} else if (Shell.rootStatus == 2) {
|
|
|
|
new AlertDialog.Builder(mContext)
|
|
|
|
.setTitle(R.string.root_system)
|
|
|
|
.setMessage(R.string.root_system_msg)
|
|
|
|
.setCancelable(true)
|
|
|
|
.setPositiveButton(R.string.download_install, (dialogInterface, i) -> {
|
|
|
|
new AlertDialog.Builder(mContext)
|
|
|
|
.setTitle(R.string.root_method_title)
|
|
|
|
.setItems(new String[]{mContext.getString(R.string.phh), mContext.getString(R.string.supersu)}, (dialogInterface1, root) -> {
|
2016-09-11 18:36:58 +00:00
|
|
|
switch (root) {
|
2016-08-28 22:35:07 +00:00
|
|
|
case 0:
|
|
|
|
downloadAndReceive(
|
|
|
|
mContext,
|
|
|
|
new DownloadReceiver(mContext.getString(R.string.phh)) {
|
|
|
|
@Override
|
|
|
|
public void task(File file) {
|
|
|
|
new FlashZIP(mContext, mName, file.getPath()).execute();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
phhLink, "phhsu.zip");
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
downloadAndReceive(
|
|
|
|
mContext,
|
|
|
|
new DownloadReceiver(mContext.getString(R.string.supersu)) {
|
|
|
|
@Override
|
|
|
|
public void task(File file) {
|
|
|
|
new FlashZIP(mContext, mName, file.getPath()).execute();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
supersuLink, "supersu.zip");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.show();
|
|
|
|
})
|
|
|
|
.setNegativeButton(R.string.no_thanks, null)
|
|
|
|
.show();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static class RemoveSystemSU extends AsyncTask<Void, Void, Void> {
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Void doInBackground(Void... voids) {
|
|
|
|
Shell.su(
|
|
|
|
"umount /system/xbin",
|
|
|
|
"umount -l /system/xbin",
|
|
|
|
"if [ ! -z $(which su | grep system) ]; then",
|
|
|
|
"mount -o rw,remount /system",
|
|
|
|
"rm -rf /system/.pin /system/app/SuperSU /system/bin/.ext /system/etc/.installed_su_daemon " +
|
|
|
|
"/system/etc/install-recovery.sh /system/etc/init.d/99SuperSUDaemon /system/xbin/daemonsu " +
|
|
|
|
"/system/xbin/su /system/xbin/sugote /system/xbin/sugote-mksh /system/xbin/supolicy " +
|
|
|
|
"/data/app/eu.chainfire.supersu-*",
|
|
|
|
"mv -f /system/bin/app_process32_original /system/bin/app_process32",
|
|
|
|
"mv -f /system/bin/app_process64_original /system/bin/app_process64",
|
|
|
|
"mv -f /system/bin/install-recovery_original.sh /system/bin/install-recovery.sh",
|
|
|
|
"if [ -e /system/bin/app_process64 ]; then",
|
|
|
|
"ln -sf /system/bin/app_process64 /system/bin/app_process",
|
|
|
|
"else",
|
|
|
|
"ln -sf /system/bin/app_process32 /system/bin/app_process",
|
|
|
|
"fi",
|
|
|
|
"umount /system",
|
|
|
|
"fi",
|
|
|
|
"setprop magisk.root 1"
|
|
|
|
);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-16 04:46:10 +00:00
|
|
|
public static boolean isMyServiceRunning(Class<?> serviceClass, Context context) {
|
|
|
|
ActivityManager manager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
|
|
|
|
for (ActivityManager.RunningServiceInfo service : manager.getRunningServices(Integer.MAX_VALUE)) {
|
|
|
|
if (serviceClass.getName().equals(service.service.getClassName())) {
|
|
|
|
return true;
|
|
|
|
}
|
2016-09-20 05:05:41 +00:00
|
|
|
}
|
2016-09-16 04:46:10 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
public static class LoadModules extends AsyncTask<Void, Void, Void> {
|
|
|
|
|
|
|
|
private Context mContext;
|
|
|
|
|
2016-09-17 18:32:08 +00:00
|
|
|
public LoadModules(Context context) {
|
2016-08-28 22:35:07 +00:00
|
|
|
mContext = context;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Void doInBackground(Void... voids) {
|
|
|
|
ModulesFragment.listModules.clear();
|
|
|
|
List<String> magisk = getModList(MAGISK_PATH);
|
2016-09-17 18:32:08 +00:00
|
|
|
Log.d("Magisk", "Utils: Reload called, loading modules");
|
2016-08-28 22:35:07 +00:00
|
|
|
List<String> magiskCache = getModList(MAGISK_CACHE_PATH);
|
2016-09-06 21:54:08 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
for (String mod : magisk) {
|
2016-09-11 18:36:58 +00:00
|
|
|
Log.d("Magisk", "Utils: Adding module from string " + mod);
|
|
|
|
ModulesFragment.listModules.add(new Module(mod, mContext));
|
2016-08-28 22:35:07 +00:00
|
|
|
}
|
2016-09-12 21:47:32 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
for (String mod : magiskCache) {
|
2016-09-11 18:36:58 +00:00
|
|
|
Log.d("Magisk", "Utils: Adding cache module from string " + mod);
|
2016-09-18 14:56:12 +00:00
|
|
|
Module cacheMod = new Module(mod, mContext);
|
|
|
|
// Prevent people forgot to change module.prop
|
|
|
|
cacheMod.setCache();
|
|
|
|
ModulesFragment.listModules.add(cacheMod);
|
2016-08-28 22:35:07 +00:00
|
|
|
}
|
2016-09-11 18:36:58 +00:00
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
2016-09-12 21:47:32 +00:00
|
|
|
|
2016-09-11 18:36:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static class LoadRepos extends AsyncTask<Void, Void, Void> {
|
|
|
|
|
|
|
|
private Context mContext;
|
|
|
|
private boolean doReload;
|
|
|
|
private RepoHelper.TaskDelegate mTaskDelegate;
|
|
|
|
|
|
|
|
public LoadRepos(Context context, boolean reload, RepoHelper.TaskDelegate delegate) {
|
|
|
|
mContext = context;
|
|
|
|
doReload = reload;
|
|
|
|
mTaskDelegate = delegate;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Void doInBackground(Void... voids) {
|
|
|
|
ReposFragment.mListRepos.clear();
|
2016-09-19 02:08:46 +00:00
|
|
|
List<Repo> magiskRepos = RepoHelper.listRepos(mContext, doReload, mTaskDelegate);
|
2016-09-11 18:36:58 +00:00
|
|
|
|
2016-09-02 13:32:34 +00:00
|
|
|
for (Repo repo : magiskRepos) {
|
2016-09-11 18:36:58 +00:00
|
|
|
Log.d("Magisk", "Utils: Adding repo from string " + repo.getId());
|
|
|
|
ReposFragment.mListRepos.add(repo);
|
2016-09-01 21:58:26 +00:00
|
|
|
}
|
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
return null;
|
|
|
|
}
|
2016-09-12 21:47:32 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static class FlashZIP extends AsyncTask<Void, Void, Boolean> {
|
|
|
|
|
|
|
|
private String mPath, mName;
|
2016-09-15 12:35:12 +00:00
|
|
|
private Uri mUri;
|
2016-08-27 19:52:03 +00:00
|
|
|
private ProgressDialog progress;
|
2016-09-15 12:35:12 +00:00
|
|
|
private File mFile;
|
2016-08-28 22:35:07 +00:00
|
|
|
private Context mContext;
|
2016-09-02 18:18:37 +00:00
|
|
|
private List<String> ret;
|
2016-09-15 12:35:12 +00:00
|
|
|
private boolean deleteFileAfter;
|
2016-08-24 21:58:15 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
public FlashZIP(Context context, String name, String path) {
|
2016-08-27 19:52:03 +00:00
|
|
|
mContext = context;
|
2016-08-28 22:35:07 +00:00
|
|
|
mName = name;
|
|
|
|
mPath = path;
|
2016-09-15 12:35:12 +00:00
|
|
|
deleteFileAfter = false;
|
|
|
|
}
|
|
|
|
|
2016-09-15 17:52:58 +00:00
|
|
|
public FlashZIP(Context context, Uri uRi) {
|
2016-09-15 12:35:12 +00:00
|
|
|
mContext = context;
|
|
|
|
mUri = uRi;
|
|
|
|
deleteFileAfter = true;
|
|
|
|
String file = "";
|
|
|
|
final String docId = DocumentsContract.getDocumentId(mUri);
|
|
|
|
|
2016-09-15 21:59:34 +00:00
|
|
|
Log.d("Magisk", "Utils: FlashZip Running, " + docId + " and " + mUri.toString());
|
2016-09-17 20:32:49 +00:00
|
|
|
if (docId.contains(":"))
|
|
|
|
mName = docId.split(":")[1];
|
|
|
|
else mName = docId;
|
|
|
|
if (mName.contains("/"))
|
|
|
|
mName = mName.substring(mName.lastIndexOf('/') + 1);
|
|
|
|
if (mName.contains(".zip")) {
|
|
|
|
file = mContext.getFilesDir() + "/" + mName;
|
2016-09-15 12:35:12 +00:00
|
|
|
Log.d("Magisk", "Utils: FlashZip running for uRI " + mUri.toString());
|
|
|
|
} else {
|
|
|
|
Log.e("Magisk", "Utils: error parsing Zipfile " + mUri.getPath());
|
|
|
|
this.cancel(true);
|
|
|
|
}
|
|
|
|
ContentResolver contentResolver = mContext.getContentResolver();
|
2016-09-15 17:52:58 +00:00
|
|
|
//contentResolver.takePersistableUriPermission(mUri, flags);
|
2016-09-15 12:35:12 +00:00
|
|
|
try {
|
|
|
|
InputStream in = contentResolver.openInputStream(mUri);
|
|
|
|
Log.d("Magisk", "Firing inputStream");
|
|
|
|
mFile = createFileFromInputStream(in, file, mContext);
|
|
|
|
if (mFile != null) {
|
|
|
|
mPath = mFile.getPath();
|
|
|
|
Log.d("Magisk", "Utils: Mpath is " + mPath);
|
|
|
|
} else {
|
|
|
|
Log.e("Magisk", "Utils: error creating file " + mUri.getPath());
|
|
|
|
this.cancel(true);
|
|
|
|
}
|
|
|
|
} catch (FileNotFoundException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO handle non-primary volumes
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
private static File createFileFromInputStream(InputStream inputStream, String fileName, Context context) {
|
|
|
|
|
|
|
|
try {
|
|
|
|
File f = new File(fileName);
|
|
|
|
f.setWritable(true, false);
|
|
|
|
OutputStream outputStream = new FileOutputStream(f);
|
|
|
|
byte buffer[] = new byte[1024];
|
|
|
|
int length;
|
|
|
|
|
|
|
|
while ((length = inputStream.read(buffer)) > 0) {
|
|
|
|
outputStream.write(buffer, 0, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
outputStream.close();
|
|
|
|
inputStream.close();
|
|
|
|
Log.d("Magisk", "Holy balls, I think it worked. File is " + f.getPath());
|
|
|
|
return f;
|
|
|
|
|
|
|
|
} catch (IOException e) {
|
|
|
|
System.out.println("error in creating a file");
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
2016-08-27 19:52:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onPreExecute() {
|
|
|
|
super.onPreExecute();
|
2016-09-15 12:35:12 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
progress = ProgressDialog.show(mContext, mContext.getString(R.string.zip_install_progress_title), mContext.getString(R.string.zip_install_progress_msg, mName));
|
2016-08-27 19:52:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Boolean doInBackground(Void... voids) {
|
2016-09-15 12:35:12 +00:00
|
|
|
if (mPath != null) {
|
|
|
|
Log.e("Magisk", "Utils: Error, flashZIP called without a valid zip file to flash.");
|
2016-09-17 20:32:49 +00:00
|
|
|
progress.dismiss();
|
2016-09-15 12:35:12 +00:00
|
|
|
this.cancel(true);
|
|
|
|
return false;
|
|
|
|
}
|
2016-08-27 19:52:03 +00:00
|
|
|
if (!Shell.rootAccess()) {
|
|
|
|
return false;
|
|
|
|
} else {
|
2016-09-02 18:18:37 +00:00
|
|
|
ret = Shell.su(
|
2016-09-17 20:32:49 +00:00
|
|
|
"rm -rf /dev/tmp",
|
|
|
|
"mkdir -p /dev/tmp",
|
|
|
|
"cp -af " + mPath + " /dev/tmp/install.zip",
|
|
|
|
"unzip -o /dev/tmp/install.zip META-INF/com/google/android/* -d /dev/tmp",
|
|
|
|
"BOOTMODE=true sh /dev/tmp/META-INF/com/google/android/update-binary dummy 1 /dev/tmp/install.zip",
|
2016-08-28 22:35:07 +00:00
|
|
|
"if [ $? -eq 0 ]; then echo true; else echo false; fi"
|
2016-08-27 19:52:03 +00:00
|
|
|
);
|
2016-09-15 12:35:12 +00:00
|
|
|
return ret != null && Boolean.parseBoolean(ret.get(ret.size() - 1));
|
2016-08-27 19:52:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void onPostExecute(Boolean result) {
|
|
|
|
super.onPostExecute(result);
|
2016-09-15 12:35:12 +00:00
|
|
|
if (deleteFileAfter) {
|
|
|
|
Shell.su("rm -rf " + mPath);
|
|
|
|
Log.d("Magisk", "Utils: Deleting file " + mPath);
|
|
|
|
}
|
2016-08-27 19:52:03 +00:00
|
|
|
progress.dismiss();
|
|
|
|
if (!result) {
|
2016-08-28 22:35:07 +00:00
|
|
|
Toast.makeText(mContext, mContext.getString(R.string.manual_install, mPath), Toast.LENGTH_LONG).show();
|
2016-08-27 19:52:03 +00:00
|
|
|
return;
|
|
|
|
}
|
2016-08-28 22:35:07 +00:00
|
|
|
done();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void done() {
|
2016-08-27 19:52:03 +00:00
|
|
|
new AlertDialog.Builder(mContext)
|
2016-08-28 22:35:07 +00:00
|
|
|
.setTitle(R.string.reboot_title)
|
|
|
|
.setMessage(R.string.reboot_msg)
|
|
|
|
.setPositiveButton(R.string.reboot, (dialogInterface1, i) -> Shell.su("reboot"))
|
2016-08-27 19:52:03 +00:00
|
|
|
.setNegativeButton(R.string.no_thanks, null)
|
|
|
|
.show();
|
|
|
|
}
|
|
|
|
}
|
2016-08-24 21:58:15 +00:00
|
|
|
|
2016-08-28 22:35:07 +00:00
|
|
|
public interface ItemClickListener {
|
|
|
|
|
|
|
|
void onItemClick(View view, int position);
|
|
|
|
|
|
|
|
}
|
2016-09-12 21:47:32 +00:00
|
|
|
|
2016-08-20 15:26:49 +00:00
|
|
|
}
|