implementation 'androidx.viewpager:viewpager:1.0.0'
implementation 'androidx.webkit:webkit:1.4.0'
- // Include the Kotlin standard libraries. This should be the same version number listed in project build.gradle.
- implementation 'org.jetbrains.kotlin:kotlin-stdlib-jdk8:1.5.31'
+ // Include the Kotlin standard library. This should be the same version number listed in project build.gradle.
+ implementation 'org.jetbrains.kotlin:kotlin-stdlib-jdk8:1.6.0'
// Include the Google material library.
implementation 'com.google.android.material:material:1.4.0'
package com.stoutner.privacybrowser.activities
-import android.content.Intent
-import android.net.Uri
import android.os.Bundle
import android.view.WindowManager
-import android.widget.EditText
-import android.widget.LinearLayout
import androidx.appcompat.app.AppCompatActivity
import androidx.appcompat.widget.Toolbar
-import androidx.fragment.app.DialogFragment
import androidx.preference.PreferenceManager
import androidx.viewpager.widget.ViewPager
-import com.google.android.material.snackbar.Snackbar
import com.google.android.material.tabs.TabLayout
import com.stoutner.privacybrowser.R
import com.stoutner.privacybrowser.adapters.AboutPagerAdapter
-import com.stoutner.privacybrowser.asynctasks.SaveAboutVersionImage
-import com.stoutner.privacybrowser.dialogs.SaveDialog
-import com.stoutner.privacybrowser.dialogs.SaveDialog.SaveListener
-import com.stoutner.privacybrowser.fragments.AboutVersionFragment
-import java.io.ByteArrayInputStream
-import java.io.InputStream
-import java.lang.Exception
-import java.nio.charset.StandardCharsets
-
-class AboutActivity : AppCompatActivity(), SaveListener {
+class AboutActivity : AppCompatActivity() {
// Declare the class variables.
private lateinit var aboutPagerAdapter: AboutPagerAdapter
// Connect the tab layout to the view pager.
aboutTabLayout.setupWithViewPager(aboutViewPager)
}
-
- // The activity result is called after browsing for a file in the save alert dialog.
- public override fun onActivityResult(requestCode: Int, resultCode: Int, returnedIntent: Intent?) {
- // Run the default commands.
- super.onActivityResult(requestCode, resultCode, returnedIntent)
-
- // Only do something if the user didn't press back from the file picker.
- if (resultCode == RESULT_OK) {
- // Get a handle for the save dialog fragment.
- val saveDialogFragment = supportFragmentManager.findFragmentByTag(getString(R.string.save_dialog)) as DialogFragment?
-
- // Only update the file name if the dialog still exists.
- if (saveDialogFragment != null) {
- // Get a handle for the save dialog.
- val saveDialog = saveDialogFragment.dialog!!
-
- // Get a handle for the file name edit text.
- val fileNameEditText = saveDialog.findViewById<EditText>(R.id.file_name_edittext)
-
- // Get the file name URI from the intent.
- val fileNameUri = returnedIntent!!.data
-
- // Get the file name string from the URI.
- val fileNameString = fileNameUri.toString()
-
- // Set the file name text.
- fileNameEditText.setText(fileNameString)
-
- // Move the cursor to the end of the file name edit text.
- fileNameEditText.setSelection(fileNameString.length)
- }
- }
- }
-
- override fun onSave(saveType: Int, dialogFragment: DialogFragment) {
- // Get a handle for the dialog.
- val dialog = dialogFragment.dialog!!
-
- // Get a handle for the file name edit text.
- val fileNameEditText = dialog.findViewById<EditText>(R.id.file_name_edittext)
-
- // Get the file name string.
- val fileNameString = fileNameEditText.text.toString()
-
- // Get a handle for the about version linear layout.
- val aboutVersionLinearLayout = findViewById<LinearLayout>(R.id.about_version_linearlayout)
-
- // Process the save event according to the type.
- when (saveType) {
- SaveDialog.SAVE_ABOUT_VERSION_TEXT -> try {
- // Get a handle for the about version fragment.
- val aboutVersionFragment = aboutPagerAdapter.getTabFragment(0) as AboutVersionFragment
-
- // Get the about version text.
- val aboutVersionString = aboutVersionFragment.aboutVersionString
-
- // Create an input stream with the contents of about version.
- val aboutVersionInputStream: InputStream = ByteArrayInputStream(aboutVersionString.toByteArray(StandardCharsets.UTF_8))
-
- // Open an output stream.
- val outputStream = contentResolver.openOutputStream(Uri.parse(fileNameString))!!
-
- // Copy the input stream to the output stream.
- aboutVersionInputStream.copyTo(outputStream, 2048)
-
- // Close the streams.
- aboutVersionInputStream.close()
- outputStream.close()
-
- // Display a snackbar with the saved about version information.
- Snackbar.make(aboutVersionLinearLayout, getString(R.string.file_saved) + " " + fileNameString, Snackbar.LENGTH_SHORT).show()
- } catch (exception: Exception) {
- // Display a snackbar with the error message.
- Snackbar.make(aboutVersionLinearLayout, getString(R.string.error_saving_file) + " " + exception.toString(), Snackbar.LENGTH_INDEFINITE).show()
- }
-
- SaveDialog.SAVE_ABOUT_VERSION_IMAGE ->
- // Save the about version image.
- SaveAboutVersionImage(this, fileNameString, aboutVersionLinearLayout).execute()
- }
- }
}
import android.content.ClipData
import android.content.ClipboardManager
-import android.content.Intent
import android.net.Uri
import android.os.Build
import android.os.Bundle
import android.view.Menu
import android.view.MenuItem
import android.view.WindowManager
-import android.widget.EditText
import android.widget.TextView
import android.widget.ScrollView
+import androidx.activity.result.contract.ActivityResultContracts
import androidx.appcompat.app.AppCompatActivity
import androidx.appcompat.widget.Toolbar
-import androidx.fragment.app.DialogFragment
import androidx.preference.PreferenceManager
import androidx.swiperefreshlayout.widget.SwipeRefreshLayout
import com.google.android.material.snackbar.Snackbar
+import com.stoutner.privacybrowser.BuildConfig
import com.stoutner.privacybrowser.R
-import com.stoutner.privacybrowser.dialogs.SaveDialog
import java.io.BufferedReader
import java.io.IOException
// Define the class constants.
private const val SCROLLVIEW_POSITION = "scrollview_position"
-class LogcatActivity : AppCompatActivity(), SaveDialog.SaveListener {
+class LogcatActivity : AppCompatActivity() {
// Define the class variables.
private var scrollViewYPositionInt = 0
private lateinit var logcatScrollView: ScrollView
private lateinit var logcatTextView: TextView
+ // Define the save logcat activity result launcher. It must be defined before `onCreate()` is run or the app will crash.
+ private val saveLogcatActivityResultLauncher = registerForActivityResult(ActivityResultContracts.CreateDocument()) { fileNameUri: Uri? ->
+ // Only save the file if the URI is not null, which happens if the user exited the file picker by pressing back.
+ if (fileNameUri != null) {
+ try {
+ // Get the logcat string.
+ val logcatString = logcatTextView.text.toString()
+
+ // Open an output stream.
+ val outputStream = contentResolver.openOutputStream(fileNameUri)!!
+
+ // Write the logcat string to the output stream.
+ outputStream.write(logcatString.toByteArray(StandardCharsets.UTF_8))
+
+ // Close the output stream.
+ outputStream.close()
+
+ // Initialize the file name string from the file name URI last path segment.
+ var fileNameString = fileNameUri.lastPathSegment
+
+ // Query the exact file name if the API >= 26.
+ if (Build.VERSION.SDK_INT >= 26) {
+ // Get a cursor from the content resolver.
+ val contentResolverCursor = contentResolver.query(fileNameUri, null, null, null)!!
+
+ // Move to the fist row.
+ contentResolverCursor.moveToFirst()
+
+ // Get the file name from the cursor.
+ fileNameString = contentResolverCursor.getString(contentResolverCursor.getColumnIndex(OpenableColumns.DISPLAY_NAME))
+
+ // Close the cursor.
+ contentResolverCursor.close()
+ }
+
+ // Display a snackbar with the saved logcat information.
+ Snackbar.make(logcatTextView, getString(R.string.saved, fileNameString), Snackbar.LENGTH_SHORT).show()
+ } catch (exception: Exception) {
+ // Display a snackbar with the error message.
+ Snackbar.make(logcatTextView, getString(R.string.error_saving_logcat, exception.toString()), Snackbar.LENGTH_INDEFINITE).show()
+ }
+ }
+ }
+
public override fun onCreate(savedInstanceState: Bundle?) {
// Get a handle for the shared preferences.
val sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this)
}
R.id.save -> { // Save was selected.
- // Instantiate the save alert dialog.
- val saveDialogFragment: DialogFragment = SaveDialog.save(SaveDialog.SAVE_LOGCAT)
-
- // Show the save alert dialog.
- saveDialogFragment.show(supportFragmentManager, getString(R.string.save_logcat))
+ // Open the file picker.
+ saveLogcatActivityResultLauncher.launch(getString(R.string.privacy_browser_logcat_txt, BuildConfig.VERSION_NAME))
// Consume the event.
true
// Stop the swipe to refresh animation if it is displayed.
swipeRefreshLayout.isRefreshing = false
}
-
- // The activity result is called after browsing for a file in the save alert dialog.
- public override fun onActivityResult(requestCode: Int, resultCode: Int, returnedIntent: Intent?) {
- // Run the default commands.
- super.onActivityResult(requestCode, resultCode, returnedIntent)
-
- // Only do something if the user didn't press back from the file picker.
- if (resultCode == RESULT_OK) {
- // Get a handle for the save dialog fragment.
- val saveDialogFragment = supportFragmentManager.findFragmentByTag(getString(R.string.save_logcat)) as DialogFragment?
-
- // Only update the file name if the dialog still exists.
- if (saveDialogFragment != null) {
- // Get a handle for the save dialog.
- val saveDialog = saveDialogFragment.dialog!!
-
- // Get a handle for the file name edit text.
- val fileNameEditText = saveDialog.findViewById<EditText>(R.id.file_name_edittext)
-
- // Get the file name URI from the intent.
- val fileNameUri = returnedIntent!!.data
-
- // Get the file name string from the URI.
- val fileNameString = fileNameUri.toString()
-
- // Set the file name text.
- fileNameEditText.setText(fileNameString)
-
- // Move the cursor to the end of the file name edit text.
- fileNameEditText.setSelection(fileNameString.length)
- }
- }
- }
-
- override fun onSave(saveType: Int, dialogFragment: DialogFragment) {
- // Get a handle for the dialog.
- val dialog = dialogFragment.dialog!!
-
- // Get a handle for the file name edit text.
- val fileNameEditText = dialog.findViewById<EditText>(R.id.file_name_edittext)
-
- // Get the file path string.
- var fileNameString = fileNameEditText.text.toString()
-
- try {
- // Get the logcat as a string.
- val logcatString = logcatTextView.text.toString()
-
- // Open an output stream.
- val outputStream = contentResolver.openOutputStream(Uri.parse(fileNameString))!!
-
- // Write the logcat string to the output stream.
- outputStream.write(logcatString.toByteArray(StandardCharsets.UTF_8))
-
- // Close the output stream.
- outputStream.close()
-
- // Get the actual file name if the API >= 26.
- if (Build.VERSION.SDK_INT >= 26) {
- // Get a cursor from the content resolver.
- val contentResolverCursor = contentResolver.query(Uri.parse(fileNameString), null, null, null)!!
-
- // Move to the first row.
- contentResolverCursor.moveToFirst()
-
- // Get the file name from the cursor.
- fileNameString = contentResolverCursor.getString(contentResolverCursor.getColumnIndex(OpenableColumns.DISPLAY_NAME))
-
- // Close the cursor.
- contentResolverCursor.close()
- }
-
- // Display a snackbar with the saved logcat information.
- Snackbar.make(logcatTextView, getString(R.string.file_saved) + " " + fileNameString, Snackbar.LENGTH_SHORT).show()
- } catch (exception: Exception) {
- // Display a snackbar with the error message.
- Snackbar.make(logcatTextView, getString(R.string.error_saving_file) + " " + exception.toString(), Snackbar.LENGTH_INDEFINITE).show()
- }
- }
}
\ No newline at end of file
package com.stoutner.privacybrowser.asynctasks;
import android.app.Activity;
+import android.database.Cursor;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.net.Uri;
import android.os.AsyncTask;
+import android.os.Build;
+import android.provider.OpenableColumns;
import android.widget.LinearLayout;
import com.google.android.material.snackbar.Snackbar;
import java.lang.ref.WeakReference;
public class SaveAboutVersionImage extends AsyncTask<Void, Void, String> {
+ // Declare the class constants.
+ private final String SUCCESS = "Success";
+
// Declare the weak references.
private final WeakReference<Activity> activityWeakReference;
private final WeakReference<LinearLayout> aboutVersionLinearLayoutWeakReference;
- // Declare the class constants.
- private final String SUCCESS = "Success";
-
// Declare the class variables.
private Snackbar savingImageSnackbar;
private Bitmap aboutVersionBitmap;
+ private final Uri fileNameUri;
private final String fileNameString;
// The public constructor.
- public SaveAboutVersionImage(Activity activity, String fileNameString, LinearLayout aboutVersionLinearLayout) {
+ public SaveAboutVersionImage(Activity activity, Uri fileNameUri, LinearLayout aboutVersionLinearLayout) {
// Populate the weak references.
activityWeakReference = new WeakReference<>(activity);
aboutVersionLinearLayoutWeakReference = new WeakReference<>(aboutVersionLinearLayout);
// Store the class variables.
- this.fileNameString = fileNameString;
+ this.fileNameUri = fileNameUri;
+
+ // Query the exact file name if the API >= 26.
+ if (Build.VERSION.SDK_INT >= 26) {
+ // Get a cursor from the content resolver.
+ Cursor contentResolverCursor = activity.getContentResolver().query(fileNameUri, null, null, null);
+
+ // Move to the first row.
+ contentResolverCursor.moveToFirst();
+
+ // Get the file name from the cursor.
+ fileNameString = contentResolverCursor.getString(contentResolverCursor.getColumnIndex(OpenableColumns.DISPLAY_NAME));
+
+ // Close the cursor.
+ contentResolverCursor.close();
+ } else {
+ // Use the URI last path segment as the file name string.
+ fileNameString = fileNameUri.getLastPathSegment();
+ }
}
// `onPreExecute()` operates on the UI thread.
try {
// Open an output stream.
- OutputStream outputStream = activity.getContentResolver().openOutputStream(Uri.parse(fileNameString));
+ OutputStream outputStream = activity.getContentResolver().openOutputStream(fileNameUri);
// Write the webpage image to the image file.
aboutVersionByteArrayOutputStream.writeTo(outputStream);
+++ /dev/null
-/*
- * Copyright © 2016-2021 Soren Stoutner <soren@stoutner.com>.
- *
- * This file is part of Privacy Browser <https://www.stoutner.com/privacy-browser>.
- *
- * Privacy Browser 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.
- *
- * Privacy Browser 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 Privacy Browser. If not, see <http://www.gnu.org/licenses/>.
- */
-
-package com.stoutner.privacybrowser.dialogs
-
-import android.app.Dialog
-import android.content.Context
-import android.content.DialogInterface
-import android.content.Intent
-import android.os.Bundle
-import android.text.Editable
-import android.text.TextWatcher
-import android.view.WindowManager
-import android.widget.Button
-import android.widget.EditText
-
-import androidx.appcompat.app.AlertDialog
-import androidx.fragment.app.DialogFragment
-import androidx.preference.PreferenceManager
-
-import com.stoutner.privacybrowser.R
-
-// Define the class constants.
-private const val SAVE_TYPE = "save_type"
-
-class SaveDialog : DialogFragment() {
- // Declare the class variables.
- private lateinit var saveListener: SaveListener
- private lateinit var fileName: String
-
- // The public interface is used to send information back to the parent activity.
- interface SaveListener {
- fun onSave(saveType: Int, dialogFragment: DialogFragment)
- }
-
- override fun onAttach(context: Context) {
- // Run the default commands.
- super.onAttach(context)
-
- // Get a handle for the save listener from the launching context.
- saveListener = context as SaveListener
- }
-
- companion object {
- // Define the companion object constants. These can be moved to class constants once all of the code has transitioned to Kotlin.
- const val SAVE_LOGCAT = 0
- const val SAVE_ABOUT_VERSION_TEXT = 1
- const val SAVE_ABOUT_VERSION_IMAGE = 2
-
- // `@JvmStatic` will no longer be required once all the code has transitioned to Kotlin.
- @JvmStatic
- fun save(saveType: Int): SaveDialog {
- // Create an arguments bundle.
- val argumentsBundle = Bundle()
-
- // Store the arguments in the bundle.
- argumentsBundle.putInt(SAVE_TYPE, saveType)
-
- // Create a new instance of the save dialog.
- val saveDialog = SaveDialog()
-
- // Add the arguments bundle to the new dialog.
- saveDialog.arguments = argumentsBundle
-
- // Return the new dialog.
- return saveDialog
- }
- }
-
- override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
- // Get the arguments from the bundle.
- val saveType = requireArguments().getInt(SAVE_TYPE)
-
- // Use an alert dialog builder to create the alert dialog.
- val dialogBuilder = AlertDialog.Builder(requireContext(), R.style.PrivacyBrowserAlertDialog)
-
- // Set the title and the icon according to the save type.
- when (saveType) {
- SAVE_LOGCAT -> {
- // Set the title.
- dialogBuilder.setTitle(R.string.save_logcat)
-
- // Set the icon according to the theme.
- dialogBuilder.setIconAttribute(R.attr.saveBlueIcon)
- }
-
- SAVE_ABOUT_VERSION_TEXT -> {
- // Set the title.
- dialogBuilder.setTitle(R.string.save_text)
-
- // Set the icon according to the theme.
- dialogBuilder.setIconAttribute(R.attr.saveTextBlueIcon)
- }
-
- SAVE_ABOUT_VERSION_IMAGE -> {
- // Set the title.
- dialogBuilder.setTitle(R.string.save_image)
-
- // Set the icon according to the theme.
- dialogBuilder.setIconAttribute(R.attr.imagesBlueIcon)
- }
- }
-
- // Set the view.
- dialogBuilder.setView(R.layout.save_dialog)
-
- // Set the cancel button listener. Using `null` as the listener closes the dialog without doing anything else.
- dialogBuilder.setNegativeButton(R.string.cancel, null)
-
- // Set the save button listener.
- dialogBuilder.setPositiveButton(R.string.save) { _: DialogInterface?, _: Int ->
- // Return the dialog fragment to the parent activity.
- saveListener.onSave(saveType, this)
- }
-
- // Create an alert dialog from the builder.
- val alertDialog = dialogBuilder.create()
-
- // Get a handle for the shared preferences.
- val sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context)
-
- // Get the screenshot preference.
- val allowScreenshots = sharedPreferences.getBoolean(getString(R.string.allow_screenshots_key), false)
-
- // Disable screenshots if not allowed.
- if (!allowScreenshots) {
- alertDialog.window!!.addFlags(WindowManager.LayoutParams.FLAG_SECURE)
- }
-
- // The alert dialog must be shown before items in the layout can be modified.
- alertDialog.show()
-
- // Get handles for the layout items.
- val fileNameEditText = alertDialog.findViewById<EditText>(R.id.file_name_edittext)!!
- val browseButton = alertDialog.findViewById<Button>(R.id.browse_button)!!
- val saveButton = alertDialog.getButton(AlertDialog.BUTTON_POSITIVE)
-
- // Initially disable the save button.
- saveButton.isEnabled = false
-
- // Update the status of the save button when the file name changes.
- fileNameEditText.addTextChangedListener(object : TextWatcher {
- override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) {
- // Do nothing.
- }
-
- override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {
- // Do nothing.
- }
-
- override fun afterTextChanged(s: Editable) {
- // Get the current file name.
- val fileNameString = fileNameEditText.text.toString()
-
- // Enable the save button if the file name is populated.
- saveButton.isEnabled = fileNameString.isNotEmpty()
- }
- })
-
- // Set the file name according to the type.
- when (saveType) {
- SAVE_LOGCAT -> fileName = getString(R.string.privacy_browser_logcat_txt)
- SAVE_ABOUT_VERSION_TEXT -> fileName = getString(R.string.privacy_browser_version_txt)
- SAVE_ABOUT_VERSION_IMAGE -> fileName = getString(R.string.privacy_browser_version_png)
- }
-
- // Handle clicks on the browse button.
- browseButton.setOnClickListener {
- // Create the file picker intent.
- val browseIntent = Intent(Intent.ACTION_CREATE_DOCUMENT)
-
- // Set the intent MIME type to include all files so that everything is visible.
- browseIntent.type = "*/*"
-
- // Set the initial file name.
- browseIntent.putExtra(Intent.EXTRA_TITLE, fileName)
-
- // Request a file that can be opened.
- browseIntent.addCategory(Intent.CATEGORY_OPENABLE)
-
- // Launch the file picker. There is only one `startActivityForResult()`, so the request code is simply set to 0, but it must be run under `activity` so the response is processed correctly.
- requireActivity().startActivityForResult(browseIntent, 0)
- }
-
- // Return the alert dialog.
- return alertDialog
- }
-}
\ No newline at end of file
+++ /dev/null
-/*
- * Copyright © 2016-2020 Soren Stoutner <soren@stoutner.com>.
- *
- * This file is part of Privacy Browser <https://www.stoutner.com/privacy-browser>.
- *
- * Privacy Browser 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.
- *
- * Privacy Browser 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 Privacy Browser. If not, see <http://www.gnu.org/licenses/>.
- */
-
-package com.stoutner.privacybrowser.fragments;
-
-import android.annotation.SuppressLint;
-import android.app.Activity;
-import android.app.ActivityManager;
-import android.content.ClipData;
-import android.content.ClipboardManager;
-import android.content.Context;
-import android.content.Intent;
-import android.content.pm.PackageInfo;
-import android.content.pm.PackageManager;
-import android.content.pm.Signature;
-import android.content.res.Configuration;
-import android.os.Build;
-import android.os.Bundle;
-import android.os.Handler;
-import android.text.SpannableStringBuilder;
-import android.text.Spanned;
-import android.text.style.ForegroundColorSpan;
-import android.view.LayoutInflater;
-import android.view.Menu;
-import android.view.MenuInflater;
-import android.view.MenuItem;
-import android.view.View;
-import android.view.ViewGroup;
-import android.webkit.WebView;
-import android.widget.TextView;
-
-import com.google.android.material.snackbar.Snackbar;
-import com.stoutner.privacybrowser.BuildConfig;
-import com.stoutner.privacybrowser.R;
-import com.stoutner.privacybrowser.dialogs.SaveDialog;
-
-import java.io.ByteArrayInputStream;
-import java.io.InputStream;
-import java.math.BigInteger;
-import java.security.Principal;
-import java.security.cert.CertificateException;
-import java.security.cert.CertificateFactory;
-import java.security.cert.X509Certificate;
-import java.text.DateFormat;
-import java.text.NumberFormat;
-import java.util.Date;
-
-import androidx.annotation.NonNull;
-import androidx.fragment.app.DialogFragment;
-import androidx.fragment.app.Fragment;
-import androidx.webkit.WebViewCompat;
-
-public class AboutVersionFragment extends Fragment {
- // Declare the class constants.
- final static String BLOCKLIST_VERSIONS = "blocklist_versions";
- final long MEBIBYTE = 1048576;
-
- // Declare the class variables.
- private boolean updateMemoryUsageBoolean = true;
- private String[] blocklistVersions;
- private View aboutVersionLayout;
- private String appConsumedMemoryLabel;
- private String appAvailableMemoryLabel;
- private String appTotalMemoryLabel;
- private String appMaximumMemoryLabel;
- private String systemConsumedMemoryLabel;
- private String systemAvailableMemoryLabel;
- private String systemTotalMemoryLabel;
- private Runtime runtime;
- private ActivityManager activityManager;
- private ActivityManager.MemoryInfo memoryInfo;
- private NumberFormat numberFormat;
- private ForegroundColorSpan blueColorSpan;
-
- // Declare the class views.
- private TextView privacyBrowserTextView;
- private TextView versionTextView;
- private TextView hardwareTextView;
- private TextView brandTextView;
- private TextView manufacturerTextView;
- private TextView modelTextView;
- private TextView deviceTextView;
- private TextView bootloaderTextView;
- private TextView radioTextView;
- private TextView softwareTextView;
- private TextView androidTextView;
- private TextView securityPatchTextView;
- private TextView buildTextView;
- private TextView webViewProviderTextView;
- private TextView webViewVersionTextView;
- private TextView orbotTextView;
- private TextView i2pTextView;
- private TextView openKeychainTextView;
- private TextView memoryUsageTextView;
- private TextView appConsumedMemoryTextView;
- private TextView appAvailableMemoryTextView;
- private TextView appTotalMemoryTextView;
- private TextView appMaximumMemoryTextView;
- private TextView systemConsumedMemoryTextView;
- private TextView systemAvailableMemoryTextView;
- private TextView systemTotalMemoryTextView;
- private TextView blocklistsTextView;
- private TextView easyListTextView;
- private TextView easyPrivacyTextView;
- private TextView fanboyAnnoyanceTextView;
- private TextView fanboySocialTextView;
- private TextView ultraListTextView;
- private TextView ultraPrivacyTextView;
- private TextView packageSignatureTextView;
- private TextView certificateIssuerDnTextView;
- private TextView certificateSubjectDnTextView;
- private TextView certificateStartDateTextView;
- private TextView certificateEndDateTextView;
- private TextView certificateVersionTextView;
- private TextView certificateSerialNumberTextView;
- private TextView certificateSignatureAlgorithmTextView;
-
- public static AboutVersionFragment createTab(String[] blocklistVersions) {
- // Create an arguments bundle.
- Bundle argumentsBundle = new Bundle();
-
- // Store the arguments in the bundle.
- argumentsBundle.putStringArray(BLOCKLIST_VERSIONS, blocklistVersions);
-
- // Create a new instance of the tab fragment.
- AboutVersionFragment aboutVersionFragment = new AboutVersionFragment();
-
- // Add the arguments bundle to the fragment.
- aboutVersionFragment.setArguments(argumentsBundle);
-
- // Return the new fragment.
- return aboutVersionFragment;
- }
-
- @Override
- public void onCreate(Bundle savedInstanceState) {
- // Run the default commands.
- super.onCreate(savedInstanceState);
-
- // Get a handle for the arguments.
- Bundle arguments = getArguments();
-
- // Remove the incorrect lint warning below that the arguments might be null.
- assert arguments != null;
-
- // Store the arguments in class variables.
- blocklistVersions = arguments.getStringArray(BLOCKLIST_VERSIONS);
-
- // Enable the options menu for this fragment.
- setHasOptionsMenu(true);
- }
-
- @Override
- public View onCreateView(@NonNull LayoutInflater layoutInflater, ViewGroup container, Bundle savedInstanceState) {
- // Get a handle for the context.
- Context context = getContext();
-
- // Remove the incorrect lint warning below that the context might be null.
- assert context != null;
-
- // Get the current theme status.
- int currentThemeStatus = getResources().getConfiguration().uiMode & Configuration.UI_MODE_NIGHT_MASK;
-
- // Inflate the layout. Setting false at the end of inflater.inflate does not attach the inflated layout as a child of container. The fragment will take care of attaching the root automatically.
- aboutVersionLayout = layoutInflater.inflate(R.layout.about_version, container, false);
-
- // Get handles for the views.
- privacyBrowserTextView = aboutVersionLayout.findViewById(R.id.privacy_browser_textview);
- versionTextView = aboutVersionLayout.findViewById(R.id.version);
- hardwareTextView = aboutVersionLayout.findViewById(R.id.hardware);
- brandTextView = aboutVersionLayout.findViewById(R.id.brand);
- manufacturerTextView = aboutVersionLayout.findViewById(R.id.manufacturer);
- modelTextView = aboutVersionLayout.findViewById(R.id.model);
- deviceTextView = aboutVersionLayout.findViewById(R.id.device);
- bootloaderTextView = aboutVersionLayout.findViewById(R.id.bootloader);
- radioTextView = aboutVersionLayout.findViewById(R.id.radio);
- softwareTextView = aboutVersionLayout.findViewById(R.id.software);
- androidTextView = aboutVersionLayout.findViewById(R.id.android);
- securityPatchTextView = aboutVersionLayout.findViewById(R.id.security_patch);
- buildTextView = aboutVersionLayout.findViewById(R.id.build);
- webViewProviderTextView = aboutVersionLayout.findViewById(R.id.webview_provider);
- webViewVersionTextView = aboutVersionLayout.findViewById(R.id.webview_version);
- orbotTextView = aboutVersionLayout.findViewById(R.id.orbot);
- i2pTextView = aboutVersionLayout.findViewById(R.id.i2p);
- openKeychainTextView = aboutVersionLayout.findViewById(R.id.open_keychain);
- memoryUsageTextView = aboutVersionLayout.findViewById(R.id.memory_usage);
- appConsumedMemoryTextView = aboutVersionLayout.findViewById(R.id.app_consumed_memory);
- appAvailableMemoryTextView = aboutVersionLayout.findViewById(R.id.app_available_memory);
- appTotalMemoryTextView = aboutVersionLayout.findViewById(R.id.app_total_memory);
- appMaximumMemoryTextView = aboutVersionLayout.findViewById(R.id.app_maximum_memory);
- systemConsumedMemoryTextView = aboutVersionLayout.findViewById(R.id.system_consumed_memory);
- systemAvailableMemoryTextView = aboutVersionLayout.findViewById(R.id.system_available_memory);
- systemTotalMemoryTextView = aboutVersionLayout.findViewById(R.id.system_total_memory);
- blocklistsTextView = aboutVersionLayout.findViewById(R.id.blocklists);
- easyListTextView = aboutVersionLayout.findViewById(R.id.easylist);
- easyPrivacyTextView = aboutVersionLayout.findViewById(R.id.easyprivacy);
- fanboyAnnoyanceTextView = aboutVersionLayout.findViewById(R.id.fanboy_annoyance);
- fanboySocialTextView = aboutVersionLayout.findViewById(R.id.fanboy_social);
- ultraListTextView = aboutVersionLayout.findViewById(R.id.ultralist);
- ultraPrivacyTextView = aboutVersionLayout.findViewById(R.id.ultraprivacy);
- packageSignatureTextView = aboutVersionLayout.findViewById(R.id.package_signature);
- certificateIssuerDnTextView = aboutVersionLayout.findViewById(R.id.certificate_issuer_dn);
- certificateSubjectDnTextView = aboutVersionLayout.findViewById(R.id.certificate_subject_dn);
- certificateStartDateTextView = aboutVersionLayout.findViewById(R.id.certificate_start_date);
- certificateEndDateTextView = aboutVersionLayout.findViewById(R.id.certificate_end_date);
- certificateVersionTextView = aboutVersionLayout.findViewById(R.id.certificate_version);
- certificateSerialNumberTextView = aboutVersionLayout.findViewById(R.id.certificate_serial_number);
- certificateSignatureAlgorithmTextView = aboutVersionLayout.findViewById(R.id.certificate_signature_algorithm);
-
- // Setup the labels.
- String version = getString(R.string.version) + " " + BuildConfig.VERSION_NAME + " (" + getString(R.string.version_code) + " " + BuildConfig.VERSION_CODE + ")";
- String brandLabel = getString(R.string.brand) + " ";
- String manufacturerLabel = getString(R.string.manufacturer) + " ";
- String modelLabel = getString(R.string.model) + " ";
- String deviceLabel = getString(R.string.device) + " ";
- String bootloaderLabel = getString(R.string.bootloader) + " ";
- String androidLabel = getString(R.string.android) + " ";
- String buildLabel = getString(R.string.build) + " ";
- String webViewVersionLabel = getString(R.string.webview_version) + " ";
- appConsumedMemoryLabel = getString(R.string.app_consumed_memory) + " ";
- appAvailableMemoryLabel = getString(R.string.app_available_memory) + " ";
- appTotalMemoryLabel = getString(R.string.app_total_memory) + " ";
- appMaximumMemoryLabel = getString(R.string.app_maximum_memory) + " ";
- systemConsumedMemoryLabel = getString(R.string.system_consumed_memory) + " ";
- systemAvailableMemoryLabel = getString(R.string.system_available_memory) + " ";
- systemTotalMemoryLabel = getString(R.string.system_total_memory) + " ";
- String easyListLabel = getString(R.string.easylist_label) + " ";
- String easyPrivacyLabel = getString(R.string.easyprivacy_label) + " ";
- String fanboyAnnoyanceLabel = getString(R.string.fanboy_annoyance_label) + " ";
- String fanboySocialLabel = getString(R.string.fanboy_social_label) + " ";
- String ultraListLabel = getString(R.string.ultralist_label) + " ";
- String ultraPrivacyLabel = getString(R.string.ultraprivacy_label) + " ";
- String issuerDNLabel = getString(R.string.issuer_dn) + " ";
- String subjectDNLabel = getString(R.string.subject_dn) + " ";
- String startDateLabel = getString(R.string.start_date) + " ";
- String endDateLabel = getString(R.string.end_date) + " ";
- String certificateVersionLabel = getString(R.string.certificate_version) + " ";
- String serialNumberLabel = getString(R.string.serial_number) + " ";
- String signatureAlgorithmLabel = getString(R.string.signature_algorithm) + " ";
-
- // The WebView layout is only used to get the default user agent from `bare_webview`. It is not used to render content on the screen.
- // Once the minimum API >= 26 this can be accomplished with the WebView package info.
- View webViewLayout = layoutInflater.inflate(R.layout.bare_webview, container, false);
- WebView tabLayoutWebView = webViewLayout.findViewById(R.id.bare_webview);
- String userAgentString = tabLayoutWebView.getSettings().getUserAgentString();
-
- // Get the device's information and store it in strings.
- String brand = Build.BRAND;
- String manufacturer = Build.MANUFACTURER;
- String model = Build.MODEL;
- String device = Build.DEVICE;
- String bootloader = Build.BOOTLOADER;
- String radio = Build.getRadioVersion();
- String android = Build.VERSION.RELEASE + " (" + getString(R.string.api) + " " + Build.VERSION.SDK_INT + ")";
- String build = Build.DISPLAY;
- // Select the substring that begins after `Chrome/` and goes until the next ` `.
- String webView = userAgentString.substring(userAgentString.indexOf("Chrome/") + 7, userAgentString.indexOf(" ", userAgentString.indexOf("Chrome/")));
-
- // Get the Orbot version name if Orbot is installed.
- String orbot;
- try {
- // Store the version name.
- orbot = context.getPackageManager().getPackageInfo("org.torproject.android", 0).versionName;
- } catch (PackageManager.NameNotFoundException exception) { // Orbot is not installed.
- orbot = "";
- }
-
- // Get the I2P version name if I2P is installed.
- String i2p;
- try {
- // Store the version name.
- i2p = context.getPackageManager().getPackageInfo("net.i2p.android.router", 0).versionName;
- } catch (PackageManager.NameNotFoundException exception) { // I2P is not installed.
- i2p = "";
- }
-
- // Get the OpenKeychain version name if it is installed.
- String openKeychain;
- try {
- // Store the version name.
- openKeychain = context.getPackageManager().getPackageInfo("org.sufficientlysecure.keychain", 0).versionName;
- } catch (PackageManager.NameNotFoundException exception) { // OpenKeychain is not installed.
- openKeychain = "";
- }
-
- // Create a spannable string builder for the hardware and software text views that needs multiple colors of text.
- SpannableStringBuilder brandStringBuilder = new SpannableStringBuilder(brandLabel + brand);
- SpannableStringBuilder manufacturerStringBuilder = new SpannableStringBuilder(manufacturerLabel + manufacturer);
- SpannableStringBuilder modelStringBuilder = new SpannableStringBuilder(modelLabel + model);
- SpannableStringBuilder deviceStringBuilder = new SpannableStringBuilder(deviceLabel + device);
- SpannableStringBuilder bootloaderStringBuilder = new SpannableStringBuilder(bootloaderLabel + bootloader);
- SpannableStringBuilder androidStringBuilder = new SpannableStringBuilder(androidLabel + android);
- SpannableStringBuilder buildStringBuilder = new SpannableStringBuilder(buildLabel + build);
- SpannableStringBuilder webViewVersionStringBuilder = new SpannableStringBuilder(webViewVersionLabel + webView);
- SpannableStringBuilder easyListStringBuilder = new SpannableStringBuilder(easyListLabel + blocklistVersions[0]);
- SpannableStringBuilder easyPrivacyStringBuilder = new SpannableStringBuilder(easyPrivacyLabel + blocklistVersions[1]);
- SpannableStringBuilder fanboyAnnoyanceStringBuilder = new SpannableStringBuilder(fanboyAnnoyanceLabel + blocklistVersions[2]);
- SpannableStringBuilder fanboySocialStringBuilder = new SpannableStringBuilder(fanboySocialLabel + blocklistVersions[3]);
- SpannableStringBuilder ultraListStringBuilder = new SpannableStringBuilder(ultraListLabel + blocklistVersions[4]);
- SpannableStringBuilder ultraPrivacyStringBuilder = new SpannableStringBuilder(ultraPrivacyLabel + blocklistVersions[5]);
-
- // Set the blue color span according to the theme. The deprecated `getResources()` must be used until the minimum API >= 23.
- if (currentThemeStatus == Configuration.UI_MODE_NIGHT_NO) {
- blueColorSpan = new ForegroundColorSpan(getResources().getColor(R.color.blue_700));
- } else {
- blueColorSpan = new ForegroundColorSpan(getResources().getColor(R.color.violet_700));
- }
-
- // Setup the spans to display the device information in blue. `SPAN_INCLUSIVE_INCLUSIVE` allows the span to grow in either direction.
- brandStringBuilder.setSpan(blueColorSpan, brandLabel.length(), brandStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- manufacturerStringBuilder.setSpan(blueColorSpan, manufacturerLabel.length(), manufacturerStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- modelStringBuilder.setSpan(blueColorSpan, modelLabel.length(), modelStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- deviceStringBuilder.setSpan(blueColorSpan, deviceLabel.length(), deviceStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- bootloaderStringBuilder.setSpan(blueColorSpan, bootloaderLabel.length(), bootloaderStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- androidStringBuilder.setSpan(blueColorSpan, androidLabel.length(), androidStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- buildStringBuilder.setSpan(blueColorSpan, buildLabel.length(), buildStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- webViewVersionStringBuilder.setSpan(blueColorSpan, webViewVersionLabel.length(), webViewVersionStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- easyListStringBuilder.setSpan(blueColorSpan, easyListLabel.length(), easyListStringBuilder.length(), Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
- easyPrivacyStringBuilder.setSpan(blueColorSpan, easyPrivacyLabel.length(), easyPrivacyStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- fanboyAnnoyanceStringBuilder.setSpan(blueColorSpan, fanboyAnnoyanceLabel.length(), fanboyAnnoyanceStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- fanboySocialStringBuilder.setSpan(blueColorSpan, fanboySocialLabel.length(), fanboySocialStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- ultraListStringBuilder.setSpan(blueColorSpan, ultraListLabel.length(), ultraListStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- ultraPrivacyStringBuilder.setSpan(blueColorSpan, ultraPrivacyLabel.length(), ultraPrivacyStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
-
- // Display the strings in the text boxes.
- versionTextView.setText(version);
- brandTextView.setText(brandStringBuilder);
- manufacturerTextView.setText(manufacturerStringBuilder);
- modelTextView.setText(modelStringBuilder);
- deviceTextView.setText(deviceStringBuilder);
- bootloaderTextView.setText(bootloaderStringBuilder);
- androidTextView.setText(androidStringBuilder);
- buildTextView.setText(buildStringBuilder);
- webViewVersionTextView.setText(webViewVersionStringBuilder);
- easyListTextView.setText(easyListStringBuilder);
- easyPrivacyTextView.setText(easyPrivacyStringBuilder);
- fanboyAnnoyanceTextView.setText(fanboyAnnoyanceStringBuilder);
- fanboySocialTextView.setText(fanboySocialStringBuilder);
- ultraListTextView.setText(ultraListStringBuilder);
- ultraPrivacyTextView.setText(ultraPrivacyStringBuilder);
-
- // Only populate the radio text view if there is a radio in the device.
- // Null must be checked because some Samsung tablets report a null value for the radio instead of an empty string. Grrrr. <https://redmine.stoutner.com/issues/701>
- if ((radio != null) && !radio.isEmpty()) {
- String radioLabel = getString(R.string.radio) + " ";
- SpannableStringBuilder radioStringBuilder = new SpannableStringBuilder(radioLabel + radio);
- radioStringBuilder.setSpan(blueColorSpan, radioLabel.length(), radioStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- radioTextView.setText(radioStringBuilder);
- } else { // This device does not have a radio.
- radioTextView.setVisibility(View.GONE);
- }
-
- // Build.VERSION.SECURITY_PATCH is only available for SDK_INT >= 23.
- if (Build.VERSION.SDK_INT >= 23) {
- String securityPatchLabel = getString(R.string.security_patch) + " ";
- String securityPatch = Build.VERSION.SECURITY_PATCH;
- SpannableStringBuilder securityPatchStringBuilder = new SpannableStringBuilder(securityPatchLabel + securityPatch);
- securityPatchStringBuilder.setSpan(blueColorSpan, securityPatchLabel.length(), securityPatchStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- securityPatchTextView.setText(securityPatchStringBuilder);
- } else { // The API < 23.
- // Hide the security patch text view.
- securityPatchTextView.setVisibility(View.GONE);
- }
-
- // Only populate the WebView provider if the SDK >= 21.
- if (Build.VERSION.SDK_INT >= 21) {
- // Create the WebView provider label.
- String webViewProviderLabel = getString(R.string.webview_provider) + " ";
-
- // Get the current WebView package info.
- PackageInfo webViewPackageInfo = WebViewCompat.getCurrentWebViewPackage(context);
-
- // Remove the warning below that the package info might be null.
- assert webViewPackageInfo != null;
-
- // Get the WebView provider name.
- String webViewPackageName = webViewPackageInfo.packageName;
-
- // Create the spannable string builder.
- SpannableStringBuilder webViewProviderStringBuilder = new SpannableStringBuilder(webViewProviderLabel + webViewPackageName);
-
- // Apply the coloration.
- webViewProviderStringBuilder.setSpan(blueColorSpan, webViewProviderLabel.length(), webViewProviderStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
-
- // Display the WebView provider.
- webViewProviderTextView.setText(webViewProviderStringBuilder);
- } else { // The API < 21.
- // Hide the WebView provider text view.
- webViewProviderTextView.setVisibility(View.GONE);
- }
-
- // Only populate the Orbot text view if it is installed.
- if (!orbot.isEmpty()) {
- String orbotLabel = getString(R.string.orbot) + " ";
- SpannableStringBuilder orbotStringBuilder = new SpannableStringBuilder(orbotLabel + orbot);
- orbotStringBuilder.setSpan(blueColorSpan, orbotLabel.length(), orbotStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- orbotTextView.setText(orbotStringBuilder);
- } else { // Orbot is not installed.
- orbotTextView.setVisibility(View.GONE);
- }
-
- // Only populate the I2P text view if it is installed.
- if (!i2p.isEmpty()) {
- String i2pLabel = getString(R.string.i2p) + " ";
- SpannableStringBuilder i2pStringBuilder = new SpannableStringBuilder(i2pLabel + i2p);
- i2pStringBuilder.setSpan(blueColorSpan, i2pLabel.length(), i2pStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- i2pTextView.setText(i2pStringBuilder);
- } else { // I2P is not installed.
- i2pTextView.setVisibility(View.GONE);
- }
-
- // Only populate the OpenKeychain text view if it is installed.
- if (!openKeychain.isEmpty()) {
- String openKeychainLabel = getString(R.string.openkeychain) + " ";
- SpannableStringBuilder openKeychainStringBuilder = new SpannableStringBuilder(openKeychainLabel + openKeychain);
- openKeychainStringBuilder.setSpan(blueColorSpan, openKeychainLabel.length(), openKeychainStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- openKeychainTextView.setText(openKeychainStringBuilder);
- } else { //OpenKeychain is not installed.
- openKeychainTextView.setVisibility(View.GONE);
- }
-
- // Display the package signature.
- try {
- // Get the first package signature. Suppress the lint warning about the need to be careful in implementing comparison of certificates for security purposes.
- @SuppressLint("PackageManagerGetSignatures") Signature packageSignature = context.getPackageManager().getPackageInfo(context.getPackageName(),
- PackageManager.GET_SIGNATURES).signatures[0];
-
- // Convert the signature to a byte array input stream.
- InputStream certificateByteArrayInputStream = new ByteArrayInputStream(packageSignature.toByteArray());
-
- // Display the certificate information on the screen.
- try {
- // Instantiate a `CertificateFactory`.
- CertificateFactory certificateFactory = CertificateFactory.getInstance("X509");
-
- // Generate an `X509Certificate`.
- X509Certificate x509Certificate = (X509Certificate) certificateFactory.generateCertificate(certificateByteArrayInputStream);
-
- // Store the individual sections of the certificate that we are interested in.
- Principal issuerDNPrincipal = x509Certificate.getIssuerDN();
- Principal subjectDNPrincipal = x509Certificate.getSubjectDN();
- Date startDate = x509Certificate.getNotBefore();
- Date endDate = x509Certificate.getNotAfter();
- int certificateVersion = x509Certificate.getVersion();
- BigInteger serialNumberBigInteger = x509Certificate.getSerialNumber();
- String signatureAlgorithmNameString = x509Certificate.getSigAlgName();
-
- // Create a `SpannableStringBuilder` for each `TextView` that needs multiple colors of text.
- SpannableStringBuilder issuerDNStringBuilder = new SpannableStringBuilder(issuerDNLabel + issuerDNPrincipal.toString());
- SpannableStringBuilder subjectDNStringBuilder = new SpannableStringBuilder(subjectDNLabel + subjectDNPrincipal.toString());
- SpannableStringBuilder startDateStringBuilder = new SpannableStringBuilder(startDateLabel + DateFormat.getDateTimeInstance(DateFormat.MEDIUM, DateFormat.LONG).format(startDate));
- SpannableStringBuilder endDataStringBuilder = new SpannableStringBuilder(endDateLabel + DateFormat.getDateTimeInstance(DateFormat.MEDIUM, DateFormat.LONG).format(endDate));
- SpannableStringBuilder certificateVersionStringBuilder = new SpannableStringBuilder(certificateVersionLabel + certificateVersion);
- SpannableStringBuilder serialNumberStringBuilder = new SpannableStringBuilder(serialNumberLabel + serialNumberBigInteger);
- SpannableStringBuilder signatureAlgorithmStringBuilder = new SpannableStringBuilder(signatureAlgorithmLabel + signatureAlgorithmNameString);
-
- // Setup the spans to display the device information in blue. `SPAN_INCLUSIVE_INCLUSIVE` allows the span to grow in either direction.
- issuerDNStringBuilder.setSpan(blueColorSpan, issuerDNLabel.length(), issuerDNStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- subjectDNStringBuilder.setSpan(blueColorSpan, subjectDNLabel.length(), subjectDNStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- startDateStringBuilder.setSpan(blueColorSpan, startDateLabel.length(), startDateStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- endDataStringBuilder.setSpan(blueColorSpan, endDateLabel.length(), endDataStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- certificateVersionStringBuilder.setSpan(blueColorSpan, certificateVersionLabel.length(), certificateVersionStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- serialNumberStringBuilder.setSpan(blueColorSpan, serialNumberLabel.length(), serialNumberStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- signatureAlgorithmStringBuilder.setSpan(blueColorSpan, signatureAlgorithmLabel.length(), signatureAlgorithmStringBuilder.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
-
- // Display the strings in the text boxes.
- certificateIssuerDnTextView.setText(issuerDNStringBuilder);
- certificateSubjectDnTextView.setText(subjectDNStringBuilder);
- certificateStartDateTextView.setText(startDateStringBuilder);
- certificateEndDateTextView.setText(endDataStringBuilder);
- certificateVersionTextView.setText(certificateVersionStringBuilder);
- certificateSerialNumberTextView.setText(serialNumberStringBuilder);
- certificateSignatureAlgorithmTextView.setText(signatureAlgorithmStringBuilder);
- } catch (CertificateException e) {
- // Do nothing if there is a certificate error.
- }
-
- // Get a handle for the runtime.
- runtime = Runtime.getRuntime();
-
- // Get a handle for the activity.
- Activity activity = getActivity();
-
- // Remove the incorrect lint warning below that the activity might be null.
- assert activity != null;
-
- // Get a handle for the activity manager.
- activityManager = (ActivityManager) activity.getSystemService(Context.ACTIVITY_SERVICE);
-
- // Remove the incorrect lint warning below that the activity manager might be null.
- assert activityManager != null;
-
- // Instantiate a memory info variable.
- memoryInfo = new ActivityManager.MemoryInfo();
-
- // Define a number format.
- numberFormat = NumberFormat.getInstance();
-
- // Set the minimum and maximum number of fraction digits.
- numberFormat.setMinimumFractionDigits(2);
- numberFormat.setMaximumFractionDigits(2);
-
- // Update the memory usage.
- updateMemoryUsage(getActivity());
- } catch (PackageManager.NameNotFoundException e) {
- // Do nothing if `PackageManager` says Privacy Browser isn't installed.
- }
-
- // Scroll the tab if the saved instance state is not null.
- if (savedInstanceState != null) {
- aboutVersionLayout.post(() -> {
- aboutVersionLayout.setScrollX(savedInstanceState.getInt("scroll_x"));
- aboutVersionLayout.setScrollY(savedInstanceState.getInt("scroll_y"));
- });
- }
-
- // Return the tab layout.
- return aboutVersionLayout;
- }
-
- @Override
- public void onCreateOptionsMenu(@NonNull Menu menu, @NonNull MenuInflater menuInflater) {
- // Inflate the about version menu.
- menuInflater.inflate(R.menu.about_version_options_menu, menu);
-
- // Run the default commands.
- super.onCreateOptionsMenu(menu, menuInflater);
- }
-
- @Override
- public boolean onOptionsItemSelected(@NonNull MenuItem menuItem) {
- // Remove the incorrect lint warning below that the activity might be null.
- assert getActivity() != null;
-
- // Get the ID of the menu item that was selected.
- int menuItemId = menuItem.getItemId();
-
- // Run the appropriate commands.
- if (menuItemId == R.id.copy) { // Copy.
- // Get the about version string.
- String aboutVersionString = getAboutVersionString();
-
- // Get a handle for the clipboard manager.
- ClipboardManager clipboardManager = (ClipboardManager) getActivity().getSystemService(Context.CLIPBOARD_SERVICE);
-
- // Remove the incorrect lint error below that the clipboard manager might be null.
- assert clipboardManager != null;
-
- // Save the about version string in a clip data.
- ClipData aboutVersionClipData = ClipData.newPlainText(getString(R.string.about), aboutVersionString);
-
- // Place the clip data on the clipboard.
- clipboardManager.setPrimaryClip(aboutVersionClipData);
-
- // Display a snackbar.
- Snackbar.make(aboutVersionLayout, R.string.version_info_copied, Snackbar.LENGTH_SHORT).show();
-
- // Consume the event.
- return true;
- } else if (menuItemId == R.id.share) { // Share.
- // Get the about version string.
- String aboutString = getAboutVersionString();
-
- // Create an email intent.
- Intent emailIntent = new Intent(Intent.ACTION_SEND);
-
- // Add the about version string to the intent.
- emailIntent.putExtra(Intent.EXTRA_TEXT, aboutString);
-
- // Set the MIME type.
- emailIntent.setType("text/plain");
-
- // Set the intent to open in a new task.
- emailIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
-
- // Make it so.
- startActivity(Intent.createChooser(emailIntent, getString(R.string.share)));
-
- // Consume the event.
- return true;
- } else if (menuItemId == R.id.save_text) { // Save text.
- // Instantiate the save alert dialog.
- DialogFragment saveTextDialogFragment = SaveDialog.save(SaveDialog.SAVE_ABOUT_VERSION_TEXT);
-
- // Show the save alert dialog.
- saveTextDialogFragment.show(getActivity().getSupportFragmentManager(), getString(R.string.save_dialog));
-
- // Consume the event.
- return true;
- } else if (menuItemId == R.id.save_image) { // Save image.
- // Instantiate the save alert dialog.
- DialogFragment saveImageDialogFragment = SaveDialog.save(SaveDialog.SAVE_ABOUT_VERSION_IMAGE);
-
- // Show the save alert dialog.
- saveImageDialogFragment.show(getActivity().getSupportFragmentManager(), getString(R.string.save_dialog));
-
- // Consume the event.
- return true;
- } else { // The home button was selected.
- // Return the parent class.
- return super.onOptionsItemSelected(menuItem);
- }
- }
-
- @Override
- public void onSaveInstanceState(@NonNull Bundle savedInstanceState) {
- // Run the default commands.
- super.onSaveInstanceState(savedInstanceState);
-
- // Save the scroll positions if the layout is not null, which can happen if a tab is not currently selected.
- if (aboutVersionLayout != null) {
- savedInstanceState.putInt("scroll_x", aboutVersionLayout.getScrollX());
- savedInstanceState.putInt("scroll_y", aboutVersionLayout.getScrollY());
- }
- }
-
- @Override
- public void onPause() {
- // Run the default commands.
- super.onPause();
-
- // Pause the updating of the memory usage.
- updateMemoryUsageBoolean = false;
- }
-
- @Override
- public void onResume() {
- // Run the default commands.
- super.onResume();
-
- // Resume the updating of the memory usage.
- updateMemoryUsageBoolean = true;
- }
-
- public void updateMemoryUsage(Activity activity) {
- try {
- // Update the memory usage if enabled.
- if (updateMemoryUsageBoolean) {
- // Populate the memory info variable.
- activityManager.getMemoryInfo(memoryInfo);
-
- // Get the app memory information.
- long appAvailableMemoryLong = runtime.freeMemory();
- long appTotalMemoryLong = runtime.totalMemory();
- long appMaximumMemoryLong = runtime.maxMemory();
-
- // Calculate the app consumed memory.
- long appConsumedMemoryLong = appTotalMemoryLong - appAvailableMemoryLong;
-
- // Get the system memory information.
- long systemTotalMemoryLong = memoryInfo.totalMem;
- long systemAvailableMemoryLong = memoryInfo.availMem;
-
- // Calculate the system consumed memory.
- long systemConsumedMemoryLong = systemTotalMemoryLong - systemAvailableMemoryLong;
-
- // Convert the memory information into mebibytes.
- float appConsumedMemoryFloat = (float) appConsumedMemoryLong / MEBIBYTE;
- float appAvailableMemoryFloat = (float) appAvailableMemoryLong / MEBIBYTE;
- float appTotalMemoryFloat = (float) appTotalMemoryLong / MEBIBYTE;
- float appMaximumMemoryFloat = (float) appMaximumMemoryLong / MEBIBYTE;
- float systemConsumedMemoryFloat = (float) systemConsumedMemoryLong / MEBIBYTE;
- float systemAvailableMemoryFloat = (float) systemAvailableMemoryLong / MEBIBYTE;
- float systemTotalMemoryFloat = (float) systemTotalMemoryLong / MEBIBYTE;
-
- // Get the mebibyte string.
- String mebibyte = getString(R.string.mebibyte);
-
- // Calculate the mebibyte length.
- int mebibyteLength = mebibyte.length();
-
- // Create spannable string builders.
- SpannableStringBuilder appConsumedMemoryStringBuilder = new SpannableStringBuilder(appConsumedMemoryLabel + numberFormat.format(appConsumedMemoryFloat) + " " + mebibyte);
- SpannableStringBuilder appAvailableMemoryStringBuilder = new SpannableStringBuilder(appAvailableMemoryLabel + numberFormat.format(appAvailableMemoryFloat) + " " + mebibyte);
- SpannableStringBuilder appTotalMemoryStringBuilder = new SpannableStringBuilder(appTotalMemoryLabel + numberFormat.format(appTotalMemoryFloat) + " " + mebibyte);
- SpannableStringBuilder appMaximumMemoryStringBuilder = new SpannableStringBuilder(appMaximumMemoryLabel + numberFormat.format(appMaximumMemoryFloat) + " " + mebibyte);
- SpannableStringBuilder systemConsumedMemoryStringBuilder = new SpannableStringBuilder(systemConsumedMemoryLabel + numberFormat.format(systemConsumedMemoryFloat) + " " + mebibyte);
- SpannableStringBuilder systemAvailableMemoryStringBuilder = new SpannableStringBuilder(systemAvailableMemoryLabel + numberFormat.format(systemAvailableMemoryFloat) + " " + mebibyte);
- SpannableStringBuilder systemTotalMemoryStringBuilder = new SpannableStringBuilder(systemTotalMemoryLabel + numberFormat.format(systemTotalMemoryFloat) + " " + mebibyte);
-
- // Setup the spans to display the memory information in blue. `SPAN_INCLUSIVE_INCLUSIVE` allows the span to grow in either direction.
- appConsumedMemoryStringBuilder.setSpan(blueColorSpan, appConsumedMemoryLabel.length(), appConsumedMemoryStringBuilder.length() - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- appAvailableMemoryStringBuilder.setSpan(blueColorSpan, appAvailableMemoryLabel.length(), appAvailableMemoryStringBuilder.length() - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- appTotalMemoryStringBuilder.setSpan(blueColorSpan, appTotalMemoryLabel.length(), appTotalMemoryStringBuilder.length() - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- appMaximumMemoryStringBuilder.setSpan(blueColorSpan, appMaximumMemoryLabel.length(), appMaximumMemoryStringBuilder.length() - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- systemConsumedMemoryStringBuilder.setSpan(blueColorSpan, systemConsumedMemoryLabel.length(), systemConsumedMemoryStringBuilder.length() - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- systemAvailableMemoryStringBuilder.setSpan(blueColorSpan, systemAvailableMemoryLabel.length(), systemAvailableMemoryStringBuilder.length() - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
- systemTotalMemoryStringBuilder.setSpan(blueColorSpan, systemTotalMemoryLabel.length(), systemTotalMemoryStringBuilder.length() - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
-
- // Display the string in the text boxes.
- appConsumedMemoryTextView.setText(appConsumedMemoryStringBuilder);
- appAvailableMemoryTextView.setText(appAvailableMemoryStringBuilder);
- appTotalMemoryTextView.setText(appTotalMemoryStringBuilder);
- appMaximumMemoryTextView.setText(appMaximumMemoryStringBuilder);
- systemConsumedMemoryTextView.setText(systemConsumedMemoryStringBuilder);
- systemAvailableMemoryTextView.setText(systemAvailableMemoryStringBuilder);
- systemTotalMemoryTextView.setText(systemTotalMemoryStringBuilder);
- }
-
- // Schedule another memory update if the activity has not been destroyed.
- if (!activity.isDestroyed()) {
- // Create a handler to update the memory usage.
- Handler updateMemoryUsageHandler = new Handler();
-
- // Create a runnable to update the memory usage.
- Runnable updateMemoryUsageRunnable = () -> updateMemoryUsage(activity);
-
- // Update the memory usage after 1000 milliseconds
- updateMemoryUsageHandler.postDelayed(updateMemoryUsageRunnable, 1000);
- }
- } catch (Exception exception) {
- // Do nothing.
- }
- }
-
- public String getAboutVersionString() {
- // Initialize an about version string builder.
- StringBuilder aboutVersionStringBuilder = new StringBuilder();
-
- // Populate the about version string builder.
- aboutVersionStringBuilder.append(privacyBrowserTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(versionTextView.getText());
- aboutVersionStringBuilder.append("\n\n");
- aboutVersionStringBuilder.append(hardwareTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(brandTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(manufacturerTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(modelTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(deviceTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(bootloaderTextView.getText());
- aboutVersionStringBuilder.append("\n");
- if (radioTextView.getVisibility() == View.VISIBLE) {
- aboutVersionStringBuilder.append(radioTextView.getText());
- aboutVersionStringBuilder.append("\n");
- }
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(softwareTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(androidTextView.getText());
- aboutVersionStringBuilder.append("\n");
- if (securityPatchTextView.getVisibility() == View.VISIBLE) {
- aboutVersionStringBuilder.append(securityPatchTextView.getText());
- aboutVersionStringBuilder.append("\n");
- }
- aboutVersionStringBuilder.append(buildTextView.getText());
- aboutVersionStringBuilder.append("\n");
- if (webViewProviderTextView.getVisibility() == View.VISIBLE) {
- aboutVersionStringBuilder.append(webViewProviderTextView.getText());
- aboutVersionStringBuilder.append("\n");
- }
- aboutVersionStringBuilder.append(webViewVersionTextView.getText());
- aboutVersionStringBuilder.append("\n");
- if (orbotTextView.getVisibility() == View.VISIBLE) {
- aboutVersionStringBuilder.append(orbotTextView.getText());
- aboutVersionStringBuilder.append("\n");
- }
- if (i2pTextView.getVisibility() == View.VISIBLE) {
- aboutVersionStringBuilder.append(i2pTextView.getText());
- aboutVersionStringBuilder.append("\n");
- }
- if (openKeychainTextView.getVisibility() == View.VISIBLE) {
- aboutVersionStringBuilder.append(openKeychainTextView.getText());
- aboutVersionStringBuilder.append("\n");
- }
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(memoryUsageTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(appConsumedMemoryTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(appAvailableMemoryTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(appTotalMemoryTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(appMaximumMemoryTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(systemConsumedMemoryTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(systemAvailableMemoryTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(systemTotalMemoryTextView.getText());
- aboutVersionStringBuilder.append("\n\n");
- aboutVersionStringBuilder.append(blocklistsTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(easyListTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(easyPrivacyTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(fanboyAnnoyanceTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(fanboySocialTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(ultraListTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(ultraPrivacyTextView.getText());
- aboutVersionStringBuilder.append("\n\n");
- aboutVersionStringBuilder.append(packageSignatureTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(certificateIssuerDnTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(certificateSubjectDnTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(certificateStartDateTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(certificateEndDateTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(certificateVersionTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(certificateSerialNumberTextView.getText());
- aboutVersionStringBuilder.append("\n");
- aboutVersionStringBuilder.append(certificateSignatureAlgorithmTextView.getText());
-
- // Return the string.
- return aboutVersionStringBuilder.toString();
- }
-}
\ No newline at end of file
--- /dev/null
+/*
+ * Copyright © 2016-2021 Soren Stoutner <soren@stoutner.com>.
+ *
+ * This file is part of Privacy Browser <https://www.stoutner.com/privacy-browser>.
+ *
+ * Privacy Browser 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.
+ *
+ * Privacy Browser 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 Privacy Browser. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package com.stoutner.privacybrowser.fragments
+
+import android.annotation.SuppressLint
+import android.app.Activity
+import android.app.ActivityManager
+import android.content.ClipData
+import android.content.ClipboardManager
+import android.content.Context
+import android.content.Intent
+import android.content.pm.PackageManager
+import android.net.Uri
+import android.os.Build
+import android.os.Bundle
+import android.os.Handler
+import android.os.Looper
+import android.provider.OpenableColumns
+import android.text.SpannableStringBuilder
+import android.text.Spanned
+import android.text.style.ForegroundColorSpan
+import android.view.LayoutInflater
+import android.view.Menu
+import android.view.MenuInflater
+import android.view.MenuItem
+import android.view.View
+import android.view.ViewGroup
+import android.webkit.WebView
+import android.widget.TextView
+
+import androidx.activity.result.contract.ActivityResultContracts
+import androidx.fragment.app.Fragment
+import androidx.webkit.WebViewCompat
+
+import com.google.android.material.snackbar.Snackbar
+
+import com.stoutner.privacybrowser.R
+import com.stoutner.privacybrowser.BuildConfig
+import com.stoutner.privacybrowser.asynctasks.SaveAboutVersionImage
+
+import java.io.ByteArrayInputStream
+import java.io.InputStream
+import java.lang.Exception
+import java.nio.charset.StandardCharsets
+import java.security.cert.CertificateException
+import java.security.cert.CertificateFactory
+import java.security.cert.X509Certificate
+import java.text.DateFormat
+import java.text.NumberFormat
+
+import kotlin.text.StringBuilder
+
+// Define the class constants.
+private const val BLOCKLIST_VERSIONS = "blocklist_versions"
+private const val MEBIBYTE = 1048576
+
+class AboutVersionFragment : Fragment() {
+ // Define the class variables.
+ private var updateMemoryUsageBoolean = true
+
+ // Declare the class variables.
+ private lateinit var blocklistVersions: Array<String>
+ private lateinit var aboutVersionLayout: View
+ private lateinit var appConsumedMemoryLabel: String
+ private lateinit var appAvailableMemoryLabel: String
+ private lateinit var appTotalMemoryLabel: String
+ private lateinit var appMaximumMemoryLabel: String
+ private lateinit var systemConsumedMemoryLabel: String
+ private lateinit var systemAvailableMemoryLabel: String
+ private lateinit var systemTotalMemoryLabel: String
+ private lateinit var runtime: Runtime
+ private lateinit var activityManager: ActivityManager
+ private lateinit var memoryInfo: ActivityManager.MemoryInfo
+ private lateinit var numberFormat: NumberFormat
+ private lateinit var blueColorSpan: ForegroundColorSpan
+
+ // Declare the class views.
+ private lateinit var privacyBrowserTextView: TextView
+ private lateinit var versionTextView: TextView
+ private lateinit var hardwareTextView: TextView
+ private lateinit var brandTextView: TextView
+ private lateinit var manufacturerTextView: TextView
+ private lateinit var modelTextView: TextView
+ private lateinit var deviceTextView: TextView
+ private lateinit var bootloaderTextView: TextView
+ private lateinit var radioTextView: TextView
+ private lateinit var softwareTextView: TextView
+ private lateinit var androidTextView: TextView
+ private lateinit var securityPatchTextView: TextView
+ private lateinit var buildTextView: TextView
+ private lateinit var webViewProviderTextView: TextView
+ private lateinit var webViewVersionTextView: TextView
+ private lateinit var orbotTextView: TextView
+ private lateinit var i2pTextView: TextView
+ private lateinit var openKeychainTextView: TextView
+ private lateinit var memoryUsageTextView: TextView
+ private lateinit var appConsumedMemoryTextView: TextView
+ private lateinit var appAvailableMemoryTextView: TextView
+ private lateinit var appTotalMemoryTextView: TextView
+ private lateinit var appMaximumMemoryTextView: TextView
+ private lateinit var systemConsumedMemoryTextView: TextView
+ private lateinit var systemAvailableMemoryTextView: TextView
+ private lateinit var systemTotalMemoryTextView: TextView
+ private lateinit var blocklistsTextView: TextView
+ private lateinit var easyListTextView: TextView
+ private lateinit var easyPrivacyTextView: TextView
+ private lateinit var fanboyAnnoyanceTextView: TextView
+ private lateinit var fanboySocialTextView: TextView
+ private lateinit var ultraListTextView: TextView
+ private lateinit var ultraPrivacyTextView: TextView
+ private lateinit var packageSignatureTextView: TextView
+ private lateinit var certificateIssuerDnTextView: TextView
+ private lateinit var certificateSubjectDnTextView: TextView
+ private lateinit var certificateStartDateTextView: TextView
+ private lateinit var certificateEndDateTextView: TextView
+ private lateinit var certificateVersionTextView: TextView
+ private lateinit var certificateSerialNumberTextView: TextView
+ private lateinit var certificateSignatureAlgorithmTextView: TextView
+
+ companion object {
+ fun createTab(blocklistVersions: Array<String>): AboutVersionFragment {
+ // Create an arguments bundle.
+ val argumentsBundle = Bundle()
+
+ // Store the arguments in the bundle.
+ argumentsBundle.putStringArray(BLOCKLIST_VERSIONS, blocklistVersions)
+
+ // Create a new instance of the tab fragment.
+ val aboutVersionFragment = AboutVersionFragment()
+
+ // Add the arguments bundle to the fragment.
+ aboutVersionFragment.arguments = argumentsBundle
+
+ // Return the new fragment.
+ return aboutVersionFragment
+ }
+ }
+
+ // Define the save about version text activity result launcher. It must be defined before `onCreate()` is run or the app will crash.
+ private val saveAboutVersionTextActivityResultLauncher = registerForActivityResult(ActivityResultContracts.CreateDocument()) { fileNameUri: Uri? ->
+ // Only save the file if the URI is not null, which happens if the user exited the file picker by pressing back.
+ if (fileNameUri != null) {
+ try {
+ // Get the about version string.
+ val aboutVersionString = getAboutVersionString()
+
+ // Open an output stream.
+ val outputStream = requireActivity().contentResolver.openOutputStream(fileNameUri)!!
+
+ // Write the about version string to the output stream.
+ outputStream.write(aboutVersionString.toByteArray(StandardCharsets.UTF_8))
+
+ // Close the output stream.
+ outputStream.close()
+
+ // Initialize the file name string from the file name URI last path segment.
+ var fileNameString = fileNameUri.lastPathSegment
+
+ // Query the exact file name if the API >= 26.
+ if (Build.VERSION.SDK_INT >= 26) {
+ // Get a cursor from the content resolver.
+ val contentResolverCursor = requireActivity().contentResolver.query(fileNameUri, null, null, null)!!
+
+ // Move to the first row.
+ contentResolverCursor.moveToFirst()
+
+ // Get the file name from the cursor.
+ fileNameString = contentResolverCursor.getString(contentResolverCursor.getColumnIndex(OpenableColumns.DISPLAY_NAME))
+
+ // Close the cursor.
+ contentResolverCursor.close()
+ }
+
+ // Display a snackbar with the saved logcat information.
+ Snackbar.make(aboutVersionLayout, getString(R.string.saved, fileNameString), Snackbar.LENGTH_SHORT).show()
+ } catch (exception: Exception) {
+ // Display a snackbar with the error message.
+ Snackbar.make(aboutVersionLayout, getString(R.string.error_saving_file) + " " + exception.toString(), Snackbar.LENGTH_INDEFINITE).show()
+ }
+ }
+ }
+
+ // Define the save about version image activity result launcher. It must be defined before `onCreate()` is run or the app will crash.
+ private val saveAboutVersionImageActivityResultLauncher = registerForActivityResult(ActivityResultContracts.CreateDocument()) { fileNameUri: Uri? ->
+ // Only save the file if the URI is not null, which happens if the user exited the file picker by pressing back.
+ if (fileNameUri != null) {
+ // Save the about version image.
+ SaveAboutVersionImage(requireActivity(), fileNameUri, aboutVersionLayout.findViewById(R.id.about_version_linearlayout)).execute()
+ }
+ }
+
+ override fun onCreate(savedInstanceState: Bundle?) {
+ // Run the default commands.
+ super.onCreate(savedInstanceState)
+
+ // Store the arguments in class variables.
+ blocklistVersions = requireArguments().getStringArray(BLOCKLIST_VERSIONS)!!
+
+ // Enable the options menu for this fragment.
+ setHasOptionsMenu(true)
+ }
+
+ override fun onCreateView(layoutInflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View {
+ // Inflate the layout. Setting false at the end of inflater.inflate does not attach the inflated layout as a child of container. The fragment will take care of attaching the root automatically.
+ aboutVersionLayout = layoutInflater.inflate(R.layout.about_version_scrollview, container, false)
+
+ // Get handles for the views.
+ privacyBrowserTextView = aboutVersionLayout.findViewById(R.id.privacy_browser_textview)
+ versionTextView = aboutVersionLayout.findViewById(R.id.version)
+ hardwareTextView = aboutVersionLayout.findViewById(R.id.hardware)
+ brandTextView = aboutVersionLayout.findViewById(R.id.brand)
+ manufacturerTextView = aboutVersionLayout.findViewById(R.id.manufacturer)
+ modelTextView = aboutVersionLayout.findViewById(R.id.model)
+ deviceTextView = aboutVersionLayout.findViewById(R.id.device)
+ bootloaderTextView = aboutVersionLayout.findViewById(R.id.bootloader)
+ radioTextView = aboutVersionLayout.findViewById(R.id.radio)
+ softwareTextView = aboutVersionLayout.findViewById(R.id.software)
+ androidTextView = aboutVersionLayout.findViewById(R.id.android)
+ securityPatchTextView = aboutVersionLayout.findViewById(R.id.security_patch)
+ buildTextView = aboutVersionLayout.findViewById(R.id.build)
+ webViewProviderTextView = aboutVersionLayout.findViewById(R.id.webview_provider)
+ webViewVersionTextView = aboutVersionLayout.findViewById(R.id.webview_version)
+ orbotTextView = aboutVersionLayout.findViewById(R.id.orbot)
+ i2pTextView = aboutVersionLayout.findViewById(R.id.i2p)
+ openKeychainTextView = aboutVersionLayout.findViewById(R.id.open_keychain)
+ memoryUsageTextView = aboutVersionLayout.findViewById(R.id.memory_usage)
+ appConsumedMemoryTextView = aboutVersionLayout.findViewById(R.id.app_consumed_memory)
+ appAvailableMemoryTextView = aboutVersionLayout.findViewById(R.id.app_available_memory)
+ appTotalMemoryTextView = aboutVersionLayout.findViewById(R.id.app_total_memory)
+ appMaximumMemoryTextView = aboutVersionLayout.findViewById(R.id.app_maximum_memory)
+ systemConsumedMemoryTextView = aboutVersionLayout.findViewById(R.id.system_consumed_memory)
+ systemAvailableMemoryTextView = aboutVersionLayout.findViewById(R.id.system_available_memory)
+ systemTotalMemoryTextView = aboutVersionLayout.findViewById(R.id.system_total_memory)
+ blocklistsTextView = aboutVersionLayout.findViewById(R.id.blocklists)
+ easyListTextView = aboutVersionLayout.findViewById(R.id.easylist)
+ easyPrivacyTextView = aboutVersionLayout.findViewById(R.id.easyprivacy)
+ fanboyAnnoyanceTextView = aboutVersionLayout.findViewById(R.id.fanboy_annoyance)
+ fanboySocialTextView = aboutVersionLayout.findViewById(R.id.fanboy_social)
+ ultraListTextView = aboutVersionLayout.findViewById(R.id.ultralist)
+ ultraPrivacyTextView = aboutVersionLayout.findViewById(R.id.ultraprivacy)
+ packageSignatureTextView = aboutVersionLayout.findViewById(R.id.package_signature)
+ certificateIssuerDnTextView = aboutVersionLayout.findViewById(R.id.certificate_issuer_dn)
+ certificateSubjectDnTextView = aboutVersionLayout.findViewById(R.id.certificate_subject_dn)
+ certificateStartDateTextView = aboutVersionLayout.findViewById(R.id.certificate_start_date)
+ certificateEndDateTextView = aboutVersionLayout.findViewById(R.id.certificate_end_date)
+ certificateVersionTextView = aboutVersionLayout.findViewById(R.id.certificate_version)
+ certificateSerialNumberTextView = aboutVersionLayout.findViewById(R.id.certificate_serial_number)
+ certificateSignatureAlgorithmTextView = aboutVersionLayout.findViewById(R.id.certificate_signature_algorithm)
+
+ // Setup the labels.
+ val version = getString(R.string.version) + " " + BuildConfig.VERSION_NAME + " (" + getString(R.string.version_code) + " " + BuildConfig.VERSION_CODE + ")"
+ val brandLabel = getString(R.string.brand) + " "
+ val manufacturerLabel = getString(R.string.manufacturer) + " "
+ val modelLabel = getString(R.string.model) + " "
+ val deviceLabel = getString(R.string.device) + " "
+ val bootloaderLabel = getString(R.string.bootloader) + " "
+ val androidLabel = getString(R.string.android) + " "
+ val buildLabel = getString(R.string.build) + " "
+ val webViewVersionLabel = getString(R.string.webview_version) + " "
+ appConsumedMemoryLabel = getString(R.string.app_consumed_memory) + " "
+ appAvailableMemoryLabel = getString(R.string.app_available_memory) + " "
+ appTotalMemoryLabel = getString(R.string.app_total_memory) + " "
+ appMaximumMemoryLabel = getString(R.string.app_maximum_memory) + " "
+ systemConsumedMemoryLabel = getString(R.string.system_consumed_memory) + " "
+ systemAvailableMemoryLabel = getString(R.string.system_available_memory) + " "
+ systemTotalMemoryLabel = getString(R.string.system_total_memory) + " "
+ val easyListLabel = getString(R.string.easylist_label) + " "
+ val easyPrivacyLabel = getString(R.string.easyprivacy_label) + " "
+ val fanboyAnnoyanceLabel = getString(R.string.fanboy_annoyance_label) + " "
+ val fanboySocialLabel = getString(R.string.fanboy_social_label) + " "
+ val ultraListLabel = getString(R.string.ultralist_label) + " "
+ val ultraPrivacyLabel = getString(R.string.ultraprivacy_label) + " "
+ val issuerDNLabel = getString(R.string.issuer_dn) + " "
+ val subjectDNLabel = getString(R.string.subject_dn) + " "
+ val startDateLabel = getString(R.string.start_date) + " "
+ val endDateLabel = getString(R.string.end_date) + " "
+ val certificateVersionLabel = getString(R.string.certificate_version) + " "
+ val serialNumberLabel = getString(R.string.serial_number) + " "
+ val signatureAlgorithmLabel = getString(R.string.signature_algorithm) + " "
+
+ // The WebView layout is only used to get the default user agent from `bare_webview`. It is not used to render content on the screen.
+ // Once the minimum API >= 26 this can be accomplished with the WebView package info.
+ val webViewLayout = layoutInflater.inflate(R.layout.bare_webview, container, false)
+ val tabLayoutWebView = webViewLayout.findViewById<WebView>(R.id.bare_webview)
+ val userAgentString = tabLayoutWebView.settings.userAgentString
+
+ // Get the device's information and store it in strings.
+ val brand = Build.BRAND
+ val manufacturer = Build.MANUFACTURER
+ val model = Build.MODEL
+ val device = Build.DEVICE
+ val bootloader = Build.BOOTLOADER
+ val radio = Build.getRadioVersion()
+ val android = Build.VERSION.RELEASE + " (" + getString(R.string.api) + " " + Build.VERSION.SDK_INT + ")"
+ val build = Build.DISPLAY
+ // Select the substring that begins after `Chrome/` and goes until the next ` `.
+ val webView = userAgentString.substring(userAgentString.indexOf("Chrome/") + 7, userAgentString.indexOf(" ", userAgentString.indexOf("Chrome/")))
+
+ // Get the Orbot version name if Orbot is installed.
+ val orbot: String = try {
+ // Store the version name.
+ requireContext().packageManager.getPackageInfo("org.torproject.android", 0).versionName
+ } catch (exception: PackageManager.NameNotFoundException) { // Orbot is not installed.
+ // Store an empty string.
+ ""
+ }
+
+ // Get the I2P version name if I2P is installed.
+ val i2p: String = try {
+ // Store the version name.
+ requireContext().packageManager.getPackageInfo("net.i2p.android.router", 0).versionName
+ } catch (exception: PackageManager.NameNotFoundException) { // I2P is not installed.
+ // Store an empty string.
+ ""
+ }
+
+ // Get the OpenKeychain version name if it is installed.
+ val openKeychain: String = try {
+ // Store the version name.
+ requireContext().packageManager.getPackageInfo("org.sufficientlysecure.keychain", 0).versionName
+ } catch (exception: PackageManager.NameNotFoundException) { // OpenKeychain is not installed.
+ // Store an empty string.
+ ""
+ }
+
+ // Create a spannable string builder for the hardware and software text views that need multiple colors of text.
+ val brandStringBuilder = SpannableStringBuilder(brandLabel + brand)
+ val manufacturerStringBuilder = SpannableStringBuilder(manufacturerLabel + manufacturer)
+ val modelStringBuilder = SpannableStringBuilder(modelLabel + model)
+ val deviceStringBuilder = SpannableStringBuilder(deviceLabel + device)
+ val bootloaderStringBuilder = SpannableStringBuilder(bootloaderLabel + bootloader)
+ val androidStringBuilder = SpannableStringBuilder(androidLabel + android)
+ val buildStringBuilder = SpannableStringBuilder(buildLabel + build)
+ val webViewVersionStringBuilder = SpannableStringBuilder(webViewVersionLabel + webView)
+ val easyListStringBuilder = SpannableStringBuilder(easyListLabel + blocklistVersions[0])
+ val easyPrivacyStringBuilder = SpannableStringBuilder(easyPrivacyLabel + blocklistVersions[1])
+ val fanboyAnnoyanceStringBuilder = SpannableStringBuilder(fanboyAnnoyanceLabel + blocklistVersions[2])
+ val fanboySocialStringBuilder = SpannableStringBuilder(fanboySocialLabel + blocklistVersions[3])
+ val ultraListStringBuilder = SpannableStringBuilder(ultraListLabel + blocklistVersions[4])
+ val ultraPrivacyStringBuilder = SpannableStringBuilder(ultraPrivacyLabel + blocklistVersions[5])
+
+ // Set the blue color span according to the theme. The deprecated `getColor()` must be used until the minimum API >= 23.
+ blueColorSpan = ForegroundColorSpan(resources.getColor(R.color.about_version_blue_text))
+
+ // Set the spans to display the device information in blue. `SPAN_INCLUSIVE_INCLUSIVE` allows the span to grow in either direction.
+ brandStringBuilder.setSpan(blueColorSpan, brandLabel.length, brandStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ manufacturerStringBuilder.setSpan(blueColorSpan, manufacturerLabel.length, manufacturerStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ modelStringBuilder.setSpan(blueColorSpan, modelLabel.length, modelStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ deviceStringBuilder.setSpan(blueColorSpan, deviceLabel.length, deviceStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ bootloaderStringBuilder.setSpan(blueColorSpan, bootloaderLabel.length, bootloaderStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ androidStringBuilder.setSpan(blueColorSpan, androidLabel.length, androidStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ buildStringBuilder.setSpan(blueColorSpan, buildLabel.length, buildStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ webViewVersionStringBuilder.setSpan(blueColorSpan, webViewVersionLabel.length, webViewVersionStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ easyListStringBuilder.setSpan(blueColorSpan, easyListLabel.length, easyListStringBuilder.length, Spanned.SPAN_INCLUSIVE_EXCLUSIVE)
+ easyPrivacyStringBuilder.setSpan(blueColorSpan, easyPrivacyLabel.length, easyPrivacyStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ fanboyAnnoyanceStringBuilder.setSpan(blueColorSpan, fanboyAnnoyanceLabel.length, fanboyAnnoyanceStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ fanboySocialStringBuilder.setSpan(blueColorSpan, fanboySocialLabel.length, fanboySocialStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ ultraListStringBuilder.setSpan(blueColorSpan, ultraListLabel.length, ultraListStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ ultraPrivacyStringBuilder.setSpan(blueColorSpan, ultraPrivacyLabel.length, ultraPrivacyStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the strings in the text boxes.
+ versionTextView.text = version
+ brandTextView.text = brandStringBuilder
+ manufacturerTextView.text = manufacturerStringBuilder
+ modelTextView.text = modelStringBuilder
+ deviceTextView.text = deviceStringBuilder
+ bootloaderTextView.text = bootloaderStringBuilder
+ androidTextView.text = androidStringBuilder
+ buildTextView.text = buildStringBuilder
+ webViewVersionTextView.text = webViewVersionStringBuilder
+ easyListTextView.text = easyListStringBuilder
+ easyPrivacyTextView.text = easyPrivacyStringBuilder
+ fanboyAnnoyanceTextView.text = fanboyAnnoyanceStringBuilder
+ fanboySocialTextView.text = fanboySocialStringBuilder
+ ultraListTextView.text = ultraListStringBuilder
+ ultraPrivacyTextView.text = ultraPrivacyStringBuilder
+
+ // Only populate the radio text view if there is a radio in the device.
+ // Null must be checked because some Samsung tablets report a null value for the radio instead of an empty string. Grrrr. <https://redmine.stoutner.com/issues/701>
+ if (radio != null && radio.isNotEmpty()) {
+ // Setup the label.
+ val radioLabel = getString(R.string.radio) + " "
+
+ // Create a spannable string builder.
+ val radioStringBuilder = SpannableStringBuilder(radioLabel + radio)
+
+ // Set the span to display the radio in blue.
+ radioStringBuilder.setSpan(blueColorSpan, radioLabel.length, radioStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the string in the text view.
+ radioTextView.text = radioStringBuilder
+ } else { // This device does not have a radio.
+ // Hide the radio text view.
+ radioTextView.visibility = View.GONE
+ }
+
+ // Build.VERSION.SECURITY_PATCH is only available for SDK_INT >= 23.
+ if (Build.VERSION.SDK_INT >= 23) {
+ // Setup the label.
+ val securityPatchLabel = getString(R.string.security_patch) + " "
+
+ // Get the security patch version.
+ val securityPatch = Build.VERSION.SECURITY_PATCH
+
+ // Create a spannable string builder.
+ val securityPatchStringBuilder = SpannableStringBuilder(securityPatchLabel + securityPatch)
+
+ // Set the span to display the security patch version in blue.
+ securityPatchStringBuilder.setSpan(blueColorSpan, securityPatchLabel.length, securityPatchStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the string in the text view.
+ securityPatchTextView.text = securityPatchStringBuilder
+ } else { // The API < 23.
+ // Hide the security patch text view.
+ securityPatchTextView.visibility = View.GONE
+ }
+
+ // Only populate the WebView provider if the SDK >= 21.
+ if (Build.VERSION.SDK_INT >= 21) {
+ // Create the WebView provider label.
+ val webViewProviderLabel = getString(R.string.webview_provider) + " "
+
+ // Get the current WebView package info.
+ val webViewPackageInfo = WebViewCompat.getCurrentWebViewPackage(requireContext())!!
+
+ // Get the WebView provider name.
+ val webViewPackageName = webViewPackageInfo.packageName
+
+ // Create the spannable string builder.
+ val webViewProviderStringBuilder = SpannableStringBuilder(webViewProviderLabel + webViewPackageName)
+
+ // Apply the coloration.
+ webViewProviderStringBuilder.setSpan(blueColorSpan, webViewProviderLabel.length, webViewProviderStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the WebView provider.
+ webViewProviderTextView.text = webViewProviderStringBuilder
+ } else { // The API < 21.
+ // Hide the WebView provider text view.
+ webViewProviderTextView.visibility = View.GONE
+ }
+
+ // Only populate the Orbot text view if it is installed.
+ if (orbot.isNotEmpty()) {
+ // Setup the label.
+ val orbotLabel = getString(R.string.orbot) + " "
+
+ // Create a spannable string builder.
+ val orbotStringBuilder = SpannableStringBuilder(orbotLabel + orbot)
+
+ // Set the span to display the Orbot version.
+ orbotStringBuilder.setSpan(blueColorSpan, orbotLabel.length, orbotStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the string in the text view.
+ orbotTextView.text = orbotStringBuilder
+ } else { // Orbot is not installed.
+ // Hide the Orbot text view.
+ orbotTextView.visibility = View.GONE
+ }
+
+ // Only populate the I2P text view if it is installed.
+ if (i2p.isNotEmpty()) {
+ // Setup the label.
+ val i2pLabel = getString(R.string.i2p) + " "
+
+ // Create a spannable string builder.
+ val i2pStringBuilder = SpannableStringBuilder(i2pLabel + i2p)
+
+ // Set the span to display the I2P version.
+ i2pStringBuilder.setSpan(blueColorSpan, i2pLabel.length, i2pStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the string in the text view.
+ i2pTextView.text = i2pStringBuilder
+ } else { // I2P is not installed.
+ // Hide the I2P text view.
+ i2pTextView.visibility = View.GONE
+ }
+
+ // Only populate the OpenKeychain text view if it is installed.
+ if (openKeychain.isNotEmpty()) {
+ // Setup the label.
+ val openKeychainLabel = getString(R.string.openkeychain) + " "
+
+ // Create a spannable string builder.
+ val openKeychainStringBuilder = SpannableStringBuilder(openKeychainLabel + openKeychain)
+
+ // Set the span to display the OpenKeychain version.
+ openKeychainStringBuilder.setSpan(blueColorSpan, openKeychainLabel.length, openKeychainStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the string in the text view.
+ openKeychainTextView.text = openKeychainStringBuilder
+ } else { //OpenKeychain is not installed.
+ // Hide the OpenKeychain text view.
+ openKeychainTextView.visibility = View.GONE
+ }
+
+ // Display the package signature.
+ try {
+ // Get the first package signature. Suppress the lint warning about the need to be careful in implementing comparison of certificates for security purposes.
+ // Once the minimum API >= 28, `GET_SIGNING_CERTIFICATES` can be used instead.
+ @SuppressLint("PackageManagerGetSignatures") val packageSignature = requireContext().packageManager.getPackageInfo(requireContext().packageName,PackageManager.GET_SIGNATURES)
+ .signatures[0]
+
+ // Convert the signature to a byte array input stream.
+ val certificateByteArrayInputStream: InputStream = ByteArrayInputStream(packageSignature.toByteArray())
+
+ // Display the certificate information on the screen.
+ try {
+ // Instantiate a certificate factory.
+ val certificateFactory = CertificateFactory.getInstance("X509")
+
+ // Generate an X509 certificate.
+ val x509Certificate = certificateFactory.generateCertificate(certificateByteArrayInputStream) as X509Certificate
+
+ // Store the individual sections of the certificate.
+ val issuerDNPrincipal = x509Certificate.issuerDN
+ val subjectDNPrincipal = x509Certificate.subjectDN
+ val startDate = x509Certificate.notBefore
+ val endDate = x509Certificate.notAfter
+ val certificateVersion = x509Certificate.version
+ val serialNumberBigInteger = x509Certificate.serialNumber
+ val signatureAlgorithmNameString = x509Certificate.sigAlgName
+
+ // Create a spannable string builder for each text view that needs multiple colors of text.
+ val issuerDNStringBuilder = SpannableStringBuilder(issuerDNLabel + issuerDNPrincipal.toString())
+ val subjectDNStringBuilder = SpannableStringBuilder(subjectDNLabel + subjectDNPrincipal.toString())
+ val startDateStringBuilder = SpannableStringBuilder(startDateLabel + DateFormat.getDateTimeInstance(DateFormat.MEDIUM, DateFormat.LONG).format(startDate))
+ val endDataStringBuilder = SpannableStringBuilder(endDateLabel + DateFormat.getDateTimeInstance(DateFormat.MEDIUM, DateFormat.LONG).format(endDate))
+ val certificateVersionStringBuilder = SpannableStringBuilder(certificateVersionLabel + certificateVersion)
+ val serialNumberStringBuilder = SpannableStringBuilder(serialNumberLabel + serialNumberBigInteger)
+ val signatureAlgorithmStringBuilder = SpannableStringBuilder(signatureAlgorithmLabel + signatureAlgorithmNameString)
+
+ // Setup the spans to display the device information in blue. `SPAN_INCLUSIVE_INCLUSIVE` allows the span to grow in either direction.
+ issuerDNStringBuilder.setSpan(blueColorSpan, issuerDNLabel.length, issuerDNStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ subjectDNStringBuilder.setSpan(blueColorSpan, subjectDNLabel.length, subjectDNStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ startDateStringBuilder.setSpan(blueColorSpan, startDateLabel.length, startDateStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ endDataStringBuilder.setSpan(blueColorSpan, endDateLabel.length, endDataStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ certificateVersionStringBuilder.setSpan(blueColorSpan, certificateVersionLabel.length, certificateVersionStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ serialNumberStringBuilder.setSpan(blueColorSpan, serialNumberLabel.length, serialNumberStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ signatureAlgorithmStringBuilder.setSpan(blueColorSpan, signatureAlgorithmLabel.length, signatureAlgorithmStringBuilder.length, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the strings in the text boxes.
+ certificateIssuerDnTextView.text = issuerDNStringBuilder
+ certificateSubjectDnTextView.text = subjectDNStringBuilder
+ certificateStartDateTextView.text = startDateStringBuilder
+ certificateEndDateTextView.text = endDataStringBuilder
+ certificateVersionTextView.text = certificateVersionStringBuilder
+ certificateSerialNumberTextView.text = serialNumberStringBuilder
+ certificateSignatureAlgorithmTextView.text = signatureAlgorithmStringBuilder
+ } catch (certificateException: CertificateException) {
+ // Do nothing if there is a certificate error.
+ }
+
+ // Get a handle for the runtime.
+ runtime = Runtime.getRuntime()
+
+ // Get a handle for the activity manager.
+ activityManager = requireActivity().getSystemService(Context.ACTIVITY_SERVICE) as ActivityManager
+
+ // Instantiate a memory info variable.
+ memoryInfo = ActivityManager.MemoryInfo()
+
+ // Define a number format.
+ numberFormat = NumberFormat.getInstance()
+
+ // Set the minimum and maximum number of fraction digits.
+ numberFormat.minimumFractionDigits = 2
+ numberFormat.maximumFractionDigits = 2
+
+ // Update the memory usage.
+ updateMemoryUsage(requireActivity())
+ } catch (e: PackageManager.NameNotFoundException) {
+ // Do nothing if the package manager says Privacy Browser isn't installed.
+ }
+
+ // Scroll the tab if the saved instance state is not null.
+ if (savedInstanceState != null) {
+ aboutVersionLayout.post {
+ aboutVersionLayout.scrollX = savedInstanceState.getInt("scroll_x")
+ aboutVersionLayout.scrollY = savedInstanceState.getInt("scroll_y")
+ }
+ }
+
+ // Return the tab layout.
+ return aboutVersionLayout
+ }
+
+ override fun onCreateOptionsMenu(menu: Menu, menuInflater: MenuInflater) {
+ // Inflate the about version menu.
+ menuInflater.inflate(R.menu.about_version_options_menu, menu)
+
+ // Run the default commands.
+ super.onCreateOptionsMenu(menu, menuInflater)
+ }
+
+ override fun onOptionsItemSelected(menuItem: MenuItem): Boolean {
+ // Run the appropriate commands.
+ when (menuItem.itemId) {
+ R.id.copy -> { // Copy.
+ // Get the about version string.
+ val aboutVersionString = getAboutVersionString()
+
+ // Get a handle for the clipboard manager.
+ val clipboardManager = (requireActivity().getSystemService(Context.CLIPBOARD_SERVICE) as ClipboardManager)
+
+ // Save the about version string in a clip data.
+ val aboutVersionClipData = ClipData.newPlainText(getString(R.string.about), aboutVersionString)
+
+ // Place the clip data on the clipboard.
+ clipboardManager.setPrimaryClip(aboutVersionClipData)
+
+ // Display a snackbar.
+ Snackbar.make(aboutVersionLayout, R.string.version_info_copied, Snackbar.LENGTH_SHORT).show()
+
+ // Consume the event.
+ return true
+ }
+
+ R.id.share -> { // Share.
+ // Get the about version string.
+ val aboutString = getAboutVersionString()
+
+ // Create an email intent.
+ val emailIntent = Intent(Intent.ACTION_SEND)
+
+ // Add the about version string to the intent.
+ emailIntent.putExtra(Intent.EXTRA_TEXT, aboutString)
+
+ // Set the MIME type.
+ emailIntent.type = "text/plain"
+
+ // Set the intent to open in a new task.
+ emailIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
+
+ // Make it so.
+ startActivity(Intent.createChooser(emailIntent, getString(R.string.share)))
+
+ // Consume the event.
+ return true
+ }
+
+ R.id.save_text -> { // Save text.
+ // Open the file picker.
+ saveAboutVersionTextActivityResultLauncher.launch(getString(R.string.privacy_browser_version_txt, BuildConfig.VERSION_NAME))
+
+ // Consume the event.
+ return true
+ }
+
+ R.id.save_image -> { // Save image.
+ // Open the file picker.
+ saveAboutVersionImageActivityResultLauncher.launch(getString(R.string.privacy_browser_version_png, BuildConfig.VERSION_NAME))
+
+ // Consume the event.
+ return true
+ }
+ else -> { // The home button was selected.
+ // Run the parents class on return.
+ return super.onOptionsItemSelected(menuItem)
+ }
+ }
+ }
+
+ override fun onSaveInstanceState(savedInstanceState: Bundle) {
+ // Run the default commands.
+ super.onSaveInstanceState(savedInstanceState)
+
+ // Save the scroll positions.
+ savedInstanceState.putInt("scroll_x", aboutVersionLayout.scrollX)
+ savedInstanceState.putInt("scroll_y", aboutVersionLayout.scrollY)
+ }
+
+ override fun onPause() {
+ // Run the default commands.
+ super.onPause()
+
+ // Pause the updating of the memory usage.
+ updateMemoryUsageBoolean = false
+ }
+
+ override fun onResume() {
+ // Run the default commands.
+ super.onResume()
+
+ // Resume the updating of the memory usage.
+ updateMemoryUsageBoolean = true
+ }
+
+ fun updateMemoryUsage(activity: Activity) {
+ try {
+ // Update the memory usage if enabled.
+ if (updateMemoryUsageBoolean) {
+ // Populate the memory info variable.
+ activityManager.getMemoryInfo(memoryInfo)
+
+ // Get the app memory information.
+ val appAvailableMemoryLong = runtime.freeMemory()
+ val appTotalMemoryLong = runtime.totalMemory()
+ val appMaximumMemoryLong = runtime.maxMemory()
+
+ // Calculate the app consumed memory.
+ val appConsumedMemoryLong = appTotalMemoryLong - appAvailableMemoryLong
+
+ // Get the system memory information.
+ val systemTotalMemoryLong = memoryInfo.totalMem
+ val systemAvailableMemoryLong = memoryInfo.availMem
+
+ // Calculate the system consumed memory.
+ val systemConsumedMemoryLong = systemTotalMemoryLong - systemAvailableMemoryLong
+
+ // Convert the memory information into mebibytes.
+ val appConsumedMemoryFloat = appConsumedMemoryLong.toFloat() / MEBIBYTE
+ val appAvailableMemoryFloat = appAvailableMemoryLong.toFloat() / MEBIBYTE
+ val appTotalMemoryFloat = appTotalMemoryLong.toFloat() / MEBIBYTE
+ val appMaximumMemoryFloat = appMaximumMemoryLong.toFloat() / MEBIBYTE
+ val systemConsumedMemoryFloat = systemConsumedMemoryLong.toFloat() / MEBIBYTE
+ val systemAvailableMemoryFloat = systemAvailableMemoryLong.toFloat() / MEBIBYTE
+ val systemTotalMemoryFloat = systemTotalMemoryLong.toFloat() / MEBIBYTE
+
+ // Get the mebibyte string.
+ val mebibyte = getString(R.string.mebibyte)
+
+ // Calculate the mebibyte length.
+ val mebibyteLength = mebibyte.length
+
+ // Create spannable string builders.
+ val appConsumedMemoryStringBuilder = SpannableStringBuilder(appConsumedMemoryLabel + numberFormat.format(appConsumedMemoryFloat.toDouble()) + " " + mebibyte)
+ val appAvailableMemoryStringBuilder = SpannableStringBuilder(appAvailableMemoryLabel + numberFormat.format(appAvailableMemoryFloat.toDouble()) + " " + mebibyte)
+ val appTotalMemoryStringBuilder = SpannableStringBuilder(appTotalMemoryLabel + numberFormat.format(appTotalMemoryFloat.toDouble()) + " " + mebibyte)
+ val appMaximumMemoryStringBuilder = SpannableStringBuilder(appMaximumMemoryLabel + numberFormat.format(appMaximumMemoryFloat.toDouble()) + " " + mebibyte)
+ val systemConsumedMemoryStringBuilder = SpannableStringBuilder(systemConsumedMemoryLabel + numberFormat.format(systemConsumedMemoryFloat.toDouble()) + " " + mebibyte)
+ val systemAvailableMemoryStringBuilder = SpannableStringBuilder(systemAvailableMemoryLabel + numberFormat.format(systemAvailableMemoryFloat.toDouble()) + " " + mebibyte)
+ val systemTotalMemoryStringBuilder = SpannableStringBuilder(systemTotalMemoryLabel + numberFormat.format(systemTotalMemoryFloat.toDouble()) + " " + mebibyte)
+
+ // Setup the spans to display the memory information in blue. `SPAN_INCLUSIVE_INCLUSIVE` allows the span to grow in either direction.
+ appConsumedMemoryStringBuilder.setSpan(blueColorSpan, appConsumedMemoryLabel.length, appConsumedMemoryStringBuilder.length - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ appAvailableMemoryStringBuilder.setSpan(blueColorSpan, appAvailableMemoryLabel.length, appAvailableMemoryStringBuilder.length - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ appTotalMemoryStringBuilder.setSpan(blueColorSpan, appTotalMemoryLabel.length, appTotalMemoryStringBuilder.length - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ appMaximumMemoryStringBuilder.setSpan(blueColorSpan, appMaximumMemoryLabel.length, appMaximumMemoryStringBuilder.length - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ systemConsumedMemoryStringBuilder.setSpan(blueColorSpan, systemConsumedMemoryLabel.length, systemConsumedMemoryStringBuilder.length - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ systemAvailableMemoryStringBuilder.setSpan(blueColorSpan, systemAvailableMemoryLabel.length, systemAvailableMemoryStringBuilder.length - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+ systemTotalMemoryStringBuilder.setSpan(blueColorSpan, systemTotalMemoryLabel.length, systemTotalMemoryStringBuilder.length - mebibyteLength, Spanned.SPAN_INCLUSIVE_INCLUSIVE)
+
+ // Display the string in the text boxes.
+ appConsumedMemoryTextView.text = appConsumedMemoryStringBuilder
+ appAvailableMemoryTextView.text = appAvailableMemoryStringBuilder
+ appTotalMemoryTextView.text = appTotalMemoryStringBuilder
+ appMaximumMemoryTextView.text = appMaximumMemoryStringBuilder
+ systemConsumedMemoryTextView.text = systemConsumedMemoryStringBuilder
+ systemAvailableMemoryTextView.text = systemAvailableMemoryStringBuilder
+ systemTotalMemoryTextView.text = systemTotalMemoryStringBuilder
+ }
+
+ // Schedule another memory update if the activity has not been destroyed.
+ if (!activity.isDestroyed) {
+ // Create a handler to update the memory usage.
+ val updateMemoryUsageHandler = Handler(Looper.getMainLooper())
+
+ // Create a runnable to update the memory usage.
+ val updateMemoryUsageRunnable = Runnable { updateMemoryUsage(activity) }
+
+ // Update the memory usage after 1000 milliseconds
+ updateMemoryUsageHandler.postDelayed(updateMemoryUsageRunnable, 1000)
+ }
+ } catch (exception: Exception) {
+ // Do nothing.
+ }
+ }
+
+ fun getAboutVersionString(): String {
+ // Initialize an about version string builder.
+ val aboutVersionStringBuilder = StringBuilder()
+
+ // Populate the about version string builder.
+ aboutVersionStringBuilder.append(privacyBrowserTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(versionTextView.text)
+ aboutVersionStringBuilder.append("\n\n")
+ aboutVersionStringBuilder.append(hardwareTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(brandTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(manufacturerTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(modelTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(deviceTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(bootloaderTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ if (radioTextView.visibility == View.VISIBLE) {
+ aboutVersionStringBuilder.append(radioTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ }
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(softwareTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(androidTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ if (securityPatchTextView.visibility == View.VISIBLE) {
+ aboutVersionStringBuilder.append(securityPatchTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ }
+ aboutVersionStringBuilder.append(buildTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ if (webViewProviderTextView.visibility == View.VISIBLE) {
+ aboutVersionStringBuilder.append(webViewProviderTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ }
+ aboutVersionStringBuilder.append(webViewVersionTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ if (orbotTextView.visibility == View.VISIBLE) {
+ aboutVersionStringBuilder.append(orbotTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ }
+ if (i2pTextView.visibility == View.VISIBLE) {
+ aboutVersionStringBuilder.append(i2pTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ }
+ if (openKeychainTextView.visibility == View.VISIBLE) {
+ aboutVersionStringBuilder.append(openKeychainTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ }
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(memoryUsageTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(appConsumedMemoryTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(appAvailableMemoryTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(appTotalMemoryTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(appMaximumMemoryTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(systemConsumedMemoryTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(systemAvailableMemoryTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(systemTotalMemoryTextView.text)
+ aboutVersionStringBuilder.append("\n\n")
+ aboutVersionStringBuilder.append(blocklistsTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(easyListTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(easyPrivacyTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(fanboyAnnoyanceTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(fanboySocialTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(ultraListTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(ultraPrivacyTextView.text)
+ aboutVersionStringBuilder.append("\n\n")
+ aboutVersionStringBuilder.append(packageSignatureTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(certificateIssuerDnTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(certificateSubjectDnTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(certificateStartDateTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(certificateEndDateTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(certificateVersionTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(certificateSerialNumberTextView.text)
+ aboutVersionStringBuilder.append("\n")
+ aboutVersionStringBuilder.append(certificateSignatureAlgorithmTextView.text)
+
+ // Return the string.
+ return aboutVersionStringBuilder.toString()
+ }
+}
\ No newline at end of file
+++ /dev/null
-<!-- This file comes from the Android Material icon set, where it is called `save`. It is released under the Apache License 2.0. -->
-
-<!-- `tools:ignore="VectorRaster"` removes the lint warning about `android:autoMirrored="true"` not applying to API < 21. -->
-<vector
- xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:height="24dp"
- android:width="24dp"
- android:viewportHeight="24"
- android:viewportWidth="24"
- android:autoMirrored="true"
- tools:ignore="VectorRaster" >
-
- <!-- A hard coded color must be used until API >= 21. Then `@color` or `?attr/colorControlNormal` may be used. -->
- <path
- android:fillColor="#FF1565C0"
- android:pathData="M17,3L5,3c-1.11,0 -2,0.9 -2,2v14c0,1.1 0.89,2 2,2h14c1.1,0 2,-0.9 2,-2L21,7l-4,-4zM12,19c-1.66,0 -3,-1.34 -3,-3s1.34,-3 3,-3 3,1.34 3,3 -1.34,3 -3,3zM15,9L5,9L5,5h10v4z" />
-</vector>
\ No newline at end of file
+++ /dev/null
-<!-- This file comes from the Android Material icon set, where it is called `save`. It is released under the Apache License 2.0. -->
-
-<!-- `tools:ignore="VectorRaster"` removes the lint warning about `android:autoMirrored="true"` not applying to API < 21. -->
-<vector
- xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:height="24dp"
- android:width="24dp"
- android:viewportHeight="24"
- android:viewportWidth="24"
- android:autoMirrored="true"
- tools:ignore="VectorRaster" >
-
- <!-- A hard coded color must be used until API >= 21. Then `@color` or `?attr/colorControlNormal` may be used. -->
- <path
- android:fillColor="#FF8AB4F8"
- android:pathData="M17,3L5,3c-1.11,0 -2,0.9 -2,2v14c0,1.1 0.89,2 2,2h14c1.1,0 2,-0.9 2,-2L21,7l-4,-4zM12,19c-1.66,0 -3,-1.34 -3,-3s1.34,-3 3,-3 3,1.34 3,3 -1.34,3 -3,3zM15,9L5,9L5,5h10v4z" />
-</vector>
\ No newline at end of file
+++ /dev/null
-<!-- This file comes from the Android Material icon set, where it is called `chrome_reader_mode`. It is released under the Apache License 2.0. -->
-
-<!-- `tools:ignore="VectorRaster"` removes the lint warning about `android:autoMirrored="true"` not applying to API < 21. -->
-<vector
- xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:height="24dp"
- android:width="24dp"
- android:viewportHeight="24"
- android:viewportWidth="24"
- android:autoMirrored="true"
- tools:ignore="VectorRaster" >
-
- <!-- A hard coded color must be used until API >= 21. Then `@color` or `?attr/colorControlNormal` may be used instead. -->
- <path
- android:fillColor="#FF1565C0"
- android:pathData="M13,12h7v1.5h-7zM13,9.5h7L20,11h-7zM13,14.5h7L20,16h-7zM21,4L3,4c-1.1,0 -2,0.9 -2,2v13c0,1.1 0.9,2 2,2h18c1.1,0 2,-0.9 2,-2L23,6c0,-1.1 -0.9,-2 -2,-2zM21,19h-9L12,6h9v13z"/>
-</vector>
+++ /dev/null
-<!-- This file comes from the Android Material icon set, where it is called `chrome_reader_mode`. It is released under the Apache License 2.0. -->
-
-<!-- `tools:ignore="VectorRaster"` removes the lint warning about `android:autoMirrored="true"` not applying to API < 21. -->
-<vector
- xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:height="24dp"
- android:width="24dp"
- android:viewportHeight="24"
- android:viewportWidth="24"
- android:autoMirrored="true"
- tools:ignore="VectorRaster" >
-
- <!-- A hard coded color must be used until API >= 21. Then `@color` or `?attr/colorControlNormal` may be used instead. -->
- <path
- android:fillColor="#FF8AB4F8"
- android:pathData="M13,12h7v1.5h-7zM13,9.5h7L20,11h-7zM13,14.5h7L20,16h-7zM21,4L3,4c-1.1,0 -2,0.9 -2,2v13c0,1.1 0.9,2 2,2h18c1.1,0 2,-0.9 2,-2L23,6c0,-1.1 -0.9,-2 -2,-2zM21,19h-9L12,6h9v13z"/>
-</vector>
+++ /dev/null
-<?xml version="1.0" encoding="utf-8"?>
-
-<!--
- Copyright © 2016-2018,2020 Soren Stoutner <soren@stoutner.com>.
-
- This file is part of Privacy Browser <https://www.stoutner.com/privacy-browser>.
-
- Privacy Browser 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.
-
- Privacy Browser 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 Privacy Browser. If not, see <http://www.gnu.org/licenses/>. -->
-
-<!-- The scroll view allows the linear layout to scroll if it exceeds the height of the page. -->
-<ScrollView
- xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:tools="http://schemas.android.com/tools"
- android:layout_height="wrap_content"
- android:layout_width="match_parent" >
-
- <!-- The background needs to be specified here so that it appears if about version is saved as an image. -->
- <LinearLayout
- android:id="@+id/about_version_linearlayout"
- android:layout_height="wrap_content"
- android:layout_width="match_parent"
- android:orientation="vertical"
- android:padding="16dp"
- android:background="?android:attr/colorBackground" >
-
- <!-- The `RelativeLayout` contains the header. -->
- <RelativeLayout
- android:layout_width="match_parent"
- android:layout_height="wrap_content" >
-
- <!--`tools:ignore="RtlSymmetry"` suppressed the lint warning about adding `android:paddingStart`, which wouldn't work with this layout.
- `tools:ignore="ContentDescription"` suppresses the lint warning about supplying a content description for the image view,
- which isn't needed in this case because the image view is only decorative. -->
- <ImageView
- android:id="@+id/icon"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:src="@mipmap/privacy_browser"
- android:paddingTop="10dp"
- android:paddingEnd="5dp"
- tools:ignore="RtlSymmetry,ContentDescription" />
-
- <TextView
- android:id="@+id/privacy_browser_textview"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:text="@string/privacy_browser"
- android:textStyle="bold"
- android:textSize="22sp"
- android:textColor="?attr/blueTitleTextColor"
- android:paddingTop="12dp"
- android:layout_toEndOf="@id/icon" />
-
- <TextView
- android:id="@+id/version"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textColor="?attr/blueTextColor"
- android:textIsSelectable="true"
- android:layout_below="@id/privacy_browser_textview"
- android:layout_toEndOf="@id/icon" />
- </RelativeLayout>
-
- <!-- The purpose of this linear layout is to provide padding on the start of the text views to make them line up with `about_version_icon`.
- Although we don't need it, we have to include `android:paddingEnd` to make lint happy. -->
- <LinearLayout
- android:layout_height="wrap_content"
- android:layout_width="match_parent"
- android:orientation="vertical"
- android:paddingTop="16dp"
- android:paddingStart="4dp"
- android:paddingEnd="0dp" >
-
- <!-- Hardware. -->
- <TextView
- android:id="@+id/hardware"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:text="@string/hardware"
- android:textStyle="bold"
- android:textSize="18sp"
- android:textColor="?attr/blueTitleTextColor" />
-
- <TextView
- android:id="@+id/brand"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/manufacturer"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/model"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/device"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/bootloader"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/radio"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <!-- Software. -->
- <TextView
- android:id="@+id/software"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:text="@string/software"
- android:textStyle="bold"
- android:textSize="18sp"
- android:textColor="?attr/blueTitleTextColor"
- android:paddingTop="12dp" />
-
- <TextView
- android:id="@+id/android"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/security_patch"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/build"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/webview_provider"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/webview_version"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/orbot"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/i2p"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/open_keychain"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <!-- Memory usage. -->
- <TextView
- android:id="@+id/memory_usage"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:text="@string/memory_usage"
- android:textStyle="bold"
- android:textSize="18sp"
- android:textColor="?attr/blueTitleTextColor"
- android:paddingTop="12dp" />
-
- <TextView
- android:id="@+id/app_consumed_memory"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/app_available_memory"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/app_total_memory"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/app_maximum_memory"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/system_consumed_memory"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/system_available_memory"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/system_total_memory"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <!-- Blocklists. -->
- <TextView
- android:id="@+id/blocklists"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:text="@string/blocklists"
- android:textStyle="bold"
- android:textSize="18sp"
- android:textColor="?attr/blueTitleTextColor"
- android:paddingTop="12dp" />
-
- <TextView
- android:id="@+id/easylist"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/easyprivacy"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/fanboy_annoyance"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/fanboy_social"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/ultralist"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/ultraprivacy"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <!-- Package Signature. -->
- <TextView
- android:id="@+id/package_signature"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:text="@string/package_signature"
- android:textStyle="bold"
- android:textSize="18sp"
- android:textColor="?attr/blueTitleTextColor"
- android:paddingTop="12dp" />
-
- <TextView
- android:id="@+id/certificate_issuer_dn"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/certificate_subject_dn"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/certificate_start_date"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/certificate_end_date"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/certificate_version"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/certificate_serial_number"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
-
- <TextView
- android:id="@+id/certificate_signature_algorithm"
- android:layout_height="wrap_content"
- android:layout_width="wrap_content"
- android:textIsSelectable="true" />
- </LinearLayout>
- </LinearLayout>
-</ScrollView>
\ No newline at end of file
--- /dev/null
+<?xml version="1.0" encoding="utf-8"?>
+
+<!--
+ Copyright © 2016-2018,2020 Soren Stoutner <soren@stoutner.com>.
+
+ This file is part of Privacy Browser <https://www.stoutner.com/privacy-browser>.
+
+ Privacy Browser 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.
+
+ Privacy Browser 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 Privacy Browser. If not, see <http://www.gnu.org/licenses/>. -->
+
+<!-- The scroll view allows the linear layout to scroll if it exceeds the height of the page. -->
+<ScrollView
+ xmlns:android="http://schemas.android.com/apk/res/android"
+ xmlns:tools="http://schemas.android.com/tools"
+ android:layout_height="wrap_content"
+ android:layout_width="match_parent" >
+
+ <!-- The background needs to be specified here so that it appears if about version is saved as an image. -->
+ <LinearLayout
+ android:id="@+id/about_version_linearlayout"
+ android:layout_height="wrap_content"
+ android:layout_width="match_parent"
+ android:orientation="vertical"
+ android:padding="16dp"
+ android:background="?android:attr/colorBackground" >
+
+ <!-- The `RelativeLayout` contains the header. -->
+ <RelativeLayout
+ android:layout_width="match_parent"
+ android:layout_height="wrap_content" >
+
+ <!--`tools:ignore="RtlSymmetry"` suppressed the lint warning about adding `android:paddingStart`, which wouldn't work with this layout.
+ `tools:ignore="ContentDescription"` suppresses the lint warning about supplying a content description for the image view,
+ which isn't needed in this case because the image view is only decorative. -->
+ <ImageView
+ android:id="@+id/icon"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:src="@mipmap/privacy_browser"
+ android:paddingTop="10dp"
+ android:paddingEnd="5dp"
+ tools:ignore="RtlSymmetry,ContentDescription" />
+
+ <TextView
+ android:id="@+id/privacy_browser_textview"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:text="@string/privacy_browser"
+ android:textStyle="bold"
+ android:textSize="22sp"
+ android:textColor="?attr/blueTitleTextColor"
+ android:paddingTop="12dp"
+ android:layout_toEndOf="@id/icon" />
+
+ <TextView
+ android:id="@+id/version"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textColor="?attr/blueTextColor"
+ android:textIsSelectable="true"
+ android:layout_below="@id/privacy_browser_textview"
+ android:layout_toEndOf="@id/icon" />
+ </RelativeLayout>
+
+ <!-- The purpose of this linear layout is to provide padding on the start of the text views to make them line up with `about_version_icon`.
+ Although we don't need it, we have to include `android:paddingEnd` to make lint happy. -->
+ <LinearLayout
+ android:layout_height="wrap_content"
+ android:layout_width="match_parent"
+ android:orientation="vertical"
+ android:paddingTop="16dp"
+ android:paddingStart="4dp"
+ android:paddingEnd="0dp" >
+
+ <!-- Hardware. -->
+ <TextView
+ android:id="@+id/hardware"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:text="@string/hardware"
+ android:textStyle="bold"
+ android:textSize="18sp"
+ android:textColor="?attr/blueTitleTextColor" />
+
+ <TextView
+ android:id="@+id/brand"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/manufacturer"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/model"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/device"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/bootloader"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/radio"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <!-- Software. -->
+ <TextView
+ android:id="@+id/software"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:text="@string/software"
+ android:textStyle="bold"
+ android:textSize="18sp"
+ android:textColor="?attr/blueTitleTextColor"
+ android:paddingTop="12dp" />
+
+ <TextView
+ android:id="@+id/android"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/security_patch"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/build"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/webview_provider"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/webview_version"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/orbot"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/i2p"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/open_keychain"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <!-- Memory usage. -->
+ <TextView
+ android:id="@+id/memory_usage"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:text="@string/memory_usage"
+ android:textStyle="bold"
+ android:textSize="18sp"
+ android:textColor="?attr/blueTitleTextColor"
+ android:paddingTop="12dp" />
+
+ <TextView
+ android:id="@+id/app_consumed_memory"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/app_available_memory"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/app_total_memory"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/app_maximum_memory"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/system_consumed_memory"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/system_available_memory"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/system_total_memory"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <!-- Blocklists. -->
+ <TextView
+ android:id="@+id/blocklists"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:text="@string/blocklists"
+ android:textStyle="bold"
+ android:textSize="18sp"
+ android:textColor="?attr/blueTitleTextColor"
+ android:paddingTop="12dp" />
+
+ <TextView
+ android:id="@+id/easylist"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/easyprivacy"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/fanboy_annoyance"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/fanboy_social"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/ultralist"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/ultraprivacy"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <!-- Package Signature. -->
+ <TextView
+ android:id="@+id/package_signature"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:text="@string/package_signature"
+ android:textStyle="bold"
+ android:textSize="18sp"
+ android:textColor="?attr/blueTitleTextColor"
+ android:paddingTop="12dp" />
+
+ <TextView
+ android:id="@+id/certificate_issuer_dn"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/certificate_subject_dn"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/certificate_start_date"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/certificate_end_date"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/certificate_version"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/certificate_serial_number"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+
+ <TextView
+ android:id="@+id/certificate_signature_algorithm"
+ android:layout_height="wrap_content"
+ android:layout_width="wrap_content"
+ android:textIsSelectable="true" />
+ </LinearLayout>
+ </LinearLayout>
+</ScrollView>
\ No newline at end of file
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Activities. -->
<string name="privacy_browser">Privacy Browser</string>
+ <string name="short_name">Browser</string>
<!-- For translations, `android_asset_path` should be the localization abbreviation. This should not be translated unless the Guide and About sections are localized. -->
<string name="android_asset_path">de</string>
<string name="file_is_mht">Die Datei ist ein MHT-Web-Archiv.</string>
<string name="mht_checkbox_explanation">Manchmal müssen MIME-gekapselte HTML-Web-Archive (MHT) manuell festgelegt werden, um korrekt geöffnet zu werden.</string>
- <!-- Save Dialogs. -->
+ <!-- Save Dialog. -->
<string name="save_url">URL speichern</string>
<string name="save_archive">Archiv speichern</string>
<string name="save_text">Text speichern</string>
<string name="save_image">Grafik speichern</string>
- <string name="save_logcat">Logcat speichern</string>
<string name="file_name">Dateiname</string>
- <string name="privacy_browser_logcat_txt">Privacy Browser Logcat.txt</string>
- <string name="privacy_browser_version_txt">Privacy Browser Version.txt</string>
- <string name="privacy_browser_version_png">Privacy Browser Version.png</string>
<string name="file">Datei</string>
<string name="bytes">Bytes</string>
<string name="unknown_size">Unbekannte Größe</string>
<string name="export_failed">Export fehlgeschlagen:</string>
<string name="import_failed">Import fehlgeschlagen:</string>
- <!-- Logcat. -->
+ <!-- Logcat. Android removes double spaces, but extra spaces can be manually specified with the Unicode `\u0020` formatting.
+ The `%1$s` code inserts variables into the displayed text and should be preserved in translation.-->
<string name="copy">kopieren</string>
- <string name="logcat_copied">Logcat kopiert.</string>
<string name="clear">leeren</string>
+ <string name="logcat_copied">Logcat kopiert.</string>
+ <string name="privacy_browser_logcat_txt">Privacy Browser %1$s Logcat.txt</string>
+ <string name="saved">%1$s gespeichert.</string>
+ <string name="error_saving_logcat">Fehler beim Speichern von Logcat: \u0020 %1$s</string>
<!-- Guide. -->
<string name="overview">Übersicht</string>
<string name="custom_proxy_invalid">Die benutzerdefinierte Proxy-URL ist ungültig.</string>
<string name="socks_proxies_do_not_work_on_kitkat">SOCKS-Proxies funktionieren nicht unter Android KitKat.</string>
- <!-- About Activity. -->
+ <!-- About Activity. The `%1$s` code inserts variables into the displayed text and should be preserved in translation. -->
<string name="about_privacy_browser">Über Privacy Browser</string>
<string name="version">Version</string>
<string name="version_code">Versions-Code</string>
<string name="serial_number">Seriennummer:</string>
<string name="signature_algorithm">Signaturalgorithmus:</string>
<string name="version_info_copied">Versions-Information wurde kopiert.</string>
+ <string name="privacy_browser_version_txt">Privacy Browser Version %1$s.txt</string>
+ <string name="privacy_browser_version_png">Privacy Browser Version %1$s.png</string>
<string name="permissions">Berechtigungen</string>
<string name="privacy_policy">Datenschutzrichtlinie</string>
<string name="changelog">Changelog</string>
<string name="download_with_external_app_summary">Eine externe App verwenden, um Dateien herunterzuladen.</string>
<string name="scroll_app_bar">App-Leiste scrollen</string>
<string name="scroll_app_bar_summary">Scrollt die App-Leiste mit der URL nach oben, wenn die Webseite gescrollt wird.</string>
+ <string name="bottom_app_bar">Untere Anwendungs-Leiste</string>
+ <string name="bottom_app_bar_summary">Zeigt die Anwendungs-Leiste am unteren Bildschirmrand an. Nach Ändern dieser Einstellung wird Privacy Browser neu gestartet.</string>
<string name="display_additional_app_bar_icons">Weitere Icons in der Titelleiste</string>
<string name="display_additional_app_bar_icons_summary">Icons zum Neu-Laden von WebView und - so genug Platz zur Verfügung steht -
zum Öffnen der Lesezeichen und zum Umschalten von Cookies in der App-Leiste anzeigen.</string>
<resources>
<!-- Activities. -->
<string name="privacy_browser">Navegador Privado</string>
+ <string name="short_name">Browser</string>
<!-- For translations, `android_asset_path` should be the localization abbreviation. This should not be translated unless the Guide and About sections are localized. -->
<string name="android_asset_path">es</string>
<string name="file_is_mht">El archivo es un archivo de web MHT.</string>
<string name="mht_checkbox_explanation">A veces se necesita especificar manualmente los archivos web MIME Encapsulated HTML (MHT) para que se abran correctamente.</string>
- <!-- Save Dialogs. -->
+ <!-- Save Dialog. -->
<string name="save_url">Guardar URL</string>
<string name="save_archive">Guardar archivo</string>
<string name="save_text">Guardar texto</string>
<string name="save_image">Guardar imagen</string>
- <string name="save_logcat">Guardar logcat</string>
<string name="file_name">Nombre de archivo</string>
- <string name="privacy_browser_logcat_txt">Navegador Privado Logcat.txt</string>
- <string name="privacy_browser_version_txt">Versión de Navegador Privado.txt</string>
- <string name="privacy_browser_version_png">Versiótoutnern de Navegador Privado.png</string>
<string name="file">Archivo</string>
<string name="bytes">bytes</string>
<string name="unknown_size">Tamaño desconocido</string>
<string name="export_failed">Exportación fallida:</string>
<string name="import_failed">Importación fallida:</string>
- <!-- Logcat. -->
+ <!-- Logcat. Android removes double spaces, but extra spaces can be manually specified with the Unicode `\u0020` formatting.
+ The `%1$s` code inserts variables into the displayed text and should be preserved in translation.-->
<string name="copy">Copiar</string>
- <string name="logcat_copied">Logcat copiado.</string>
<string name="clear">Borrar</string>
+ <string name="logcat_copied">Logcat copiado.</string>
+ <string name="privacy_browser_logcat_txt">Navegador Privado %1$s Logcat.txt</string>
+ <string name="saved">%1$s guardado.</string>
+ <string name="error_saving_logcat">Error guardando logcat: \u0020 %1$s</string>
<!-- Guide. -->
<string name="overview">Visión general</string>
<string name="custom_proxy_invalid">La URL del proxy personalizado no es válida.</string>
<string name="socks_proxies_do_not_work_on_kitkat">SOCKS proxies do not work on Android KitKat.</string>
- <!-- About Activity. -->
+ <!-- About Activity. The `%1$s` code inserts variables into the displayed text and should be preserved in translation. -->
<string name="about_privacy_browser">Acerca de Navegador Privado</string>
<string name="version">Versión</string>
<string name="version_code">código de versión</string>
<string name="serial_number">Número de serie:</string>
<string name="signature_algorithm">Algoritmo de firma:</string>
<string name="version_info_copied">Información de la versión copiada.</string>
+ <string name="privacy_browser_version_txt">Versión de Navegador Privado %1$s.txt</string>
+ <string name="privacy_browser_version_png">Versión de Navegador Privado %1$s.png</string>
<string name="permissions">Permisos</string>
<string name="privacy_policy">Política de privacidad</string>
<string name="changelog">Historial de cambios</string>
<string name="download_with_external_app_summary">Use una app externa para descargar archivos.</string>
<string name="scroll_app_bar">Desplazar la barra de aplicaciones</string>
<string name="scroll_app_bar_summary">Desplazar la barra de aplicaciones desde la parte superior de la pantalla cuando el WebView se desplaza hacia abajo.</string>
+ <string name="bottom_app_bar">Barra inferior de la app</string>
+ <string name="bottom_app_bar_summary">Mover la barra de apps a la parte inferior de la pantalla. Al cambiar esta configuración se reiniciará Navegador Privado.</string>
<string name="display_additional_app_bar_icons">Mostrar iconos adicionales en la barra de aplicación</string>
<string name="display_additional_app_bar_icons_summary">Mostrar iconos en la barra de aplicaciones para refrescar el WebView y, si hay espacio,
para abrir el cajón de marcadores y cambiar las cookies.</string>
<resources>
<!-- Activities. -->
<string name="privacy_browser">Privacy Browser</string>
+ <string name="short_name">Browser</string>
<!-- For translations, `android_asset_path` should be the localization abbreviation. For example, Spanish is `es`. This should not be translated unless the Guide and About sections are localized. -->
<string name="android_asset_path">fr</string>
<string name="file_is_mht">Le fichier est une archive web MHT.</string>
<string name="mht_checkbox_explanation">Parfois, les archives web MHT (MIME Encapsulated HTML) doivent être spécifiées manuellement pour être ouvertes correctement.</string>
- <!-- Save Dialogs. -->
+ <!-- Save Dialog. -->
<string name="save_url">Enregistrer l\'URL</string>
<string name="save_archive">Enregistrer l\'archive</string>
<string name="save_text">Sauvegarder le texte</string>
<string name="save_image">Sauvegarder en tant qu\'image</string>
- <string name="save_logcat">Sauvegarder le journal système</string>
<string name="file_name">Nom du fichier</string>
- <string name="privacy_browser_logcat_txt">Privacy Browser Logcat.txt</string>
- <string name="privacy_browser_version_txt">Privacy Browser Version.txt</string>
- <string name="privacy_browser_version_png">Privacy Browser Version.png</string>
<string name="file">Fichier</string>
<string name="bytes">octets</string>
<string name="unknown_size">taille inconnue</string>
<string name="export_failed">L\'export a échoué :</string>
<string name="import_failed">L\'import a échoué :</string>
- <!-- Logcat. -->
+ <!-- Logcat. Android removes double spaces, but extra spaces can be manually specified with the Unicode `\u0020` formatting.
+ The `%1$s` code inserts variables into the displayed text and should be preserved in translation.-->
<string name="copy">Copie</string>
- <string name="logcat_copied">Journal système copié.</string>
<string name="clear">Vider</string>
+ <string name="logcat_copied">Journal système copié.</string>
+ <string name="privacy_browser_logcat_txt">Privacy Browser %1$s Logcat.txt</string>
+ <string name="saved">%1$s sauvegardé.</string>
+ <string name="error_saving_logcat">Erreur de sauvegarde du logcat: \u0020 %1$s</string>
<!-- Guide. -->
<string name="overview">Présentation</string>
<string name="custom_proxy_invalid">L\'URL du proxy personnalisé n\'est pas valide.</string>
<string name="socks_proxies_do_not_work_on_kitkat">Les proxys de type SOCKS ne fonctionnent pas sur Android KitKat.</string>
- <!-- About Activity. -->
+ <!-- About Activity. The `%1$s` code inserts variables into the displayed text and should be preserved in translation. -->
<string name="about_privacy_browser">À propos</string>
<string name="version">Version</string>
<string name="version_code">Code de la version</string>
<string name="serial_number">Numéro de série :</string>
<string name="signature_algorithm">Algorithme de chiffrement :</string>
<string name="version_info_copied">Informations de version copiées.</string>
+ <string name="privacy_browser_version_txt">Privacy Browser Version %1$s.txt</string>
+ <string name="privacy_browser_version_png">Privacy Browser Version %1$s.png</string>
<string name="permissions">Permissions</string>
<string name="privacy_policy">Politique de confidentialité</string>
<string name="changelog">Journal des changements</string>
<string name="swipe_to_refresh_summary">Certains sites Web ne fonctionnent pas bien lorsque "Glisser pour rafraîchir" est activé.</string>
<string name="download_with_external_app">Téléchargement avec une app externe</string>
<string name="download_with_external_app_summary">Utiliser une application externe pour télécharger des fichiers.</string>
+ <string name="bottom_app_bar">Barre d\'application en bas</string>
+ <string name="bottom_app_bar_summary">Déplacer la barre d\'application vers le bas de l\'écran. La modification de ce paramètre entraîne le redémarrage de Privacy Browser.</string>
<string name="scroll_app_bar">Défilement barre d\'applications</string>
<string name="scroll_app_bar_summary">Faites défiler la barre d\'applications en haut de l\'écran lorsque WebView défile vers le bas.</string>
<string name="display_additional_app_bar_icons">Icônes supplémentaires dans la barre d\'applications</string>
<resources>
<!-- Activities. -->
<string name="privacy_browser">Privacy Browser</string>
+ <string name="short_name">Browser</string>
<!-- For translations, `android_asset_path` should be the localization abbreviation. This should not be translated unless the Guide and About sections are localized. -->
<string name="android_asset_path">it</string>
<string name="file_is_mht">Questo file è un archivio web MHT.</string>
<string name="mht_checkbox_explanation">Talvolta gli archivi web del tipo MIME Encapsulated HTML (MHT) devono essere specificati manualmente per essere aperti correttamente.</string>
- <!-- Save Dialogs. -->
+ <!-- Save Dialog. -->
<string name="save_url">Salva URL</string>
<string name="save_archive">Salva Archivio</string>
<string name="save_text">Salva Testo</string>
<string name="save_image">Salva Immagine</string>
- <string name="save_logcat">Salva il log</string>
<string name="file_name">Nome File</string>
- <string name="privacy_browser_logcat_txt">Privacy Browser Logcat.txt</string>
- <string name="privacy_browser_version_txt">Versione di Privacy Browser.txt</string>
- <string name="privacy_browser_version_png">Versione di Privacy Browser.png</string>
<string name="file">File</string>
<string name="bytes">byte</string>
<string name="unknown_size">Dimensione sconosciuta</string>
<string name="export_failed">Esportazione fallita:</string>
<string name="import_failed">Importazione fallita:</string>
- <!-- Logcat. -->
+ <!-- Logcat. Android removes double spaces, but extra spaces can be manually specified with the Unicode `\u0020` formatting.
+ The `%1$s` code inserts variables into the displayed text and should be preserved in translation.-->
<string name="copy">Copia</string>
- <string name="logcat_copied">Logcat copiato.</string>
<string name="clear">Cancella</string>
+ <string name="logcat_copied">Logcat copiato.</string>
+ <string name="privacy_browser_logcat_txt">Privacy Browser %1$s Logcat.txt</string>
+ <string name="saved">%1$s salvato.</string>
+ <string name="error_saving_logcat">Errore salvataggio logcat: \u0020 %1$s</string>
<!-- Guide. -->
<string name="overview">Descrizione</string>
<string name="custom_proxy_invalid">La URL del proxy personalizzato non è valida.</string>
<string name="socks_proxies_do_not_work_on_kitkat">I proxy SOCKS non funzionano con Android KitKat.</string>
- <!-- About Activity. -->
+ <!-- About Activity. The `%1$s` code inserts variables into the displayed text and should be preserved in translation. -->
<string name="about_privacy_browser">Informazioni su Privacy Browser</string>
<string name="version">Versione</string>
<string name="version_code">codice versione</string>
<string name="serial_number">Numero di Serie:</string>
<string name="signature_algorithm">Algoritmo di firma:</string>
<string name="version_info_copied">Info sulla Versione copiate.</string>
+ <string name="privacy_browser_version_txt">Versione di Privacy Browser %1$s.txt</string>
+ <string name="privacy_browser_version_png">Versione di Privacy Browser %1$s.png</string>
<string name="permissions">Autorizzazioni</string>
<string name="privacy_policy">Privacy Policy</string>
<string name="changelog">Changelog</string>
<string name="download_with_external_app_summary">Utilizza una applicazione esterna per scaricare i file.</string>
<string name="scroll_app_bar">Permetti lo scrolling della barra dell\'applicazione</string>
<string name="scroll_app_bar_summary">Permette lo scorrere della barra dell\'applicazione dalla parte alta dello schermo quando si effettua lo scrolling.</string>
+ <string name="bottom_app_bar">Barra dell\'app in basso</string>
+ <string name="bottom_app_bar_summary">Sposta la barra dell\'app nella parte bassa dello schermo. La modifica di questa impostazione provocherà il riavvio di Privacy Browser.</string>
<string name="display_additional_app_bar_icons">Mostra icone addizionali nella barra dell\'applicazione</string>
<string name="display_additional_app_bar_icons_summary">Mostra nella barra dell\'applicazione le icone per l\'aggiornamento della pagina e, se lo spazio è disponibile,
per aprire i preferiti e per abilitare o disabilitare i cookie.</string>
<item name="lockBlueIcon">@drawable/lock_night</item>
<item name="moveToFolderBlueIcon">@drawable/move_to_folder_blue_night</item>
<item name="proxyBlueIcon">@drawable/proxy_enabled_night</item>
- <item name="saveBlueIcon">@drawable/save_dialog_night</item>
- <item name="saveTextBlueIcon">@drawable/save_text_blue_night</item>
<item name="sslCertificateBlueIcon">@drawable/ssl_certificate_enabled_night</item>
</style>
</resources>
\ No newline at end of file
<item name="lockBlueIcon">@drawable/lock_night</item>
<item name="moveToFolderBlueIcon">@drawable/move_to_folder_blue_night</item>
<item name="proxyBlueIcon">@drawable/proxy_enabled_night</item>
- <item name="saveBlueIcon">@drawable/save_dialog_night</item>
- <item name="saveTextBlueIcon">@drawable/save_text_blue_night</item>
<item name="sslCertificateBlueIcon">@drawable/ssl_certificate_enabled_night</item>
</style>
</resources>
\ No newline at end of file
<!-- Colors are currently hardcoded in the vector drawables. Once the minimum API is >= 22 the hardcoded colors can be removed and they can reference these entries instead. -->
<resources>
<!-- Nicknamed colors. -->
+ <color name="about_version_blue_text">@color/violet_700</color>
<color name="blue_text">@color/violet_500</color>
<!-- Raw colors. -->
<item name="lockBlueIcon">@drawable/lock_night</item>
<item name="moveToFolderBlueIcon">@drawable/move_to_folder_blue_night</item>
<item name="proxyBlueIcon">@drawable/proxy_enabled_night</item>
- <item name="saveBlueIcon">@drawable/save_dialog_night</item>
- <item name="saveTextBlueIcon">@drawable/save_text_blue_night</item>
<item name="sslCertificateBlueIcon">@drawable/ssl_certificate_enabled_night</item>
</style>
</resources>
\ No newline at end of file
<string name="file_is_mht">Este é um arquivo MHT da web.</string>
<string name="mht_checkbox_explanation">Às vezes, os arquivos da web MIME Encapsulated HTML (MHT) precisam ser especificados manualmente para serem abertos corretamente.</string>
- <!-- Save Dialogs. -->
+ <!-- Save Dialog. -->
<string name="save_url">Salvar URL</string>
<string name="save_archive">Salvar Arquivo</string>
<string name="save_text">Salvar Texto</string>
<string name="save_image">Salvar Imagem</string>
- <string name="save_logcat">Salvar logcat</string>
<string name="file_name">Nome do Arquivo</string>
- <string name="privacy_browser_logcat_txt">Privacy Browser Logcat.txt</string>
- <string name="privacy_browser_version_txt">Privacy Browser Versão.txt</string>
- <string name="privacy_browser_version_png">Privacy Browser Versão.png</string>
<string name="file">Arquivo</string>
<string name="bytes">bytes</string>
<string name="unknown_size">tamanho desconhecido</string>
<string name="export_failed">A exportação falhou:</string>
<string name="import_failed">A importação falhou:</string>
- <!-- Logcat. -->
+ <!-- Logcat. Android removes double spaces, but extra spaces can be manually specified with the Unicode `\u0020` formatting.
+ The `%1$s` code inserts variables into the displayed text and should be preserved in translation.-->
<string name="copy">Cópia</string>
- <string name="logcat_copied">Logcat copiado.</string>
<string name="clear">Limpar</string>
+ <string name="logcat_copied">Logcat copiado.</string>
+ <string name="privacy_browser_logcat_txt">Privacy Browser %1$s Logcat.txt</string>
<!-- Guide. -->
<string name="overview">Visão geral</string>
<string name="custom_proxy_invalid">O URL do proxy personalizado é inválido.</string>
<string name="socks_proxies_do_not_work_on_kitkat">Os proxies SOCKS não funcionam no Android KitKat.</string>
- <!-- About Activity. -->
+ <!-- About Activity. The `%1$s` code inserts variables into the displayed text and should be preserved in translation. -->
<string name="about_privacy_browser">Sobre o Privacy Browser</string>
<string name="version">Versão</string>
<string name="version_code">Código da Versão</string>
<string name="serial_number">Número de série:</string>
<string name="signature_algorithm">Algoritmo de Assinatura:</string>
<string name="version_info_copied">Informações de versão copiada.</string>
+ <string name="privacy_browser_version_txt">Privacy Browser Versão %1$s.txt</string>
+ <string name="privacy_browser_version_png">Privacy Browser Versão %1$s.png</string>
<string name="permissions">Permissões</string>
<string name="privacy_policy">Política de Privacidade</string>
<string name="changelog">Changelog</string>
<resources>
<!-- Activities. -->
<string name="privacy_browser">Privacy Browser</string>
+ <string name="short_name">Браузер</string>
<!-- For translations, `android_asset_path` should be the localization abbreviation. This should not be translated unless the Guide and About sections are localized. -->
<string name="android_asset_path">ru</string> -->
<string name="file_is_mht">Файл представляет собой веб-архив MHT.</string>
<string name="mht_checkbox_explanation">Иногда для корректного открытия веб-архивов MIME Encapsulated HTML (MHT) необходимо указать вручную.</string>
- <!-- Save Dialogs. -->
+ <!-- Save Dialog. -->
<string name="save_url">Сохранить URL</string>
<string name="save_archive">Сохранить архив</string>
<string name="save_text">Сохранить текст</string>
<string name="save_image">Сохранить изображение</string>
- <string name="save_logcat">Сохранить logcat</string>
<string name="file_name">Имя файла</string>
- <string name="privacy_browser_logcat_txt">Privacy Browser Logcat.txt</string>
- <string name="privacy_browser_version_txt">Версия Privacy Browser.txt</string>
- <string name="privacy_browser_version_png">Версия Privacy Browser.png</string>
<string name="file">Файл</string>
<string name="bytes">байтов</string>
<string name="unknown_size">неизвестный размер</string>
<string name="export_failed">Сбой при экспорте:</string>
<string name="import_failed">Сбой при импорте:</string>
- <!-- Logcat. -->
+ <!-- Logcat. Android removes double spaces, but extra spaces can be manually specified with the Unicode `\u0020` formatting.
+ The `%1$s` code inserts variables into the displayed text and should be preserved in translation.-->
<string name="copy">Копировать</string>
- <string name="logcat_copied">Logcat скопирован.</string>
<string name="clear">Очистить</string>
+ <string name="logcat_copied">Logcat скопирован.</string>
+ <string name="privacy_browser_logcat_txt">Privacy Browser %1$s Logcat.txt</string>
+ <string name="saved">%1$s сохранен.</string>
+ <string name="error_saving_logcat">Ошибка сохранения logcat: \u0020 %1$s</string>
<!-- Guide. -->
<string name="overview">Обзор</string>
<string name="custom_proxy_invalid">URL пользовательского прокси недействителен.</string>
<string name="socks_proxies_do_not_work_on_kitkat">SOCKS-прокси не работает на Android KitKat.</string>
- <!-- About Activity. -->
+ <!-- About Activity. The `%1$s` code inserts variables into the displayed text and should be preserved in translation. -->
<string name="about_privacy_browser">О Privacy Browser</string>
<string name="version">Версия</string>
<string name="version_code">код версии</string>
<string name="serial_number">Серийный номер:</string>
<string name="signature_algorithm">Алгоритм подписи:</string>
<string name="version_info_copied">Информация о версии скопирована.</string>
+ <string name="privacy_browser_version_txt">Версия Privacy Browser %1$s.txt</string>
+ <string name="privacy_browser_version_png">Версия Privacy Browser %1$s.png</string>
<string name="permissions">Разрешения</string>
<string name="privacy_policy">Политика конфиденциальности</string>
<string name="changelog">История изменений</string>
<string name="download_with_external_app_summary">Использовать внешнее приложение для загрузки файлов.</string>
<string name="scroll_app_bar">Прокручивать панель приложения</string>
<string name="scroll_app_bar_summary">Прокручивает панель приложения вверху экрана при прокрутке WebView вниз.</string>
+ <string name="bottom_app_bar">Нижняя панель приложения</string>
+ <string name="bottom_app_bar_summary">Перемещает панель приложения в нижнюю часть экрана. Изменение этой настройки приведет к перезапуску Privacy Browser.</string>
<string name="display_additional_app_bar_icons">Отображать дополнительные значки на панели приложения</string>
<string name="display_additional_app_bar_icons_summary">Показывать в панели приложения значки для обновления WebView и, если есть место, для открытия окна закладок и переключения cookie.</string>
<string name="app_theme">Тема приложения</string>
<string name="previous">Önceki</string>
<string name="next">Sonraki</string>
- <!-- Save Dialogs. -->
+ <!-- Save Dialog. -->
<string name="save_image">Resmi kaydet</string>
- <string name="save_logcat">Logcat kaydet</string>
<string name="file_name">Dosya adı</string>
<string name="unknown_size">Bilinmeyen boyut</string>
- <string name="privacy_browser_logcat_txt">Privacy Browser Logcat.txt</string>
<!-- View Source. -->
<string name="request_headers">İstek Başlıkları</string>
<string name="export_failed">Dışa aktarım başarısız:</string>
<string name="import_failed">İçe aktarım başarısız:</string>
- <!-- Logcat. -->
+ <!-- Logcat. Android removes double spaces, but extra spaces can be manually specified with the Unicode `\u0020` formatting.
+ The `%1$s` code inserts variables into the displayed text and should be preserved in translation.-->
<string name="copy">Kopyala</string>
- <string name="logcat_copied">Logcat kopyalandı.</string>
<string name="clear">Temizle</string>
+ <string name="logcat_copied">Logcat kopyalandı.</string>
+ <string name="privacy_browser_logcat_txt">Privacy Browser %1$s Logcat.txt</string>
<!-- Guide. -->
<string name="overview">Genel Bakış</string>
<item name="lockBlueIcon">@drawable/lock_day</item>
<item name="moveToFolderBlueIcon">@drawable/move_to_folder_blue_day</item>
<item name="proxyBlueIcon">@drawable/proxy_enabled_day</item>
- <item name="saveBlueIcon">@drawable/save_dialog_day</item>
- <item name="saveTextBlueIcon">@drawable/save_text_blue_day</item>
<item name="sslCertificateBlueIcon">@drawable/ssl_certificate_enabled_day</item>
</style>
</resources>
\ No newline at end of file
<item name="lockBlueIcon">@drawable/lock_day</item>
<item name="moveToFolderBlueIcon">@drawable/move_to_folder_blue_day</item>
<item name="proxyBlueIcon">@drawable/proxy_enabled_day</item>
- <item name="saveBlueIcon">@drawable/save_dialog_day</item>
- <item name="saveTextBlueIcon">@drawable/save_text_blue_day</item>
<item name="sslCertificateBlueIcon">@drawable/ssl_certificate_enabled_day</item>
</style>
</resources>
\ No newline at end of file
<attr name="lockBlueIcon" format="reference" />
<attr name="moveToFolderBlueIcon" format="reference" />
<attr name="proxyBlueIcon" format="reference" />
- <attr name="saveBlueIcon" format="reference" />
- <attr name="saveTextBlueIcon" format="reference" />
<attr name="sslCertificateBlueIcon" format="reference" />
</resources>
\ No newline at end of file
<!-- Colors are currently hardcoded in the vector drawables. Once the minimum API is >= 22 the hardcoded colors can be removed and they can reference these entries instead. -->
<resources>
<!-- Nicknamed colors. -->
+ <color name="about_version_blue_text">@color/blue_700</color>
<color name="blue_text">@color/blue_700</color>
<!-- Raw colors. -->
<string name="file_is_mht">The file is an MHT web archive.</string>
<string name="mht_checkbox_explanation">Sometimes MIME Encapsulated HTML (MHT) web archives need to be manually specified to be opened correctly.</string>
- <!-- Save Dialogs. -->
+ <!-- Save Dialog. -->
<string name="save_dialog" translatable="false">Save Dialog</string> <!-- This string is used to tag the save dialog. It is never displayed to the user. -->
<string name="save_url">Save URL</string>
<string name="save_archive">Save Archive</string>
<string name="save_text">Save Text</string>
<string name="save_image">Save Image</string>
- <string name="save_logcat">Save Logcat</string>
<string name="file_name">File name</string>
- <string name="privacy_browser_logcat_txt">Privacy Browser Logcat.txt</string>
- <string name="privacy_browser_version_txt">Privacy Browser Version.txt</string>
- <string name="privacy_browser_version_png">Privacy Browser Version.png</string>
<string name="file">File</string>
<string name="bytes">bytes</string>
<string name="unknown_size">unknown size</string>
<string name="export_failed">Export failed:</string>
<string name="import_failed">Import failed:</string>
- <!-- Logcat. -->
+ <!-- Logcat. Android removes double spaces, but extra spaces can be manually specified with the Unicode `\u0020` formatting.
+ The `%1$s` code inserts variables into the displayed text and should be preserved in translation.-->
<string name="copy">Copy</string>
- <string name="logcat_copied">Logcat copied.</string>
<string name="clear">Clear</string>
+ <string name="logcat_copied">Logcat copied.</string>
+ <string name="privacy_browser_logcat_txt">Privacy Browser %1$s Logcat.txt</string>
+ <string name="saved">%1$s saved.</string>
+ <string name="error_saving_logcat">Error saving logcat: \u0020 %1$s</string>
<!-- Guide. -->
<string name="overview">Overview</string>
<string name="custom_proxy_invalid">The custom proxy URL is invalid.</string>
<string name="socks_proxies_do_not_work_on_kitkat">SOCKS proxies do not work on Android KitKat.</string>
- <!-- About Activity. -->
+ <!-- About Activity. The `%1$s` code inserts variables into the displayed text and should be preserved in translation. -->
<string name="about_privacy_browser">About Privacy Browser</string>
<string name="version">Version</string>
<string name="version_code">version code</string>
<string name="serial_number">Serial Number:</string>
<string name="signature_algorithm">Signature Algorithm:</string>
<string name="version_info_copied">Version info copied.</string>
+ <string name="privacy_browser_version_txt">Privacy Browser Version %1$s.txt</string>
+ <string name="privacy_browser_version_png">Privacy Browser Version %1$s.png</string>
<string name="permissions">Permissions</string>
<string name="privacy_policy">Privacy Policy</string>
<string name="changelog">Changelog</string>
<item name="lockBlueIcon">@drawable/lock_day</item>
<item name="moveToFolderBlueIcon">@drawable/move_to_folder_blue_day</item>
<item name="proxyBlueIcon">@drawable/proxy_enabled_day</item>
- <item name="saveBlueIcon">@drawable/save_dialog_day</item>
- <item name="saveTextBlueIcon">@drawable/save_text_blue_day</item>
<item name="sslCertificateBlueIcon">@drawable/ssl_certificate_enabled_day</item>
</style>
</resources>
\ No newline at end of file
dependencies {
classpath 'com.android.tools.build:gradle:7.0.3'
- classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:1.5.31"
+ classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:1.6.0"
// NOTE: Do not place your application dependencies here; they belong
// in the individual module build.gradle files