switch UserPreference to kotlin
This commit is contained in:
parent
82011f7f8b
commit
2f75559ac2
2 changed files with 460 additions and 560 deletions
|
@ -1,560 +0,0 @@
|
||||||
package com.zeapo.pwdstore;
|
|
||||||
|
|
||||||
import android.Manifest;
|
|
||||||
import android.accessibilityservice.AccessibilityServiceInfo;
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.app.DialogFragment;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.DialogInterface;
|
|
||||||
import android.content.Intent;
|
|
||||||
import android.content.SharedPreferences;
|
|
||||||
import android.content.pm.PackageManager;
|
|
||||||
import android.graphics.Color;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.os.Bundle;
|
|
||||||
import android.os.Environment;
|
|
||||||
import android.preference.CheckBoxPreference;
|
|
||||||
import android.preference.Preference;
|
|
||||||
import android.preference.PreferenceFragment;
|
|
||||||
import android.preference.PreferenceManager;
|
|
||||||
import android.provider.Settings;
|
|
||||||
import android.support.annotation.NonNull;
|
|
||||||
import android.support.design.widget.Snackbar;
|
|
||||||
import android.support.v4.app.ActivityCompat;
|
|
||||||
import android.support.v4.content.ContextCompat;
|
|
||||||
import android.support.v7.app.AlertDialog;
|
|
||||||
import android.support.v7.app.AppCompatActivity;
|
|
||||||
import android.util.Log;
|
|
||||||
import android.view.MenuItem;
|
|
||||||
import android.view.View;
|
|
||||||
import android.view.accessibility.AccessibilityManager;
|
|
||||||
import android.widget.TextView;
|
|
||||||
import android.widget.Toast;
|
|
||||||
|
|
||||||
import com.nononsenseapps.filepicker.FilePickerActivity;
|
|
||||||
import com.zeapo.pwdstore.autofill.AutofillPreferenceActivity;
|
|
||||||
import com.zeapo.pwdstore.crypto.PgpHandler;
|
|
||||||
import com.zeapo.pwdstore.git.GitActivity;
|
|
||||||
import com.zeapo.pwdstore.utils.PasswordRepository;
|
|
||||||
|
|
||||||
import org.apache.commons.io.FileUtils;
|
|
||||||
import org.apache.commons.io.IOUtils;
|
|
||||||
import org.openintents.openpgp.util.OpenPgpUtils;
|
|
||||||
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.text.SimpleDateFormat;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Date;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Locale;
|
|
||||||
|
|
||||||
public class UserPreference extends AppCompatActivity {
|
|
||||||
private final static int IMPORT_SSH_KEY = 1;
|
|
||||||
private final static int IMPORT_PGP_KEY = 2;
|
|
||||||
private final static int EDIT_GIT_INFO = 3;
|
|
||||||
private final static int SELECT_GIT_DIRECTORY = 4;
|
|
||||||
private final static int EXPORT_PASSWORDS = 5;
|
|
||||||
private final static int EDIT_GIT_CONFIG = 6;
|
|
||||||
private final static int REQUEST_EXTERNAL_STORAGE = 50;
|
|
||||||
private PrefsFragment prefsFragment;
|
|
||||||
|
|
||||||
public static class PrefsFragment extends PreferenceFragment {
|
|
||||||
@Override
|
|
||||||
public void onCreate(Bundle savedInstanceState) {
|
|
||||||
super.onCreate(savedInstanceState);
|
|
||||||
final UserPreference callingActivity = (UserPreference) getActivity();
|
|
||||||
final SharedPreferences sharedPreferences = getPreferenceManager().getSharedPreferences();
|
|
||||||
|
|
||||||
addPreferencesFromResource(R.xml.preference);
|
|
||||||
|
|
||||||
findPreference("app_version").setSummary(String.format("Version: %s", BuildConfig.VERSION_NAME));
|
|
||||||
|
|
||||||
findPreference("openpgp_key_id_pref").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
Intent intent = new Intent(callingActivity, PgpHandler.class);
|
|
||||||
intent.putExtra("Operation", "GET_KEY_ID");
|
|
||||||
startActivityForResult(intent, IMPORT_PGP_KEY);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
findPreference("ssh_key").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
callingActivity.getSshKeyWithPermissions(sharedPreferences.getBoolean("use_android_file_picker", false));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
findPreference("ssh_keygen").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
callingActivity.makeSshKey(true);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
findPreference("ssh_see_key").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
DialogFragment df = new SshKeyGen.ShowSshKeyFragment();
|
|
||||||
df.show(getFragmentManager(), "public_key");
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
findPreference("git_server_info").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
Intent intent = new Intent(callingActivity, GitActivity.class);
|
|
||||||
intent.putExtra("Operation", GitActivity.EDIT_SERVER);
|
|
||||||
startActivityForResult(intent, EDIT_GIT_INFO);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
findPreference("git_config").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
Intent intent = new Intent(callingActivity, GitActivity.class);
|
|
||||||
intent.putExtra("Operation", GitActivity.EDIT_GIT_CONFIG);
|
|
||||||
startActivityForResult(intent, EDIT_GIT_CONFIG);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
findPreference("git_delete_repo").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
new AlertDialog.Builder(callingActivity).
|
|
||||||
setTitle(R.string.pref_dialog_delete_title).
|
|
||||||
setMessage(getResources().getString(R.string.dialog_delete_msg)
|
|
||||||
+ " \n" + PasswordRepository.getRepositoryDirectory(callingActivity.getApplicationContext()).toString()).
|
|
||||||
setCancelable(false).
|
|
||||||
setPositiveButton(R.string.dialog_delete, new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialogInterface, int i) {
|
|
||||||
try {
|
|
||||||
FileUtils.cleanDirectory(PasswordRepository.getRepositoryDirectory(callingActivity.getApplicationContext()));
|
|
||||||
PasswordRepository.closeRepository();
|
|
||||||
} catch (Exception e) {
|
|
||||||
//TODO Handle the diffent cases of exceptions
|
|
||||||
}
|
|
||||||
|
|
||||||
sharedPreferences.edit().putBoolean("repository_initialized", false).apply();
|
|
||||||
dialogInterface.cancel();
|
|
||||||
callingActivity.finish();
|
|
||||||
}
|
|
||||||
}).
|
|
||||||
setNegativeButton(R.string.dialog_do_not_delete,
|
|
||||||
new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialogInterface, int i) {
|
|
||||||
{
|
|
||||||
dialogInterface.cancel();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}).show();
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
final Preference externalRepo = findPreference("pref_select_external");
|
|
||||||
externalRepo.setSummary(getPreferenceManager().getSharedPreferences().getString("git_external_repo", callingActivity.getString(R.string.no_repo_selected)));
|
|
||||||
externalRepo.setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
callingActivity.selectExternalGitRepository();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
Preference.OnPreferenceChangeListener resetRepo = new Preference.OnPreferenceChangeListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceChange(Preference preference, Object o) {
|
|
||||||
findPreference("git_delete_repo").setEnabled(!(Boolean) o);
|
|
||||||
PasswordRepository.closeRepository();
|
|
||||||
getPreferenceManager().getSharedPreferences().edit().putBoolean("repo_changed", true).apply();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
findPreference("pref_select_external").setOnPreferenceChangeListener(resetRepo);
|
|
||||||
findPreference("git_external").setOnPreferenceChangeListener(resetRepo);
|
|
||||||
|
|
||||||
findPreference("autofill_apps").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
Intent intent = new Intent(callingActivity, AutofillPreferenceActivity.class);
|
|
||||||
startActivity(intent);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
findPreference("autofill_enable").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
new AlertDialog.Builder(callingActivity).
|
|
||||||
setTitle(R.string.pref_autofill_enable_title).
|
|
||||||
setView(R.layout.autofill_instructions).
|
|
||||||
setPositiveButton(R.string.dialog_ok, new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
|
||||||
Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
|
|
||||||
startActivity(intent);
|
|
||||||
}
|
|
||||||
}).
|
|
||||||
setNegativeButton(R.string.dialog_cancel, null).
|
|
||||||
setOnDismissListener(new DialogInterface.OnDismissListener() {
|
|
||||||
@Override
|
|
||||||
public void onDismiss(DialogInterface dialog) {
|
|
||||||
((CheckBoxPreference) findPreference("autofill_enable"))
|
|
||||||
.setChecked(((UserPreference) getActivity()).isServiceEnabled());
|
|
||||||
}
|
|
||||||
}).show();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
findPreference("export_passwords").setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
callingActivity.exportPasswordsWithPermissions();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onStart() {
|
|
||||||
super.onStart();
|
|
||||||
final SharedPreferences sharedPreferences = getPreferenceManager().getSharedPreferences();
|
|
||||||
findPreference("pref_select_external").setSummary(getPreferenceManager().getSharedPreferences().getString("git_external_repo", getString(R.string.no_repo_selected)));
|
|
||||||
findPreference("ssh_see_key").setEnabled(sharedPreferences.getBoolean("use_generated_key", false));
|
|
||||||
findPreference("git_delete_repo").setEnabled(!sharedPreferences.getBoolean("git_external", false));
|
|
||||||
Preference keyPref = findPreference("openpgp_key_id_pref");
|
|
||||||
ArrayList<String> selectedKeys = new ArrayList<>(sharedPreferences.getStringSet("openpgp_key_ids_set", new HashSet<String>()));
|
|
||||||
if (selectedKeys.isEmpty()) {
|
|
||||||
keyPref.setSummary("No key selected");
|
|
||||||
} else {
|
|
||||||
StringBuilder summaryBuilder = new StringBuilder();
|
|
||||||
for (int i = 0; i < selectedKeys.size(); ++i) {
|
|
||||||
String s = selectedKeys.get(i);
|
|
||||||
summaryBuilder.append(OpenPgpUtils.convertKeyIdToHex(Long.valueOf(s)));
|
|
||||||
if (i < selectedKeys.size() - 1)
|
|
||||||
summaryBuilder.append("; ");
|
|
||||||
}
|
|
||||||
keyPref.setSummary(summaryBuilder.toString());
|
|
||||||
}
|
|
||||||
|
|
||||||
// see if the autofill service is enabled and check the preference accordingly
|
|
||||||
((CheckBoxPreference) findPreference("autofill_enable"))
|
|
||||||
.setChecked(((UserPreference) getActivity()).isServiceEnabled());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onCreate(Bundle savedInstanceState) {
|
|
||||||
final SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this.getApplicationContext());
|
|
||||||
super.onCreate(savedInstanceState);
|
|
||||||
if (getIntent() != null) {
|
|
||||||
if (getIntent().getStringExtra("operation") != null) {
|
|
||||||
switch (getIntent().getStringExtra("operation")) {
|
|
||||||
case "get_ssh_key":
|
|
||||||
getSshKeyWithPermissions(sharedPreferences.getBoolean("use_android_file_picker", false));
|
|
||||||
break;
|
|
||||||
case "make_ssh_key":
|
|
||||||
makeSshKey(false);
|
|
||||||
break;
|
|
||||||
case "git_external":
|
|
||||||
selectExternalGitRepository();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
prefsFragment = new PrefsFragment();
|
|
||||||
|
|
||||||
getFragmentManager().beginTransaction().replace(android.R.id.content, prefsFragment).commit();
|
|
||||||
|
|
||||||
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void selectExternalGitRepository() {
|
|
||||||
final Activity activity = this;
|
|
||||||
new AlertDialog.Builder(this).
|
|
||||||
setTitle("Choose where to store the passwords").
|
|
||||||
setMessage("You must select a directory where to store your passwords. If you want " +
|
|
||||||
"to store your passwords within the hidden storage of the application, " +
|
|
||||||
"cancel this dialog and disable the \"External Repository\" option.").
|
|
||||||
setPositiveButton(R.string.dialog_ok, new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
|
||||||
// This always works
|
|
||||||
Intent i = new Intent(activity.getApplicationContext(), FilePickerActivity.class);
|
|
||||||
// This works if you defined the intent filter
|
|
||||||
// Intent i = new Intent(Intent.ACTION_GET_CONTENT);
|
|
||||||
|
|
||||||
// Set these depending on your use case. These are the defaults.
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_ALLOW_MULTIPLE, false);
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_ALLOW_CREATE_DIR, true);
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_MODE, FilePickerActivity.MODE_DIR);
|
|
||||||
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_START_PATH, Environment.getExternalStorageDirectory().getPath());
|
|
||||||
|
|
||||||
startActivityForResult(i, SELECT_GIT_DIRECTORY);
|
|
||||||
}
|
|
||||||
}).
|
|
||||||
setNegativeButton(R.string.dialog_cancel, null).show();
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
|
||||||
// Handle action bar item clicks here. The action bar will
|
|
||||||
// automatically handle clicks on the Home/Up button, so long
|
|
||||||
// as you specify a parent activity in AndroidManifest.xml.
|
|
||||||
int id = item.getItemId();
|
|
||||||
switch (id) {
|
|
||||||
case android.R.id.home:
|
|
||||||
setResult(RESULT_OK);
|
|
||||||
finish();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return super.onOptionsItemSelected(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Opens a file explorer to import the private key
|
|
||||||
*/
|
|
||||||
private void getSshKeyWithPermissions(boolean useDefaultPicker) {
|
|
||||||
final Activity activity = this;
|
|
||||||
if (ContextCompat.checkSelfPermission(activity, Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
|
|
||||||
if (ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.READ_EXTERNAL_STORAGE)) {
|
|
||||||
Snackbar snack = Snackbar.make(prefsFragment.getView(),
|
|
||||||
"We need access to the sd-card to import the ssh-key",
|
|
||||||
Snackbar.LENGTH_INDEFINITE)
|
|
||||||
.setAction(R.string.dialog_ok, new View.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(View view) {
|
|
||||||
ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, REQUEST_EXTERNAL_STORAGE);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
snack.show();
|
|
||||||
View view = snack.getView();
|
|
||||||
TextView tv = (TextView) view.findViewById(android.support.design.R.id.snackbar_text);
|
|
||||||
tv.setTextColor(Color.WHITE);
|
|
||||||
tv.setMaxLines(10);
|
|
||||||
} else {
|
|
||||||
// No explanation needed, we can request the permission.
|
|
||||||
ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, REQUEST_EXTERNAL_STORAGE);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
getSshKey(useDefaultPicker);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Opens a file explorer to import the private key
|
|
||||||
*/
|
|
||||||
private void getSshKey(boolean useDefaultPicker) {
|
|
||||||
if (useDefaultPicker) {
|
|
||||||
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
|
|
||||||
intent.setType("*/*");
|
|
||||||
startActivityForResult(intent, IMPORT_SSH_KEY);
|
|
||||||
} else {
|
|
||||||
// This always works
|
|
||||||
Intent i = new Intent(getApplicationContext(), FilePickerActivity.class);
|
|
||||||
// This works if you defined the intent filter
|
|
||||||
// Intent i = new Intent(Intent.ACTION_GET_CONTENT);
|
|
||||||
|
|
||||||
// Set these depending on your use case. These are the defaults.
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_ALLOW_MULTIPLE, false);
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_ALLOW_CREATE_DIR, false);
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_MODE, FilePickerActivity.MODE_FILE);
|
|
||||||
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_START_PATH, Environment.getExternalStorageDirectory().getPath());
|
|
||||||
|
|
||||||
startActivityForResult(i, IMPORT_SSH_KEY);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void exportPasswordsWithPermissions() {
|
|
||||||
final Activity activity = this;
|
|
||||||
if (ContextCompat.checkSelfPermission(activity, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
|
|
||||||
if (ActivityCompat.shouldShowRequestPermissionRationale(activity, Manifest.permission.WRITE_EXTERNAL_STORAGE)) {
|
|
||||||
Snackbar snack = Snackbar.make(prefsFragment.getView(),
|
|
||||||
"We need access to the sd-card to export the passwords",
|
|
||||||
Snackbar.LENGTH_INDEFINITE)
|
|
||||||
.setAction(R.string.dialog_ok, new View.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(View view) {
|
|
||||||
ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUEST_EXTERNAL_STORAGE);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
snack.show();
|
|
||||||
View view = snack.getView();
|
|
||||||
TextView tv = (TextView) view.findViewById(android.support.design.R.id.snackbar_text);
|
|
||||||
tv.setTextColor(Color.WHITE);
|
|
||||||
tv.setMaxLines(10);
|
|
||||||
} else {
|
|
||||||
// No explanation needed, we can request the permission.
|
|
||||||
ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUEST_EXTERNAL_STORAGE);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Intent i = new Intent(getApplicationContext(), FilePickerActivity.class);
|
|
||||||
|
|
||||||
// Set these depending on your use case. These are the defaults.
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_ALLOW_MULTIPLE, false);
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_ALLOW_CREATE_DIR, true);
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_MODE, FilePickerActivity.MODE_DIR);
|
|
||||||
|
|
||||||
i.putExtra(FilePickerActivity.EXTRA_START_PATH, Environment.getExternalStorageDirectory().getPath());
|
|
||||||
|
|
||||||
startActivityForResult(i, EXPORT_PASSWORDS);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Opens a key generator to generate a public/private key pair
|
|
||||||
*/
|
|
||||||
private void makeSshKey(boolean fromPreferences) {
|
|
||||||
Intent intent = new Intent(getApplicationContext(), SshKeyGen.class);
|
|
||||||
startActivity(intent);
|
|
||||||
if (!fromPreferences) {
|
|
||||||
setResult(RESULT_OK);
|
|
||||||
finish();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void copySshKey(Uri uri) throws IOException {
|
|
||||||
InputStream sshKey = this.getContentResolver().openInputStream(uri);
|
|
||||||
byte[] privateKey = IOUtils.toByteArray(sshKey);
|
|
||||||
FileUtils.writeByteArrayToFile(new File(getFilesDir() + "/.ssh_key"), privateKey);
|
|
||||||
sshKey.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns whether the autofill service is enabled
|
|
||||||
private boolean isServiceEnabled() {
|
|
||||||
AccessibilityManager am = (AccessibilityManager) this
|
|
||||||
.getSystemService(Context.ACCESSIBILITY_SERVICE);
|
|
||||||
List<AccessibilityServiceInfo> runningServices = am
|
|
||||||
.getEnabledAccessibilityServiceList(AccessibilityServiceInfo.FEEDBACK_GENERIC);
|
|
||||||
for (AccessibilityServiceInfo service : runningServices) {
|
|
||||||
if ("com.zeapo.pwdstore/.autofill.AutofillService".equals(service.getId())) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
protected void onActivityResult(int requestCode, int resultCode,
|
|
||||||
Intent data) {
|
|
||||||
if (resultCode == RESULT_OK) {
|
|
||||||
switch (requestCode) {
|
|
||||||
case IMPORT_SSH_KEY: {
|
|
||||||
try {
|
|
||||||
final Uri uri = data.getData();
|
|
||||||
|
|
||||||
if (uri == null) {
|
|
||||||
throw new IOException("Unable to open file");
|
|
||||||
}
|
|
||||||
copySshKey(uri);
|
|
||||||
Toast.makeText(this, this.getResources().getString(R.string.ssh_key_success_dialog_title), Toast.LENGTH_LONG).show();
|
|
||||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
|
|
||||||
SharedPreferences.Editor editor = prefs.edit();
|
|
||||||
editor.putBoolean("use_generated_key", false);
|
|
||||||
editor.apply();
|
|
||||||
|
|
||||||
//delete the public key from generation
|
|
||||||
File file = new File(getFilesDir() + "/.ssh_key.pub");
|
|
||||||
file.delete();
|
|
||||||
|
|
||||||
setResult(RESULT_OK);
|
|
||||||
finish();
|
|
||||||
} catch (IOException e) {
|
|
||||||
new AlertDialog.Builder(this).
|
|
||||||
setTitle(this.getResources().getString(R.string.ssh_key_error_dialog_title)).
|
|
||||||
setMessage(this.getResources().getString(R.string.ssh_key_error_dialog_text) + e.getMessage()).
|
|
||||||
setPositiveButton(this.getResources().getString(R.string.dialog_ok), new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialogInterface, int i) {
|
|
||||||
// pass
|
|
||||||
}
|
|
||||||
}).show();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case EDIT_GIT_INFO: {
|
|
||||||
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case SELECT_GIT_DIRECTORY: {
|
|
||||||
final Uri uri = data.getData();
|
|
||||||
|
|
||||||
if (uri.getPath().equals(Environment.getExternalStorageDirectory().getPath())) {
|
|
||||||
// the user wants to use the root of the sdcard as a store...
|
|
||||||
new AlertDialog.Builder(this).
|
|
||||||
setTitle("SD-Card root selected").
|
|
||||||
setMessage("You have selected the root of your sdcard for the store. " +
|
|
||||||
"This is extremely dangerous and you will lose your data " +
|
|
||||||
"as its content will, eventually, be deleted").
|
|
||||||
setPositiveButton("Remove everything", new DialogInterface.OnClickListener() {
|
|
||||||
@Override
|
|
||||||
public void onClick(DialogInterface dialog, int which) {
|
|
||||||
PreferenceManager.getDefaultSharedPreferences(getApplicationContext())
|
|
||||||
.edit()
|
|
||||||
.putString("git_external_repo", uri.getPath())
|
|
||||||
.apply();
|
|
||||||
}
|
|
||||||
}).
|
|
||||||
setNegativeButton(R.string.dialog_cancel, null).show();
|
|
||||||
} else {
|
|
||||||
PreferenceManager.getDefaultSharedPreferences(getApplicationContext())
|
|
||||||
.edit()
|
|
||||||
.putString("git_external_repo", uri.getPath())
|
|
||||||
.apply();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case EXPORT_PASSWORDS: {
|
|
||||||
final Uri uri = data.getData();
|
|
||||||
final File repositoryDirectory = PasswordRepository.getRepositoryDirectory(getApplicationContext());
|
|
||||||
SimpleDateFormat fmtOut = new SimpleDateFormat("yyyy_MM_dd_HH_mm_ss", Locale.US);
|
|
||||||
Date date = new Date();
|
|
||||||
String password_now = "/password_store_" + fmtOut.format(date);
|
|
||||||
final File targetDirectory = new File(uri.getPath() + password_now);
|
|
||||||
if (repositoryDirectory != null) {
|
|
||||||
try {
|
|
||||||
FileUtils.copyDirectory(repositoryDirectory, targetDirectory, true);
|
|
||||||
} catch (IOException e) {
|
|
||||||
Log.d("PWD_EXPORT", "Exception happened : " + e.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onRequestPermissionsResult(int requestCode, @NonNull String permissions[], @NonNull int[] grantResults) {
|
|
||||||
final SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this.getApplicationContext());
|
|
||||||
switch (requestCode) {
|
|
||||||
case REQUEST_EXTERNAL_STORAGE: {
|
|
||||||
// If request is cancelled, the result arrays are empty.
|
|
||||||
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
|
|
||||||
getSshKey(sharedPreferences.getBoolean("use_android_file_picker", false));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
460
app/src/main/java/com/zeapo/pwdstore/UserPreference.kt
Normal file
460
app/src/main/java/com/zeapo/pwdstore/UserPreference.kt
Normal file
|
@ -0,0 +1,460 @@
|
||||||
|
package com.zeapo.pwdstore
|
||||||
|
|
||||||
|
import android.Manifest
|
||||||
|
import android.accessibilityservice.AccessibilityServiceInfo
|
||||||
|
import android.app.Activity
|
||||||
|
import android.content.Context
|
||||||
|
import android.content.Intent
|
||||||
|
import android.content.pm.PackageManager
|
||||||
|
import android.graphics.Color
|
||||||
|
import android.net.Uri
|
||||||
|
import android.os.Bundle
|
||||||
|
import android.os.Environment
|
||||||
|
import android.preference.CheckBoxPreference
|
||||||
|
import android.preference.Preference
|
||||||
|
import android.preference.PreferenceFragment
|
||||||
|
import android.preference.PreferenceManager
|
||||||
|
import android.provider.Settings
|
||||||
|
import android.support.design.widget.Snackbar
|
||||||
|
import android.support.v4.app.ActivityCompat
|
||||||
|
import android.support.v4.content.ContextCompat
|
||||||
|
import android.support.v7.app.AlertDialog
|
||||||
|
import android.support.v7.app.AppCompatActivity
|
||||||
|
import android.util.Log
|
||||||
|
import android.view.MenuItem
|
||||||
|
import android.view.accessibility.AccessibilityManager
|
||||||
|
import android.widget.TextView
|
||||||
|
import android.widget.Toast
|
||||||
|
import com.nononsenseapps.filepicker.FilePickerActivity
|
||||||
|
import com.zeapo.pwdstore.autofill.AutofillPreferenceActivity
|
||||||
|
import com.zeapo.pwdstore.crypto.PgpHandler
|
||||||
|
import com.zeapo.pwdstore.git.GitActivity
|
||||||
|
import com.zeapo.pwdstore.utils.PasswordRepository
|
||||||
|
import org.apache.commons.io.FileUtils
|
||||||
|
import org.apache.commons.io.IOUtils
|
||||||
|
import org.openintents.openpgp.util.OpenPgpUtils
|
||||||
|
import java.io.File
|
||||||
|
import java.io.IOException
|
||||||
|
import java.text.SimpleDateFormat
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
|
class UserPreference : AppCompatActivity() {
|
||||||
|
private lateinit var prefsFragment: PrefsFragment
|
||||||
|
|
||||||
|
class PrefsFragment : PreferenceFragment() {
|
||||||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
|
super.onCreate(savedInstanceState)
|
||||||
|
val callingActivity = activity as UserPreference
|
||||||
|
val sharedPreferences = preferenceManager.sharedPreferences
|
||||||
|
|
||||||
|
addPreferencesFromResource(R.xml.preference)
|
||||||
|
|
||||||
|
findPreference("app_version").summary = "Version: ${BuildConfig.VERSION_NAME}"
|
||||||
|
|
||||||
|
findPreference("openpgp_key_id_pref").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
val intent = Intent(callingActivity, PgpHandler::class.java)
|
||||||
|
intent.putExtra("Operation", "GET_KEY_ID")
|
||||||
|
startActivityForResult(intent, IMPORT_PGP_KEY)
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
findPreference("ssh_key").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
callingActivity.getSshKeyWithPermissions(sharedPreferences.getBoolean("use_android_file_picker", false))
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
findPreference("ssh_keygen").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
callingActivity.makeSshKey(true)
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
findPreference("ssh_see_key").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
val df = SshKeyGen.ShowSshKeyFragment()
|
||||||
|
df.show(fragmentManager, "public_key")
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
findPreference("git_server_info").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
val intent = Intent(callingActivity, GitActivity::class.java)
|
||||||
|
intent.putExtra("Operation", GitActivity.EDIT_SERVER)
|
||||||
|
startActivityForResult(intent, EDIT_GIT_INFO)
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
findPreference("git_config").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
val intent = Intent(callingActivity, GitActivity::class.java)
|
||||||
|
intent.putExtra("Operation", GitActivity.EDIT_GIT_CONFIG)
|
||||||
|
startActivityForResult(intent, EDIT_GIT_CONFIG)
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
findPreference("git_delete_repo").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
val repoDir = PasswordRepository.getRepositoryDirectory(callingActivity.applicationContext)
|
||||||
|
AlertDialog.Builder(callingActivity)
|
||||||
|
.setTitle(R.string.pref_dialog_delete_title)
|
||||||
|
.setMessage("${resources.getString(R.string.dialog_delete_msg)} \n $repoDir")
|
||||||
|
.setCancelable(false)
|
||||||
|
.setPositiveButton(R.string.dialog_delete) { dialogInterface, _ ->
|
||||||
|
try {
|
||||||
|
FileUtils.cleanDirectory(PasswordRepository.getRepositoryDirectory(callingActivity.applicationContext))
|
||||||
|
PasswordRepository.closeRepository()
|
||||||
|
} catch (e: Exception) {
|
||||||
|
//TODO Handle the diffent cases of exceptions
|
||||||
|
}
|
||||||
|
|
||||||
|
sharedPreferences.edit().putBoolean("repository_initialized", false).apply()
|
||||||
|
dialogInterface.cancel()
|
||||||
|
callingActivity.finish()
|
||||||
|
}.setNegativeButton(R.string.dialog_do_not_delete) { dialogInterface, _ -> run { dialogInterface.cancel() } }
|
||||||
|
.show()
|
||||||
|
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
val externalRepo = findPreference("pref_select_external")
|
||||||
|
externalRepo.summary = sharedPreferences.getString("git_external_repo", callingActivity.getString(R.string.no_repo_selected))
|
||||||
|
externalRepo.onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
callingActivity.selectExternalGitRepository()
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
val resetRepo = Preference.OnPreferenceChangeListener { _, o ->
|
||||||
|
findPreference("git_delete_repo").isEnabled = !(o as Boolean)
|
||||||
|
PasswordRepository.closeRepository()
|
||||||
|
sharedPreferences.edit().putBoolean("repo_changed", true).apply()
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
findPreference("pref_select_external").onPreferenceChangeListener = resetRepo
|
||||||
|
findPreference("git_external").onPreferenceChangeListener = resetRepo
|
||||||
|
|
||||||
|
findPreference("autofill_apps").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
val intent = Intent(callingActivity, AutofillPreferenceActivity::class.java)
|
||||||
|
startActivity(intent)
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
findPreference("autofill_enable").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
AlertDialog.Builder(callingActivity).setTitle(R.string.pref_autofill_enable_title).setView(R.layout.autofill_instructions).setPositiveButton(R.string.dialog_ok) { _, _ ->
|
||||||
|
val intent = Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS)
|
||||||
|
startActivity(intent)
|
||||||
|
}.setNegativeButton(R.string.dialog_cancel, null).setOnDismissListener { (findPreference("autofill_enable") as CheckBoxPreference).isChecked = (activity as UserPreference).isServiceEnabled }.show()
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
findPreference("export_passwords").onPreferenceClickListener = Preference.OnPreferenceClickListener {
|
||||||
|
callingActivity.exportPasswordsWithPermissions()
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onStart() {
|
||||||
|
super.onStart()
|
||||||
|
val sharedPreferences = preferenceManager.sharedPreferences
|
||||||
|
findPreference("pref_select_external").summary = preferenceManager.sharedPreferences.getString("git_external_repo", getString(R.string.no_repo_selected))
|
||||||
|
findPreference("ssh_see_key").isEnabled = sharedPreferences.getBoolean("use_generated_key", false)
|
||||||
|
findPreference("git_delete_repo").isEnabled = !sharedPreferences.getBoolean("git_external", false)
|
||||||
|
val keyPref = findPreference("openpgp_key_id_pref")
|
||||||
|
val selectedKeys: Array<String> = ArrayList<String>(sharedPreferences.getStringSet("openpgp_key_ids_set", HashSet<String>())).toTypedArray()
|
||||||
|
if (selectedKeys.isEmpty()) {
|
||||||
|
keyPref.summary = "No key selected"
|
||||||
|
} else {
|
||||||
|
keyPref.summary = selectedKeys.joinToString(separator = ";") {
|
||||||
|
s ->
|
||||||
|
OpenPgpUtils.convertKeyIdToHex(java.lang.Long.valueOf(s))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// see if the autofill service is enabled and check the preference accordingly
|
||||||
|
(findPreference("autofill_enable") as CheckBoxPreference).isChecked = (activity as UserPreference).isServiceEnabled
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
|
val sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this.applicationContext)
|
||||||
|
super.onCreate(savedInstanceState)
|
||||||
|
when (intent?.getStringExtra("operation")) {
|
||||||
|
"get_ssh_key" -> getSshKeyWithPermissions(sharedPreferences.getBoolean("use_android_file_picker", false))
|
||||||
|
"make_ssh_key" -> makeSshKey(false)
|
||||||
|
"git_external" -> selectExternalGitRepository()
|
||||||
|
}
|
||||||
|
prefsFragment = PrefsFragment()
|
||||||
|
|
||||||
|
fragmentManager.beginTransaction().replace(android.R.id.content, prefsFragment).commit()
|
||||||
|
|
||||||
|
supportActionBar?.setDisplayHomeAsUpEnabled(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun selectExternalGitRepository() {
|
||||||
|
val activity = this
|
||||||
|
AlertDialog.Builder(this)
|
||||||
|
.setTitle("Choose where to store the passwords")
|
||||||
|
.setMessage("You must select a directory where to store your passwords. If you want " +
|
||||||
|
"to store your passwords within the hidden storage of the application, " +
|
||||||
|
"cancel this dialog and disable the \"External Repository\" option.")
|
||||||
|
.setPositiveButton(R.string.dialog_ok) { _, _ ->
|
||||||
|
// This always works
|
||||||
|
val i = Intent(activity.applicationContext, FilePickerActivity::class.java)
|
||||||
|
// This works if you defined the intent filter
|
||||||
|
// Intent i = new Intent(Intent.ACTION_GET_CONTENT);
|
||||||
|
|
||||||
|
// Set these depending on your use case. These are the defaults.
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_ALLOW_MULTIPLE, false)
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_ALLOW_CREATE_DIR, true)
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_MODE, FilePickerActivity.MODE_DIR)
|
||||||
|
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_START_PATH, Environment.getExternalStorageDirectory().path)
|
||||||
|
|
||||||
|
startActivityForResult(i, SELECT_GIT_DIRECTORY)
|
||||||
|
}.setNegativeButton(R.string.dialog_cancel, null).show()
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onOptionsItemSelected(item: MenuItem): Boolean {
|
||||||
|
// Handle action bar item clicks here. The action bar will
|
||||||
|
// automatically handle clicks on the Home/Up button, so long
|
||||||
|
// as you specify a parent activity in AndroidManifest.xml.
|
||||||
|
val id = item.itemId
|
||||||
|
when (id) {
|
||||||
|
android.R.id.home -> {
|
||||||
|
setResult(Activity.RESULT_OK)
|
||||||
|
finish()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return super.onOptionsItemSelected(item)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Opens a file explorer to import the private key
|
||||||
|
*/
|
||||||
|
fun getSshKeyWithPermissions(useDefaultPicker: Boolean) = runWithPermissions(
|
||||||
|
requestedPermission = Manifest.permission.READ_EXTERNAL_STORAGE,
|
||||||
|
requestCode = REQUEST_EXTERNAL_STORAGE_SSH_KEY,
|
||||||
|
reason = "We need access to the sd-card to import the ssh-key"
|
||||||
|
) {
|
||||||
|
getSshKey(useDefaultPicker)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Opens a file explorer to import the private key
|
||||||
|
*/
|
||||||
|
fun getSshKey(useDefaultPicker: Boolean) {
|
||||||
|
if (useDefaultPicker) {
|
||||||
|
val intent = Intent(Intent.ACTION_GET_CONTENT)
|
||||||
|
intent.type = "*/*"
|
||||||
|
startActivityForResult(intent, IMPORT_SSH_KEY)
|
||||||
|
} else {
|
||||||
|
// This always works
|
||||||
|
val i = Intent(applicationContext, FilePickerActivity::class.java)
|
||||||
|
// This works if you defined the intent filter
|
||||||
|
// Intent i = new Intent(Intent.ACTION_GET_CONTENT);
|
||||||
|
|
||||||
|
// Set these depending on your use case. These are the defaults.
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_ALLOW_MULTIPLE, false)
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_ALLOW_CREATE_DIR, false)
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_MODE, FilePickerActivity.MODE_FILE)
|
||||||
|
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_START_PATH, Environment.getExternalStorageDirectory().path)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Run a function after checking that the permissions have been requested
|
||||||
|
*
|
||||||
|
* @param requestedPermission The permission to request
|
||||||
|
* @param requestCode The code passed to onRequestPermissionsResult
|
||||||
|
* @param reason The text to be shown to the user to explain why we're requesting this permission
|
||||||
|
* @param body The function to run
|
||||||
|
*/
|
||||||
|
fun runWithPermissions(requestedPermission: String, requestCode: Int, reason: String, body: () -> Unit): Unit {
|
||||||
|
if (ContextCompat.checkSelfPermission(this, requestedPermission) != PackageManager.PERMISSION_GRANTED) {
|
||||||
|
if (ActivityCompat.shouldShowRequestPermissionRationale(this, requestedPermission)) {
|
||||||
|
val snack = Snackbar.make(prefsFragment.view,
|
||||||
|
reason,
|
||||||
|
Snackbar.LENGTH_INDEFINITE)
|
||||||
|
.setAction(R.string.dialog_ok) {
|
||||||
|
ActivityCompat.requestPermissions(this, arrayOf(requestedPermission), requestCode)
|
||||||
|
}
|
||||||
|
snack.show()
|
||||||
|
val view = snack.view
|
||||||
|
val tv = view.findViewById(android.support.design.R.id.snackbar_text) as TextView
|
||||||
|
tv.setTextColor(Color.WHITE)
|
||||||
|
tv.maxLines = 10
|
||||||
|
} else {
|
||||||
|
// No explanation needed, we can request the permission.
|
||||||
|
ActivityCompat.requestPermissions(this, arrayOf(requestedPermission), requestCode)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
body()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exports the passwords after requesting permissions
|
||||||
|
*/
|
||||||
|
fun exportPasswordsWithPermissions() = runWithPermissions(
|
||||||
|
requestedPermission = Manifest.permission.WRITE_EXTERNAL_STORAGE,
|
||||||
|
requestCode = REQUEST_EXTERNAL_STORAGE_SSH_KEY,
|
||||||
|
reason = "We need access to the sd-card to export the passwords"
|
||||||
|
) {
|
||||||
|
exportPasswords()
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Exports the passwords
|
||||||
|
*/
|
||||||
|
fun exportPasswords(): Unit {
|
||||||
|
val i = Intent(applicationContext, FilePickerActivity::class.java)
|
||||||
|
|
||||||
|
// Set these depending on your use case. These are the defaults.
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_ALLOW_MULTIPLE, false)
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_ALLOW_CREATE_DIR, true)
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_MODE, FilePickerActivity.MODE_DIR)
|
||||||
|
|
||||||
|
i.putExtra(FilePickerActivity.EXTRA_START_PATH, Environment.getExternalStorageDirectory().path)
|
||||||
|
|
||||||
|
startActivityForResult(i, EXPORT_PASSWORDS)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Opens a key generator to generate a public/private key pair
|
||||||
|
*/
|
||||||
|
fun makeSshKey(fromPreferences: Boolean) {
|
||||||
|
val intent = Intent(applicationContext, SshKeyGen::class.java)
|
||||||
|
startActivity(intent)
|
||||||
|
if (!fromPreferences) {
|
||||||
|
setResult(Activity.RESULT_OK)
|
||||||
|
finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Throws(IOException::class)
|
||||||
|
private fun copySshKey(uri: Uri) {
|
||||||
|
val sshKey = this.contentResolver.openInputStream(uri)
|
||||||
|
val privateKey = IOUtils.toByteArray(sshKey!!)
|
||||||
|
FileUtils.writeByteArrayToFile(File(filesDir.toString() + "/.ssh_key"), privateKey)
|
||||||
|
sshKey.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns whether the autofill service is enabled
|
||||||
|
private val isServiceEnabled: Boolean
|
||||||
|
get() {
|
||||||
|
val am = this
|
||||||
|
.getSystemService(Context.ACCESSIBILITY_SERVICE) as AccessibilityManager
|
||||||
|
val runningServices = am
|
||||||
|
.getEnabledAccessibilityServiceList(AccessibilityServiceInfo.FEEDBACK_GENERIC)
|
||||||
|
return runningServices.any { "com.zeapo.pwdstore/.autofill.AutofillService" == it.id }
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
override fun onActivityResult(requestCode: Int, resultCode: Int,
|
||||||
|
data: Intent?) {
|
||||||
|
if (resultCode == Activity.RESULT_OK) {
|
||||||
|
if (data == null) {
|
||||||
|
setResult(Activity.RESULT_CANCELED)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
when (requestCode) {
|
||||||
|
IMPORT_SSH_KEY -> {
|
||||||
|
try {
|
||||||
|
val uri: Uri = data.data ?: throw IOException("Unable to open file")
|
||||||
|
|
||||||
|
copySshKey(uri)
|
||||||
|
Toast.makeText(this, this.resources.getString(R.string.ssh_key_success_dialog_title), Toast.LENGTH_LONG).show()
|
||||||
|
val prefs = PreferenceManager.getDefaultSharedPreferences(applicationContext)
|
||||||
|
|
||||||
|
prefs.edit().putBoolean("use_generated_key", false).apply()
|
||||||
|
|
||||||
|
//delete the public key from generation
|
||||||
|
val file = File(filesDir.toString() + "/.ssh_key.pub")
|
||||||
|
file.delete()
|
||||||
|
setResult(Activity.RESULT_OK)
|
||||||
|
|
||||||
|
finish()
|
||||||
|
} catch (e: IOException) {
|
||||||
|
AlertDialog.Builder(this).setTitle(this.resources.getString(R.string.ssh_key_error_dialog_title)).setMessage(this.resources.getString(R.string.ssh_key_error_dialog_text) + e.message).setPositiveButton(this.resources.getString(R.string.dialog_ok)) { _, _ ->
|
||||||
|
// pass
|
||||||
|
}.show()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
EDIT_GIT_INFO -> {
|
||||||
|
|
||||||
|
}
|
||||||
|
SELECT_GIT_DIRECTORY -> {
|
||||||
|
val uri = data.data
|
||||||
|
|
||||||
|
if (uri.path == Environment.getExternalStorageDirectory().path) {
|
||||||
|
// the user wants to use the root of the sdcard as a store...
|
||||||
|
AlertDialog.Builder(this)
|
||||||
|
.setTitle("SD-Card root selected")
|
||||||
|
.setMessage("You have selected the root of your sdcard for the store. " +
|
||||||
|
"This is extremely dangerous and you will lose your data " +
|
||||||
|
"as its content will, eventually, be deleted")
|
||||||
|
.setPositiveButton("Remove everything") { _, _ ->
|
||||||
|
PreferenceManager.getDefaultSharedPreferences(applicationContext)
|
||||||
|
.edit()
|
||||||
|
.putString("git_external_repo", uri.path)
|
||||||
|
.apply()
|
||||||
|
}.setNegativeButton(R.string.dialog_cancel, null).show()
|
||||||
|
} else {
|
||||||
|
PreferenceManager.getDefaultSharedPreferences(applicationContext)
|
||||||
|
.edit()
|
||||||
|
.putString("git_external_repo", uri.path)
|
||||||
|
.apply()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
EXPORT_PASSWORDS -> {
|
||||||
|
val uri = data.data
|
||||||
|
val repositoryDirectory = PasswordRepository.getRepositoryDirectory(applicationContext)
|
||||||
|
val fmtOut = SimpleDateFormat("yyyy_MM_dd_HH_mm_ss", Locale.US)
|
||||||
|
val date = Date()
|
||||||
|
val password_now = "/password_store_" + fmtOut.format(date)
|
||||||
|
val targetDirectory = File(uri.path + password_now)
|
||||||
|
if (repositoryDirectory != null) {
|
||||||
|
try {
|
||||||
|
FileUtils.copyDirectory(repositoryDirectory, targetDirectory, true)
|
||||||
|
} catch (e: IOException) {
|
||||||
|
Log.d("PWD_EXPORT", "Exception happened : " + e.message)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<String>, grantResults: IntArray) {
|
||||||
|
val sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this.applicationContext)
|
||||||
|
when (requestCode) {
|
||||||
|
REQUEST_EXTERNAL_STORAGE_SSH_KEY -> {
|
||||||
|
// If request is cancelled, the result arrays are empty.
|
||||||
|
if (grantResults.isNotEmpty() && PackageManager.PERMISSION_GRANTED in grantResults) {
|
||||||
|
getSshKey(sharedPreferences.getBoolean("use_android_file_picker", false))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
REQUEST_EXTERNAL_STORAGE_EXPORT_PWD -> {
|
||||||
|
if (grantResults.isNotEmpty() && PackageManager.PERMISSION_GRANTED in grantResults) {
|
||||||
|
exportPasswords()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
private val IMPORT_SSH_KEY = 1
|
||||||
|
private val IMPORT_PGP_KEY = 2
|
||||||
|
private val EDIT_GIT_INFO = 3
|
||||||
|
private val SELECT_GIT_DIRECTORY = 4
|
||||||
|
private val EXPORT_PASSWORDS = 5
|
||||||
|
private val EDIT_GIT_CONFIG = 6
|
||||||
|
private val REQUEST_EXTERNAL_STORAGE_SSH_KEY = 50
|
||||||
|
private val REQUEST_EXTERNAL_STORAGE_EXPORT_PWD = 51
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue