mirror of
				https://github.com/fankes/termux-app.git
				synced 2025-10-26 13:49:20 +08:00 
			
		
		
		
	Merge remote-tracking branch 'remotes/origin/master'
This commit is contained in:
		| @@ -139,14 +139,14 @@ public final class BackgroundJob { | ||||
|         try (BufferedReader in = new BufferedReader(new InputStreamReader(new FileInputStream(TermuxService.PREFIX_PATH + "/etc/apt/sources.list")))) { | ||||
|             String line; | ||||
|             while ((line = in.readLine()) != null) { | ||||
|                 if (!line.startsWith("#") && line.contains("https://dl.bintray.com/termux/termux-packages-24")) { | ||||
|                     return false; | ||||
|                 if (!line.startsWith("#") && line.contains("//termux.net stable")) { | ||||
|                     return true; | ||||
|                 } | ||||
|             } | ||||
|         } catch (IOException e) { | ||||
|             Log.e(LOG_TAG, "Error trying to read sources.list", e); | ||||
|         } | ||||
|         return true; | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     public static int getPid(Process p) { | ||||
|   | ||||
| @@ -354,9 +354,13 @@ public final class ExtraKeysView extends GridLayout { | ||||
|                     if (Settings.System.getInt(getContext().getContentResolver(), | ||||
|                         Settings.System.HAPTIC_FEEDBACK_ENABLED, 0) != 0) { | ||||
|  | ||||
|                         // Depending on DnD settings, value can be >1 but 0 means "disabled". | ||||
|                         if (Settings.Global.getInt(getContext().getContentResolver(), "zen_mode", 0) < 1) { | ||||
|                         if (Build.VERSION.SDK_INT >= 28) { | ||||
|                             finalButton.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP); | ||||
|                         } else { | ||||
|                             // Perform haptic feedback only if no total silence mode enabled. | ||||
|                             if (Settings.Global.getInt(getContext().getContentResolver(), "zen_mode", 0) != 2) { | ||||
|                                 finalButton.performHapticFeedback(HapticFeedbackConstants.KEYBOARD_TAP); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
| @@ -401,8 +405,14 @@ public final class ExtraKeysView extends GridLayout { | ||||
|                             } | ||||
|                             return true; | ||||
|  | ||||
|                         case MotionEvent.ACTION_UP: | ||||
|                         case MotionEvent.ACTION_CANCEL: | ||||
|                             v.setBackgroundColor(BUTTON_COLOR); | ||||
|                             if (scheduledExecutor != null) { | ||||
|                                 scheduledExecutor.shutdownNow(); | ||||
|                                 scheduledExecutor = null; | ||||
|                             } | ||||
|                             return true; | ||||
|                         case MotionEvent.ACTION_UP: | ||||
|                             v.setBackgroundColor(BUTTON_COLOR); | ||||
|                             if (scheduledExecutor != null) { | ||||
|                                 scheduledExecutor.shutdownNow(); | ||||
| @@ -427,11 +437,7 @@ public final class ExtraKeysView extends GridLayout { | ||||
|  | ||||
|                 LayoutParams param = new GridLayout.LayoutParams(); | ||||
|                 param.width = 0; | ||||
|                 if(Build.VERSION.SDK_INT == Build.VERSION_CODES.LOLLIPOP) { // special handle api 21 | ||||
|                     param.height = (int)(37.5 * getResources().getDisplayMetrics().density + 0.5); // 37.5 equal to R.id.viewpager layout_height / rows in DP | ||||
|                 } else { | ||||
|                     param.height = 0; | ||||
|                 } | ||||
|                 param.height = 0; | ||||
|                 param.setMargins(0, 0, 0, 0); | ||||
|                 param.columnSpec = GridLayout.spec(col, GridLayout.FILL, 1.f); | ||||
|                 param.rowSpec = GridLayout.spec(row, GridLayout.FILL, 1.f); | ||||
|   | ||||
| @@ -127,6 +127,8 @@ public final class TermuxActivity extends Activity implements ServiceConnection | ||||
|      */ | ||||
|     boolean mIsVisible; | ||||
|  | ||||
|     boolean mIsUsingBlackUI; | ||||
|  | ||||
|     final SoundPool mBellSoundPool = new SoundPool.Builder().setMaxStreams(1).setAudioAttributes( | ||||
|         new AudioAttributes.Builder().setContentType(AudioAttributes.CONTENT_TYPE_SONIFICATION) | ||||
|             .setUsage(AudioAttributes.USAGE_ASSISTANCE_SONIFICATION).build()).build(); | ||||
| @@ -186,28 +188,35 @@ public final class TermuxActivity extends Activity implements ServiceConnection | ||||
|     } | ||||
|  | ||||
|     /** For processes to access shared internal storage (/sdcard) we need this permission. */ | ||||
|     @TargetApi(Build.VERSION_CODES.M) | ||||
|     public boolean ensureStoragePermissionGranted() { | ||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||
|             if (checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) { | ||||
|                 return true; | ||||
|             } else { | ||||
|                 requestPermissions(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUESTCODE_PERMISSION_STORAGE); | ||||
|                 return false; | ||||
|             } | ||||
|         } else { | ||||
|             // Always granted before Android 6.0. | ||||
|         if (checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) { | ||||
|             return true; | ||||
|         } else { | ||||
|             requestPermissions(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUESTCODE_PERMISSION_STORAGE); | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onCreate(Bundle bundle) { | ||||
|         mSettings = new TermuxPreferences(this); | ||||
|         mIsUsingBlackUI = mSettings.isUsingBlackUI(); | ||||
|         if (mIsUsingBlackUI) { | ||||
|             this.setTheme(R.style.Theme_Termux_Black); | ||||
|         } else { | ||||
|             this.setTheme(R.style.Theme_Termux); | ||||
|         } | ||||
|  | ||||
|         super.onCreate(bundle); | ||||
|  | ||||
|         mSettings = new TermuxPreferences(this); | ||||
|  | ||||
|         setContentView(R.layout.drawer_layout); | ||||
|  | ||||
|         if (mIsUsingBlackUI) { | ||||
|             findViewById(R.id.left_drawer).setBackgroundColor( | ||||
|                 getResources().getColor(android.R.color.background_dark) | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         mTerminalView = findViewById(R.id.terminal_view); | ||||
|         mTerminalView.setOnKeyListener(new TermuxViewClient(this)); | ||||
|  | ||||
| @@ -434,7 +443,11 @@ public final class TermuxActivity extends Activity implements ServiceConnection | ||||
|                 boolean sessionRunning = sessionAtRow.isRunning(); | ||||
|  | ||||
|                 TextView firstLineView = row.findViewById(R.id.row_line); | ||||
|  | ||||
|                 if (mIsUsingBlackUI) { | ||||
|                     firstLineView.setBackground( | ||||
|                         getResources().getDrawable(R.drawable.selected_session_background_black) | ||||
|                     ); | ||||
|                 } | ||||
|                 String name = sessionAtRow.mSessionName; | ||||
|                 String sessionTitle = sessionAtRow.getTitle(); | ||||
|  | ||||
| @@ -454,7 +467,8 @@ public final class TermuxActivity extends Activity implements ServiceConnection | ||||
|                 } else { | ||||
|                     firstLineView.setPaintFlags(firstLineView.getPaintFlags() | Paint.STRIKE_THRU_TEXT_FLAG); | ||||
|                 } | ||||
|                 int color = sessionRunning || sessionAtRow.getExitStatus() == 0 ? Color.BLACK : Color.RED; | ||||
|                 int defaultColor = mIsUsingBlackUI ? Color.WHITE : Color.BLACK; | ||||
|                 int color = sessionRunning || sessionAtRow.getExitStatus() == 0 ? defaultColor : Color.RED; | ||||
|                 firstLineView.setTextColor(color); | ||||
|                 return row; | ||||
|             } | ||||
|   | ||||
| @@ -4,7 +4,6 @@ import android.app.Activity; | ||||
| import android.app.AlertDialog; | ||||
| import android.app.ProgressDialog; | ||||
| import android.content.Context; | ||||
| import android.os.Build; | ||||
| import android.os.Environment; | ||||
| import android.os.UserManager; | ||||
| import android.system.Os; | ||||
| @@ -16,14 +15,12 @@ import com.termux.R; | ||||
| import com.termux.terminal.EmulatorDebug; | ||||
|  | ||||
| import java.io.BufferedReader; | ||||
| import java.io.ByteArrayInputStream; | ||||
| import java.io.File; | ||||
| import java.io.FileOutputStream; | ||||
| import java.io.IOException; | ||||
| import java.io.InputStreamReader; | ||||
| import java.net.MalformedURLException; | ||||
| import java.net.URL; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import java.util.zip.ZipEntry; | ||||
| import java.util.zip.ZipInputStream; | ||||
| @@ -38,7 +35,7 @@ import java.util.zip.ZipInputStream; | ||||
|  * <p/> | ||||
|  * (3) A staging folder, $STAGING_PREFIX, is {@link #deleteFolder(File)} if left over from broken installation below. | ||||
|  * <p/> | ||||
|  * (4) The architecture is determined and an appropriate bootstrap zip url is determined in {@link #determineZipUrl()}. | ||||
|  * (4) The zip file is loaded from a shared library. | ||||
|  * <p/> | ||||
|  * (5) The zip, containing entries relative to the $PREFIX, is is downloaded and extracted by a zip input stream | ||||
|  * continuously encountering zip file entries: | ||||
| @@ -82,8 +79,8 @@ final class TermuxInstaller { | ||||
|                     final byte[] buffer = new byte[8096]; | ||||
|                     final List<Pair<String, String>> symlinks = new ArrayList<>(50); | ||||
|  | ||||
|                     final URL zipUrl = determineZipUrl(); | ||||
|                     try (ZipInputStream zipInput = new ZipInputStream(zipUrl.openStream())) { | ||||
|                     final byte[] zipBytes = loadZipBytes(); | ||||
|                     try (ZipInputStream zipInput = new ZipInputStream(new ByteArrayInputStream(zipBytes))) { | ||||
|                         ZipEntry zipEntry; | ||||
|                         while ((zipEntry = zipInput.getNextEntry()) != null) { | ||||
|                             if (zipEntry.getName().equals("SYMLINKS.txt")) { | ||||
| @@ -167,34 +164,13 @@ final class TermuxInstaller { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** Get bootstrap zip url for this systems cpu architecture. */ | ||||
|     private static URL determineZipUrl() throws MalformedURLException { | ||||
|         String archName = determineTermuxArchName(); | ||||
|         String url = Build.VERSION.SDK_INT >= Build.VERSION_CODES.N | ||||
|             ? "https://termux.org/bootstrap-" + archName + ".zip" | ||||
|             : "https://termux.net/bootstrap/bootstrap-" + archName + ".zip"; | ||||
|         return new URL(url); | ||||
|     public static byte[] loadZipBytes() { | ||||
|         // Only load the shared library when necessary to save memory usage. | ||||
|         System.loadLibrary("termux-bootstrap"); | ||||
|         return getZip(); | ||||
|     } | ||||
|  | ||||
|     private static String determineTermuxArchName() { | ||||
|         // Note that we cannot use System.getProperty("os.arch") since that may give e.g. "aarch64" | ||||
|         // while a 64-bit runtime may not be installed (like on the Samsung Galaxy S5 Neo). | ||||
|         // Instead we search through the supported abi:s on the device, see: | ||||
|         // http://developer.android.com/ndk/guides/abis.html | ||||
|         // Note that we search for abi:s in preferred order (the ordering of the | ||||
|         // Build.SUPPORTED_ABIS list) to avoid e.g. installing arm on an x86 system where arm | ||||
|         // emulation is available. | ||||
|         for (String androidArch : Build.SUPPORTED_ABIS) { | ||||
|             switch (androidArch) { | ||||
|                 case "arm64-v8a": return "aarch64"; | ||||
|                 case "armeabi-v7a": return "arm"; | ||||
|                 case "x86_64": return "x86_64"; | ||||
|                 case "x86": return "i686"; | ||||
|             } | ||||
|         } | ||||
|         throw new RuntimeException("Unable to determine arch from Build.SUPPORTED_ABIS =  " + | ||||
|             Arrays.toString(Build.SUPPORTED_ABIS)); | ||||
|     } | ||||
|     public static native byte[] getZip(); | ||||
|  | ||||
|     /** Delete a folder and all its content or throw. Don't follow symlinks. */ | ||||
|     static void deleteFolder(File fileOrDirectory) throws IOException { | ||||
|   | ||||
| @@ -58,6 +58,7 @@ final class TermuxPreferences { | ||||
|     private static final String CURRENT_SESSION_KEY = "current_session"; | ||||
|     private static final String SCREEN_ALWAYS_ON_KEY = "screen_always_on"; | ||||
|  | ||||
|     private String mUseDarkUI; | ||||
|     private boolean mScreenAlwaysOn; | ||||
|     private int mFontSize; | ||||
|  | ||||
| @@ -126,6 +127,10 @@ final class TermuxPreferences { | ||||
|         return mScreenAlwaysOn; | ||||
|     } | ||||
|  | ||||
|     boolean isUsingBlackUI() { | ||||
|         return mUseDarkUI.toLowerCase().equals("true"); | ||||
|     } | ||||
|  | ||||
|     void setScreenAlwaysOn(Context context, boolean newValue) { | ||||
|         mScreenAlwaysOn = newValue; | ||||
|         PreferenceManager.getDefaultSharedPreferences(context).edit().putBoolean(SCREEN_ALWAYS_ON_KEY, newValue).apply(); | ||||
| @@ -173,6 +178,8 @@ final class TermuxPreferences { | ||||
|                 break; | ||||
|         } | ||||
|  | ||||
|         mUseDarkUI = props.getProperty("use-black-ui", "false"); | ||||
|  | ||||
|         try { | ||||
|             JSONArray arr = new JSONArray(props.getProperty("extra-keys", "[['ESC', 'TAB', 'CTRL', 'ALT', '-', 'DOWN', 'UP']]")); | ||||
|  | ||||
|   | ||||
| @@ -114,19 +114,17 @@ public final class TermuxService extends Service implements SessionChangedCallba | ||||
|                 mWifiLock = wm.createWifiLock(WifiManager.WIFI_MODE_FULL_HIGH_PERF, EmulatorDebug.LOG_TAG); | ||||
|                 mWifiLock.acquire(); | ||||
|  | ||||
|                 if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||
|                     String packageName = getPackageName(); | ||||
|                     if (!pm.isIgnoringBatteryOptimizations(packageName)) { | ||||
|                         Intent whitelist = new Intent(); | ||||
|                         whitelist.setAction(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS); | ||||
|                         whitelist.setData(Uri.parse("package:" + packageName)); | ||||
|                         whitelist.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); | ||||
|                 String packageName = getPackageName(); | ||||
|                 if (!pm.isIgnoringBatteryOptimizations(packageName)) { | ||||
|                     Intent whitelist = new Intent(); | ||||
|                     whitelist.setAction(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS); | ||||
|                     whitelist.setData(Uri.parse("package:" + packageName)); | ||||
|                     whitelist.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); | ||||
|  | ||||
|                         try { | ||||
|                             startActivity(whitelist); | ||||
|                         } catch (ActivityNotFoundException e) { | ||||
|                             Log.e(EmulatorDebug.LOG_TAG, "Failed to call ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS", e); | ||||
|                         } | ||||
|                     try { | ||||
|                         startActivity(whitelist); | ||||
|                     } catch (ActivityNotFoundException e) { | ||||
|                         Log.e(EmulatorDebug.LOG_TAG, "Failed to call ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS", e); | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user