aboutsummaryrefslogtreecommitdiffstats
path: root/OpenKeychain
diff options
context:
space:
mode:
authorVincent Breitmoser <valodim@mugenguild.com>2015-03-02 18:34:51 +0100
committerVincent Breitmoser <valodim@mugenguild.com>2015-03-02 18:34:51 +0100
commite6af789bf50f9636d3455c4bd1b662ab5d04e3e2 (patch)
tree35e7560cd785aa20c1fa814332d16bd094921282 /OpenKeychain
parent5e53a5417c11f54f437d9f220d3e2e8579b8bf1b (diff)
parent145e793db1d732bdd230be92750123cd20c9502d (diff)
downloadopen-keychain-e6af789bf50f9636d3455c4bd1b662ab5d04e3e2.tar.gz
open-keychain-e6af789bf50f9636d3455c4bd1b662ab5d04e3e2.tar.bz2
open-keychain-e6af789bf50f9636d3455c4bd1b662ab5d04e3e2.zip
Merge branch 'development' into linked-identities
Conflicts: OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserAttributesAdapter.java
Diffstat (limited to 'OpenKeychain')
-rw-r--r--OpenKeychain/build.gradle41
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/operations/results/LinkedVerifyResult.java14
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyActivity.java6
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyFragment.java28
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserAttributesAdapter.java246
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserIdsAdapter.java193
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserIdsSelectableAdapter.java88
-rw-r--r--OpenKeychain/src/main/java/org/sufficientlysecure/keychain/util/ContactHelper.java27
-rw-r--r--OpenKeychain/src/main/res/layout/view_key_adv_user_id_item.xml8
-rw-r--r--OpenKeychain/src/main/res/layout/view_key_selectable_user_id_item.xml18
-rw-r--r--OpenKeychain/src/main/res/values/colors.xml2
-rw-r--r--OpenKeychain/src/main/res/values/strings.xml3
12 files changed, 382 insertions, 292 deletions
diff --git a/OpenKeychain/build.gradle b/OpenKeychain/build.gradle
index 7fa139c06..4567f54dc 100644
--- a/OpenKeychain/build.gradle
+++ b/OpenKeychain/build.gradle
@@ -1,4 +1,5 @@
apply plugin: 'com.android.application'
+apply plugin: 'witness'
dependencies {
// NOTE: Always use fixed version codes not dynamic ones, e.g. 0.7.3 instead of 0.7.+, see README for more information
@@ -15,13 +16,12 @@ dependencies {
compile project(':extern:spongycastle:pg')
compile project(':extern:spongycastle:pkix')
compile project(':extern:spongycastle:prov')
- compile project(':extern:SuperToasts:supertoasts')
compile project(':extern:minidns')
compile project(':extern:KeybaseLib:Lib')
compile project(':extern:TokenAutoComplete:library')
compile project(':extern:safeslinger-exchange')
- // TODO: Pin!
+ // NOTE: libraries are pinned to a specific build, see below
compile 'com.eftimoff:android-patternview:1.0.0@aar'
compile 'com.journeyapps:zxing-android-embedded:2.0.1@aar'
compile 'com.journeyapps:zxing-android-integration:2.0.1@aar'
@@ -32,9 +32,42 @@ dependencies {
compile 'com.getbase:floatingactionbutton:1.8.0'
}
+// Output of ./gradlew -q calculateChecksums
+// Comment out the libs referenced as git submodules!
+dependencyVerification {
+ verify = [
+ 'com.android.support:support-v4:703572d3015a088cc5604b7e38885af3d307c829d0c5ceaf8654ff41c71cd160',
+ 'com.android.support:appcompat-v7:5dbeb5316d0a6027d646ae552804c3baa5e3bd53f7f33db50904d51505c8a0e5',
+ 'com.android.support:recyclerview-v7:e525ad3f33c84bb12b73d2dc975b55364a53f0f2d0697e043efba59ba73e22d2',
+ 'com.android.support:cardview-v7:45c48c2ab056bc7a8573970b10f8902742c5d443f180dae43c56557397ac39af',
+ //'OpenKeychain.extern:openpgp-api-lib:b17bb282321351e4b00b4cd6422a57aadc13decae264019a88707bcb556439ea',
+ //'OpenKeychain.extern:openkeychain-api-lib:5f95f01c066069d4bde68992fd8da5faac21510d009b1fdae7a2e28e43e82cf4',
+ //'OpenKeychain.extern:html-textview:b58e343cf4c145e91f888806d06a2a7770a9e9331a72f08cfcf1128db30dcff3',
+ //'OpenKeychain.extern.StickyListHeaders:library:24e25da422efc08e4e7a06efbe927fdf17f7a9aa722db2b983385e2bf0004da5',
+ //'com.madgag.spongycastle:core:a9e4f60afe6b2661e0713190ade92c099b3f74ebbc67c1bc3e3fced0144307f4',
+ //'com.madgag.spongycastle:pg:29d544ff289fcaafcf6c3904185f5a6fbdb623cf1a1e377fcb239edc31ee9c17',
+ //'com.madgag.spongycastle:pkix:950d6eac8205c6a24aa87066fbf9cd0af50b95858b8d2b18d53e2fada2dbb2e3',
+ //'com.madgag.spongycastle:prov:0b78ffd7a59b1b690a05ebe9bb31d43405046a44a18e0529d7c826acb56350b7',
+ //'OpenKeychain.extern:minidns:cf332e993d7fcdc0a3821f5b997944df40582dc6c9f0ea36b5e20c1e289cb19f',
+ //'OpenKeychain.extern.KeybaseLib:Lib:af9bff087148e0859430d0b99ece096c41b315c5dc1ed500a68580b9b0e5ab11',
+ //'OpenKeychain.extern.TokenAutoComplete:library:40d4212a95e947efdb02f2ca66c95a27d49fba848471a6317eca2b9cc18e8780',
+ //'OpenKeychain.extern:safeslinger-exchange:94a1ce68217af7499579a042758283b1530912c53241bdfa06d1a079a5ae3faf',
+ 'com.eftimoff:android-patternview:a031eaed3b5cef8ea06c2d4a6e27693937f89ae483598d61b7027eeee0bed408',
+ 'com.journeyapps:zxing-android-embedded:5d6ba3931bd0b999695e363b571e95bd6bc9956340c1e6ce740cd0bff3d89a50',
+ 'com.journeyapps:zxing-android-integration:6f50bb07c057ac94319777ddfbb66f5d4f6190393418b2fc861e0e60d06f3c0d',
+ 'com.google.zxing:core:38c49045765281e4c170062fa3f48e4e988629bf985cab850c7497be5eaa72a1',
+ 'com.jpardogo.materialtabstrip:library:c6ef812fba4f74be7dc4a905faa4c2908cba261a94c13d4f96d5e67e4aad4aaa',
+ 'it.neokree:MaterialNavigationDrawer:1174d751a54689fccf53c1fbcdf439745926ae19024f4f1017afb6b29643c57d',
+ 'com.nispok:snackbar:59dc092a44c877e9ce5f9040c632d99e62d8932b0a4d67ba0ec9e35467d9047c',
+ 'com.getbase:floatingactionbutton:e63966148212e9685afad2370780ea239b6dbd2a06f6a3f919b98882318e6a32',
+ 'com.android.support:support-annotations:fdee2354787ef66b268e75958de3f7f6c4f8f325510a6dac9f49c929f83a63de',
+ 'com.balysv:material-ripple:587f19c1e27f16c7dc67ff9ac73838aa1451086ef05a15cee38bee3e4e1454ae',
+ ]
+}
+
android {
- compileSdkVersion 21
- buildToolsVersion '21.1.2'
+ compileSdkVersion rootProject.ext.compileSdkVersion
+ buildToolsVersion rootProject.ext.buildToolsVersion
defaultConfig {
minSdkVersion 15
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/operations/results/LinkedVerifyResult.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/operations/results/LinkedVerifyResult.java
index 2dae38cc4..42e9ec3f0 100644
--- a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/operations/results/LinkedVerifyResult.java
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/operations/results/LinkedVerifyResult.java
@@ -18,21 +18,7 @@
package org.sufficientlysecure.keychain.operations.results;
-import android.app.Activity;
-import android.content.Intent;
import android.os.Parcel;
-import android.os.Parcelable;
-import android.view.View;
-
-import com.github.johnpersano.supertoasts.SuperCardToast;
-import com.github.johnpersano.supertoasts.SuperToast;
-import com.github.johnpersano.supertoasts.SuperToast.Duration;
-import com.github.johnpersano.supertoasts.util.OnClickWrapper;
-import com.github.johnpersano.supertoasts.util.Style;
-
-import org.sufficientlysecure.keychain.R;
-import org.sufficientlysecure.keychain.ui.LogDisplayActivity;
-import org.sufficientlysecure.keychain.ui.LogDisplayFragment;
public class LinkedVerifyResult extends OperationResult {
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyActivity.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyActivity.java
index 742cde75c..0be6c26f6 100644
--- a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyActivity.java
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyActivity.java
@@ -803,6 +803,7 @@ public class ViewKeyActivity extends BaseActivity implements
mName.setText(R.string.user_id_no_name);
}
+ String oldFingerprint = mFingerprint;
mMasterKeyId = data.getLong(INDEX_MASTER_KEY_ID);
mFingerprint = KeyFormattingUtils.convertFingerprintToHex(data.getBlob(INDEX_FINGERPRINT));
@@ -866,8 +867,11 @@ public class ViewKeyActivity extends BaseActivity implements
mStatusText.setText(R.string.view_key_my_key);
mStatusImage.setVisibility(View.GONE);
color = getResources().getColor(R.color.primary);
+ // reload qr code only if the fingerprint changed
+ if ( !mFingerprint.equals(oldFingerprint)) {
+ loadQrCode(mFingerprint);
+ }
photoTask.execute(mFingerprint);
- loadQrCode(mFingerprint);
mQrCodeLayout.setVisibility(View.VISIBLE);
// and place leftOf qr code
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyFragment.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyFragment.java
index 453bfd499..32630b459 100644
--- a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyFragment.java
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/ViewKeyFragment.java
@@ -34,14 +34,10 @@ import org.sufficientlysecure.keychain.Constants;
import org.sufficientlysecure.keychain.R;
import org.sufficientlysecure.keychain.compatibility.DialogFragmentWorkaround;
import org.sufficientlysecure.keychain.provider.KeychainContract;
-import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
-import org.sufficientlysecure.keychain.provider.ProviderHelper;
import org.sufficientlysecure.keychain.ui.adapter.UserIdsAdapter;
import org.sufficientlysecure.keychain.ui.dialog.UserIdInfoDialogFragment;
import org.sufficientlysecure.keychain.util.Log;
-import java.util.Date;
-
public class ViewKeyFragment extends LoaderFragment implements
LoaderManager.LoaderCallbacks<Cursor> {
@@ -58,8 +54,6 @@ public class ViewKeyFragment extends LoaderFragment implements
private Uri mDataUri;
- ProviderHelper mProviderHelper;
-
/**
* Creates new instance of this fragment
*/
@@ -78,8 +72,6 @@ public class ViewKeyFragment extends LoaderFragment implements
View root = super.onCreateView(inflater, superContainer, savedInstanceState);
View view = inflater.inflate(R.layout.view_key_fragment, getContainer());
- mProviderHelper = new ProviderHelper(getActivity());
-
mUserIds = (ListView) view.findViewById(R.id.view_key_user_ids);
mUserIds.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@@ -148,16 +140,14 @@ public class ViewKeyFragment extends LoaderFragment implements
private void loadData(Uri dataUri) {
mDataUri = dataUri;
- Log.i(Constants.TAG, "mDataUri: " + mDataUri.toString());
+ Log.i(Constants.TAG, "mDataUri: " + mDataUri);
// Prepare the loaders. Either re-connect with an existing ones,
// or start new ones.
+ // TODO Is this loader the same as the one in the activity?
getLoaderManager().initLoader(LOADER_ID_UNIFIED, null, this);
}
- // don't show revoked user ids here, irrelevant for average users
- public static final String USER_IDS_WHERE = UserPackets.IS_REVOKED + " = 0";
-
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
setContentShown(false);
@@ -166,11 +156,8 @@ public class ViewKeyFragment extends LoaderFragment implements
Uri baseUri = KeychainContract.KeyRings.buildUnifiedKeyRingUri(mDataUri);
return new CursorLoader(getActivity(), baseUri, UNIFIED_PROJECTION, null, null, null);
}
- case LOADER_ID_USER_IDS: {
- Uri baseUri = UserPackets.buildUserIdsUri(mDataUri);
- return new CursorLoader(getActivity(), baseUri,
- UserIdsAdapter.USER_IDS_PROJECTION, USER_IDS_WHERE, null, null);
- }
+ case LOADER_ID_USER_IDS:
+ return UserIdsAdapter.createLoader(getActivity(), mDataUri);
default:
return null;
@@ -192,14 +179,9 @@ public class ViewKeyFragment extends LoaderFragment implements
if (data.moveToFirst()) {
mIsSecret = data.getInt(INDEX_HAS_ANY_SECRET) != 0;
- boolean hasEncrypt = data.getInt(INDEX_HAS_ENCRYPT) != 0;
- boolean isRevoked = data.getInt(INDEX_IS_REVOKED) > 0;
- boolean isExpired = !data.isNull(INDEX_EXPIRY)
- && new Date(data.getLong(INDEX_EXPIRY) * 1000).before(new Date());
- boolean isVerified = data.getInt(INDEX_VERIFIED) > 0;
// load user ids after we know if it's a secret key
- mUserIdsAdapter = new UserIdsAdapter(getActivity(), null, 0, false, !mIsSecret, null);
+ mUserIdsAdapter = new UserIdsAdapter(getActivity(), null, 0, !mIsSecret, null);
mUserIds.setAdapter(mUserIdsAdapter);
getLoaderManager().initLoader(LOADER_ID_USER_IDS, null, this);
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserAttributesAdapter.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserAttributesAdapter.java
index 36ee8ba88..457083770 100644
--- a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserAttributesAdapter.java
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserAttributesAdapter.java
@@ -1,234 +1,36 @@
-/*
- * Copyright (C) 2014 Dominik Schürmann <dominik@dominikschuermann.de>
- *
- * This program 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.
- *
- * This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
- */
-
package org.sufficientlysecure.keychain.ui.adapter;
import android.content.Context;
import android.database.Cursor;
-import android.graphics.Typeface;
import android.support.v4.widget.CursorAdapter;
-import android.view.LayoutInflater;
import android.view.View;
-import android.view.ViewGroup;
-import android.widget.AdapterView;
-import android.widget.CheckBox;
-import android.widget.CompoundButton;
-import android.widget.ImageView;
-import android.widget.TextView;
-import org.sufficientlysecure.keychain.R;
import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
-import org.sufficientlysecure.keychain.pgp.KeyRing;
-import org.sufficientlysecure.keychain.provider.KeychainContract.Certs;
-import org.sufficientlysecure.keychain.service.SaveKeyringParcel;
-import org.sufficientlysecure.keychain.ui.util.KeyFormattingUtils;
-
-import java.util.ArrayList;
-
-public abstract class UserAttributesAdapter extends CursorAdapter implements AdapterView.OnItemClickListener {
- private LayoutInflater mInflater;
- private final ArrayList<Boolean> mCheckStates;
- private SaveKeyringParcel mSaveKeyringParcel;
- private boolean mShowStatusImages;
- public static final String[] USER_PACKETS_PROJECTION = new String[]{
+public abstract class UserAttributesAdapter extends CursorAdapter {
+ public static final String[] USER_IDS_PROJECTION = new String[]{
UserPackets._ID,
UserPackets.TYPE,
UserPackets.USER_ID,
- UserPackets.ATTRIBUTE_DATA,
UserPackets.RANK,
UserPackets.VERIFIED,
UserPackets.IS_PRIMARY,
UserPackets.IS_REVOKED
};
- private static final int INDEX_ID = 0;
- private static final int INDEX_TYPE = 1;
- private static final int INDEX_USER_ID = 2;
- private static final int INDEX_ATTRIBUTE_DATA = 3;
- private static final int INDEX_RANK = 4;
- private static final int INDEX_VERIFIED = 5;
- private static final int INDEX_IS_PRIMARY = 6;
- private static final int INDEX_IS_REVOKED = 7;
-
- public UserAttributesAdapter(Context context, Cursor c, int flags, boolean showCheckBoxes,
- boolean showStatusImages, SaveKeyringParcel saveKeyringParcel) {
+ protected static final int INDEX_ID = 0;
+ protected static final int INDEX_TYPE = 1;
+ protected static final int INDEX_USER_ID = 2;
+ protected static final int INDEX_RANK = 3;
+ protected static final int INDEX_VERIFIED = 4;
+ protected static final int INDEX_IS_PRIMARY = 5;
+ protected static final int INDEX_IS_REVOKED = 6;
+
+ public UserAttributesAdapter(Context context, Cursor c, int flags) {
super(context, c, flags);
- mInflater = LayoutInflater.from(context);
-
- mCheckStates = showCheckBoxes ? new ArrayList<Boolean>() : null;
- mSaveKeyringParcel = saveKeyringParcel;
- mShowStatusImages = showStatusImages;
}
@Override
- public Cursor swapCursor(Cursor newCursor) {
- if (mCheckStates != null) {
- mCheckStates.clear();
- if (newCursor != null) {
- int count = newCursor.getCount();
- mCheckStates.ensureCapacity(count);
- // initialize to true (use case knowledge: we usually want to sign all uids)
- for (int i = 0; i < count; i++) {
- newCursor.moveToPosition(i);
- int verified = newCursor.getInt(INDEX_VERIFIED);
- mCheckStates.add(verified != Certs.VERIFIED_SECRET);
- }
- }
- }
-
- return super.swapCursor(newCursor);
- }
-
- @Override
- public void bindView(View view, Context context, Cursor cursor) {
- TextView vName = (TextView) view.findViewById(R.id.user_id_item_name);
- TextView vAddress = (TextView) view.findViewById(R.id.user_id_item_address);
- TextView vComment = (TextView) view.findViewById(R.id.user_id_item_comment);
- ImageView vVerified = (ImageView) view.findViewById(R.id.user_id_item_certified);
- View vVerifiedLayout = view.findViewById(R.id.user_id_item_certified_layout);
- ImageView vEditImage = (ImageView) view.findViewById(R.id.user_id_item_edit_image);
- ImageView vDeleteButton = (ImageView) view.findViewById(R.id.user_id_item_delete_button);
- vDeleteButton.setVisibility(View.GONE); // not used
-
- String userId = cursor.getString(INDEX_USER_ID);
- String[] splitUserId = KeyRing.splitUserId(userId);
- if (splitUserId[0] != null) {
- vName.setText(splitUserId[0]);
- } else {
- vName.setText(R.string.user_id_no_name);
- }
- if (splitUserId[1] != null) {
- vAddress.setText(splitUserId[1]);
- vAddress.setVisibility(View.VISIBLE);
- } else {
- vAddress.setVisibility(View.GONE);
- }
- if (splitUserId[2] != null) {
- vComment.setText(splitUserId[2]);
- vComment.setVisibility(View.VISIBLE);
- } else {
- vComment.setVisibility(View.GONE);
- }
-
- boolean isPrimary = cursor.getInt(INDEX_IS_PRIMARY) != 0;
- boolean isRevoked = cursor.getInt(INDEX_IS_REVOKED) > 0;
-
- // for edit key
- if (mSaveKeyringParcel != null) {
- boolean changeAnyPrimaryUserId = (mSaveKeyringParcel.mChangePrimaryUserId != null);
- boolean changeThisPrimaryUserId = (mSaveKeyringParcel.mChangePrimaryUserId != null
- && mSaveKeyringParcel.mChangePrimaryUserId.equals(userId));
- boolean revokeThisUserId = (mSaveKeyringParcel.mRevokeUserIds.contains(userId));
-
- // only if primary user id will be changed
- // (this is not triggered if the user id is currently the primary one)
- if (changeAnyPrimaryUserId) {
- // change _all_ primary user ids and set new one to true
- isPrimary = changeThisPrimaryUserId;
- }
-
- if (revokeThisUserId) {
- if (!isRevoked) {
- isRevoked = true;
- }
- }
-
- vEditImage.setVisibility(View.VISIBLE);
- vVerifiedLayout.setVisibility(View.GONE);
- } else {
- vEditImage.setVisibility(View.GONE);
-
- if (mShowStatusImages) {
- vVerifiedLayout.setVisibility(View.VISIBLE);
- } else {
- vVerifiedLayout.setVisibility(View.GONE);
- }
- }
-
- if (isRevoked) {
- // set revocation icon (can this even be primary?)
- KeyFormattingUtils.setStatusImage(mContext, vVerified, null, KeyFormattingUtils.STATE_REVOKED, R.color.bg_gray);
-
- // disable revoked user ids
- vName.setEnabled(false);
- vAddress.setEnabled(false);
- vComment.setEnabled(false);
- } else {
- vName.setEnabled(true);
- vAddress.setEnabled(true);
- vComment.setEnabled(true);
-
- if (isPrimary) {
- vName.setTypeface(null, Typeface.BOLD);
- vAddress.setTypeface(null, Typeface.BOLD);
- } else {
- vName.setTypeface(null, Typeface.NORMAL);
- vAddress.setTypeface(null, Typeface.NORMAL);
- }
-
- int isVerified = cursor.getInt(INDEX_VERIFIED);
- switch (isVerified) {
- case Certs.VERIFIED_SECRET:
- KeyFormattingUtils.setStatusImage(mContext, vVerified, null, KeyFormattingUtils.STATE_VERIFIED, KeyFormattingUtils.DEFAULT_COLOR);
- break;
- case Certs.VERIFIED_SELF:
- KeyFormattingUtils.setStatusImage(mContext, vVerified, null, KeyFormattingUtils.STATE_UNVERIFIED, KeyFormattingUtils.DEFAULT_COLOR);
- break;
- default:
- KeyFormattingUtils.setStatusImage(mContext, vVerified, null, KeyFormattingUtils.STATE_INVALID, KeyFormattingUtils.DEFAULT_COLOR);
- break;
- }
- }
-
- // don't care further if checkboxes aren't shown
- if (mCheckStates == null) {
- return;
- }
-
- final CheckBox vCheckBox = (CheckBox) view.findViewById(R.id.user_id_item_check_box);
- final int position = cursor.getPosition();
- vCheckBox.setOnCheckedChangeListener(null);
- vCheckBox.setChecked(mCheckStates.get(position));
- vCheckBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
- @Override
- public void onCheckedChanged(CompoundButton compoundButton, boolean b) {
- mCheckStates.set(position, b);
- }
- });
- vCheckBox.setClickable(false);
- }
-
- public void onItemClick(AdapterView<?> adapter, View view, int position, long id) {
- CheckBox box = ((CheckBox) view.findViewById(R.id.user_id_item_check_box));
- if (box != null) {
- box.toggle();
- }
- }
-
- public ArrayList<String> getSelectedUserIds() {
- ArrayList<String> result = new ArrayList<>();
- for (int i = 0; i < mCheckStates.size(); i++) {
- if (mCheckStates.get(i)) {
- mCursor.moveToPosition(i);
- result.add(mCursor.getString(INDEX_USER_ID));
- }
- }
- return result;
- }
+ public abstract void bindView(View view, Context context, Cursor cursor);
public String getUserId(int position) {
mCursor.moveToPosition(position);
@@ -244,28 +46,4 @@ public abstract class UserAttributesAdapter extends CursorAdapter implements Ada
mCursor.moveToPosition(position);
return mCursor.getInt(INDEX_VERIFIED);
}
-
- public boolean getIsRevokedPending(int position) {
- mCursor.moveToPosition(position);
- String userId = mCursor.getString(INDEX_USER_ID);
-
- boolean isRevokedPending = false;
- if (mSaveKeyringParcel != null) {
- if (mSaveKeyringParcel.mRevokeUserIds.contains(userId)) {
- isRevokedPending = true;
- }
-
- }
-
- return isRevokedPending;
- }
-
- @Override
- public View newView(Context context, Cursor cursor, ViewGroup parent) {
- View view = mInflater.inflate(R.layout.view_key_adv_user_id_item, null);
- // only need to do this once ever, since mShowCheckBoxes is final
- view.findViewById(R.id.user_id_item_check_box).setVisibility(mCheckStates != null ? View.VISIBLE : View.GONE);
- return view;
- }
-
}
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserIdsAdapter.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserIdsAdapter.java
new file mode 100644
index 000000000..6a4f61f4b
--- /dev/null
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserIdsAdapter.java
@@ -0,0 +1,193 @@
+/*
+ * Copyright (C) 2014-2015 Dominik Schürmann <dominik@dominikschuermann.de>
+ * Copyright (C) 2015 Vincent Breitmoser <v.breitmoser@mugenguild.com>
+ *
+ * This program 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.
+ *
+ * This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+package org.sufficientlysecure.keychain.ui.adapter;
+
+import android.app.Activity;
+import android.content.Context;
+import android.database.Cursor;
+import android.graphics.Typeface;
+import android.net.Uri;
+import android.support.v4.content.CursorLoader;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.ImageView;
+import android.widget.TextView;
+
+import org.sufficientlysecure.keychain.R;
+import org.sufficientlysecure.keychain.pgp.KeyRing;
+import org.sufficientlysecure.keychain.provider.KeychainContract.Certs;
+import org.sufficientlysecure.keychain.provider.KeychainContract.UserPackets;
+import org.sufficientlysecure.keychain.service.SaveKeyringParcel;
+import org.sufficientlysecure.keychain.ui.util.KeyFormattingUtils;
+
+public class UserIdsAdapter extends UserAttributesAdapter {
+ protected LayoutInflater mInflater;
+ private SaveKeyringParcel mSaveKeyringParcel;
+ private boolean mShowStatusImages;
+
+ public UserIdsAdapter(Context context, Cursor c, int flags,
+ boolean showStatusImages, SaveKeyringParcel saveKeyringParcel) {
+ super(context, c, flags);
+ mInflater = LayoutInflater.from(context);
+
+ mSaveKeyringParcel = saveKeyringParcel;
+ mShowStatusImages = showStatusImages;
+ }
+
+ public UserIdsAdapter(Context context, Cursor c, int flags, SaveKeyringParcel saveKeyringParcel) {
+ this(context, c, flags, true, saveKeyringParcel);
+ }
+
+ public UserIdsAdapter(Context context, Cursor c, int flags) {
+ this(context, c, flags, true, null);
+ }
+
+ @Override
+ public void bindView(View view, Context context, Cursor cursor) {
+ TextView vName = (TextView) view.findViewById(R.id.user_id_item_name);
+ TextView vAddress = (TextView) view.findViewById(R.id.user_id_item_address);
+ TextView vComment = (TextView) view.findViewById(R.id.user_id_item_comment);
+ ImageView vVerified = (ImageView) view.findViewById(R.id.user_id_item_certified);
+ View vVerifiedLayout = view.findViewById(R.id.user_id_item_certified_layout);
+ ImageView vEditImage = (ImageView) view.findViewById(R.id.user_id_item_edit_image);
+ ImageView vDeleteButton = (ImageView) view.findViewById(R.id.user_id_item_delete_button);
+ vDeleteButton.setVisibility(View.GONE); // not used
+
+ String userId = cursor.getString(INDEX_USER_ID);
+ String[] splitUserId = KeyRing.splitUserId(userId);
+ if (splitUserId[0] != null) {
+ vName.setText(splitUserId[0]);
+ } else {
+ vName.setText(R.string.user_id_no_name);
+ }
+ if (splitUserId[1] != null) {
+ vAddress.setText(splitUserId[1]);
+ vAddress.setVisibility(View.VISIBLE);
+ } else {
+ vAddress.setVisibility(View.GONE);
+ }
+ if (splitUserId[2] != null) {
+ vComment.setText(splitUserId[2]);
+ vComment.setVisibility(View.VISIBLE);
+ } else {
+ vComment.setVisibility(View.GONE);
+ }
+
+ boolean isPrimary = cursor.getInt(INDEX_IS_PRIMARY) != 0;
+ boolean isRevoked = cursor.getInt(INDEX_IS_REVOKED) > 0;
+
+ // for edit key
+ if (mSaveKeyringParcel != null) {
+ boolean changeAnyPrimaryUserId = (mSaveKeyringParcel.mChangePrimaryUserId != null);
+ boolean changeThisPrimaryUserId = (mSaveKeyringParcel.mChangePrimaryUserId != null
+ && mSaveKeyringParcel.mChangePrimaryUserId.equals(userId));
+ boolean revokeThisUserId = (mSaveKeyringParcel.mRevokeUserIds.contains(userId));
+
+ // only if primary user id will be changed
+ // (this is not triggered if the user id is currently the primary one)
+ if (changeAnyPrimaryUserId) {
+ // change _all_ primary user ids and set new one to true
+ isPrimary = changeThisPrimaryUserId;
+ }
+
+ if (revokeThisUserId) {
+ if (!isRevoked) {
+ isRevoked = true;
+ }
+ }
+
+ vEditImage.setVisibility(View.VISIBLE);
+ vVerifiedLayout.setVisibility(View.GONE);
+ } else {
+ vEditImage.setVisibility(View.GONE);
+
+ if (mShowStatusImages) {
+ vVerifiedLayout.setVisibility(View.VISIBLE);
+ } else {
+ vVerifiedLayout.setVisibility(View.GONE);
+ }
+ }
+
+ if (isRevoked) {
+ // set revocation icon (can this even be primary?)
+ KeyFormattingUtils.setStatusImage(mContext, vVerified, null, KeyFormattingUtils.STATE_REVOKED, R.color.bg_gray);
+
+ // disable revoked user ids
+ vName.setEnabled(false);
+ vAddress.setEnabled(false);
+ vComment.setEnabled(false);
+ } else {
+ vName.setEnabled(true);
+ vAddress.setEnabled(true);
+ vComment.setEnabled(true);
+
+ if (isPrimary) {
+ vName.setTypeface(null, Typeface.BOLD);
+ vAddress.setTypeface(null, Typeface.BOLD);
+ } else {
+ vName.setTypeface(null, Typeface.NORMAL);
+ vAddress.setTypeface(null, Typeface.NORMAL);
+ }
+
+ int isVerified = cursor.getInt(INDEX_VERIFIED);
+ switch (isVerified) {
+ case Certs.VERIFIED_SECRET:
+ KeyFormattingUtils.setStatusImage(mContext, vVerified, null, KeyFormattingUtils.STATE_VERIFIED, KeyFormattingUtils.DEFAULT_COLOR);
+ break;
+ case Certs.VERIFIED_SELF:
+ KeyFormattingUtils.setStatusImage(mContext, vVerified, null, KeyFormattingUtils.STATE_UNVERIFIED, KeyFormattingUtils.DEFAULT_COLOR);
+ break;
+ default:
+ KeyFormattingUtils.setStatusImage(mContext, vVerified, null, KeyFormattingUtils.STATE_INVALID, KeyFormattingUtils.DEFAULT_COLOR);
+ break;
+ }
+ }
+ }
+
+ public boolean getIsRevokedPending(int position) {
+ mCursor.moveToPosition(position);
+ String userId = mCursor.getString(INDEX_USER_ID);
+
+ boolean isRevokedPending = false;
+ if (mSaveKeyringParcel != null) {
+ if (mSaveKeyringParcel.mRevokeUserIds.contains(userId)) {
+ isRevokedPending = true;
+ }
+
+ }
+
+ return isRevokedPending;
+ }
+
+ @Override
+ public View newView(Context context, Cursor cursor, ViewGroup parent) {
+ return mInflater.inflate(R.layout.view_key_adv_user_id_item, null);
+ }
+
+ // don't show revoked user ids, irrelevant for average users
+ public static final String USER_IDS_WHERE = UserPackets.IS_REVOKED + " = 0";
+
+ public static CursorLoader createLoader(Activity activity, Uri dataUri) {
+ Uri baseUri = UserPackets.buildUserIdsUri(dataUri);
+ return new CursorLoader(activity, baseUri,
+ UserIdsAdapter.USER_IDS_PROJECTION, USER_IDS_WHERE, null, null);
+ }
+
+}
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserIdsSelectableAdapter.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserIdsSelectableAdapter.java
new file mode 100644
index 000000000..947d911c3
--- /dev/null
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/ui/adapter/UserIdsSelectableAdapter.java
@@ -0,0 +1,88 @@
+package org.sufficientlysecure.keychain.ui.adapter;
+
+import android.content.Context;
+import android.database.Cursor;
+import android.view.View;
+import android.view.ViewGroup;
+import android.widget.AdapterView;
+import android.widget.CheckBox;
+import android.widget.CompoundButton;
+
+import org.sufficientlysecure.keychain.R;
+import org.sufficientlysecure.keychain.provider.KeychainContract.Certs;
+import org.sufficientlysecure.keychain.service.SaveKeyringParcel;
+
+import java.util.ArrayList;
+
+public class UserIdsSelectableAdapter extends UserIdsAdapter implements AdapterView.OnItemClickListener {
+
+ private final ArrayList<Boolean> mCheckStates;
+
+ public UserIdsSelectableAdapter(Context context, Cursor c, int flags, SaveKeyringParcel saveKeyringParcel) {
+ super(context, c, flags, saveKeyringParcel);
+
+ mCheckStates = new ArrayList<Boolean>();
+ }
+
+ @Override
+ public Cursor swapCursor(Cursor newCursor) {
+ if (mCheckStates != null) {
+ mCheckStates.clear();
+ if (newCursor != null) {
+ int count = newCursor.getCount();
+ mCheckStates.ensureCapacity(count);
+ // initialize to true (use case knowledge: we usually want to sign all uids)
+ for (int i = 0; i < count; i++) {
+ newCursor.moveToPosition(i);
+ int verified = newCursor.getInt(INDEX_VERIFIED);
+ mCheckStates.add(verified != Certs.VERIFIED_SECRET);
+ }
+ }
+ }
+
+ return super.swapCursor(newCursor);
+ }
+
+ @Override
+ public void onItemClick(AdapterView<?> adapter, View view, int position, long id) {
+ CheckBox box = ((CheckBox) view.findViewById(R.id.user_id_item_check_box));
+ if (box != null) {
+ box.toggle();
+ }
+ }
+
+ @Override
+ public void bindView(View view, Context context, Cursor cursor) {
+ super.bindView(view, context, cursor);
+
+ final CheckBox vCheckBox = (CheckBox) view.findViewById(R.id.user_id_item_check_box);
+ final int position = cursor.getPosition();
+ vCheckBox.setOnCheckedChangeListener(null);
+ vCheckBox.setChecked(mCheckStates.get(position));
+ vCheckBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
+ @Override
+ public void onCheckedChanged(CompoundButton compoundButton, boolean b) {
+ mCheckStates.set(position, b);
+ }
+ });
+ vCheckBox.setClickable(false);
+ }
+
+ public ArrayList<String> getSelectedUserIds() {
+ ArrayList<String> result = new ArrayList<>();
+ for (int i = 0; i < mCheckStates.size(); i++) {
+ if (mCheckStates.get(i)) {
+ mCursor.moveToPosition(i);
+ result.add(mCursor.getString(INDEX_USER_ID));
+ }
+ }
+ return result;
+ }
+
+ @Override
+ public View newView(Context context, Cursor cursor, ViewGroup parent) {
+ View view = mInflater.inflate(R.layout.view_key_selectable_user_id_item, null);
+ return view;
+ }
+
+}
diff --git a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/util/ContactHelper.java b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/util/ContactHelper.java
index 60bc846b2..3cf201ed7 100644
--- a/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/util/ContactHelper.java
+++ b/OpenKeychain/src/main/java/org/sufficientlysecure/keychain/util/ContactHelper.java
@@ -57,10 +57,20 @@ public class ContactHelper {
KeychainContract.KeyRings.MASTER_KEY_ID,
KeychainContract.KeyRings.EXPIRY,
KeychainContract.KeyRings.IS_REVOKED};
+
+ public static final int INDEX_USER_ID = 0;
+ public static final int INDEX_FINGERPRINT = 1;
+ public static final int INDEX_KEY_ID = 2;
+ public static final int INDEX_MASTER_KEY_ID = 3;
+ public static final int INDEX_EXPIRY = 4;
+ public static final int INDEX_IS_REVOKED = 5;
+
public static final String[] USER_IDS_PROJECTION = new String[]{
UserPackets.USER_ID
};
+ public static final int INDEX_USER_IDS_USER_ID = 0;
+
public static final String NON_REVOKED_SELECTION = UserPackets.IS_REVOKED + "=0";
public static final String[] ID_PROJECTION = new String[]{ContactsContract.RawContacts._ID};
@@ -301,23 +311,26 @@ public class ContactHelper {
null, null, null);
if (cursor != null) {
while (cursor.moveToNext()) {
- String[] primaryUserId = KeyRing.splitUserId(cursor.getString(0));
- String fingerprint = KeyFormattingUtils.convertFingerprintToHex(cursor.getBlob(1));
+ String[] primaryUserId = KeyRing.splitUserId(cursor.getString(INDEX_USER_ID));
+ String fingerprint = KeyFormattingUtils.convertFingerprintToHex(cursor.getBlob(INDEX_FINGERPRINT));
contactFingerprints.remove(fingerprint);
- String keyIdShort = KeyFormattingUtils.convertKeyIdToHexShort(cursor.getLong(2));
- long masterKeyId = cursor.getLong(3);
- boolean isExpired = !cursor.isNull(4) && new Date(cursor.getLong(4) * 1000).before(new Date());
- boolean isRevoked = cursor.getInt(5) > 0;
+ String keyIdShort = KeyFormattingUtils.convertKeyIdToHexShort(cursor.getLong(INDEX_KEY_ID));
+ long masterKeyId = cursor.getLong(INDEX_MASTER_KEY_ID);
+ boolean isExpired = !cursor.isNull(INDEX_EXPIRY)
+ && new Date(cursor.getLong(INDEX_EXPIRY) * 1000).before(new Date());
+ boolean isRevoked = cursor.getInt(INDEX_IS_REVOKED) > 0;
int rawContactId = findRawContactId(resolver, fingerprint);
ArrayList<ContentProviderOperation> ops = new ArrayList<>();
+ Log.d(Constants.TAG, "raw contact id: "+rawContactId);
+
// Do not store expired or revoked keys in contact db - and remove them if they already exist
if (isExpired || isRevoked) {
if (rawContactId != -1) {
resolver.delete(ContactsContract.RawContacts.CONTENT_URI, ID_SELECTION,
new String[]{Integer.toString(rawContactId)});
}
- } else {
+ } else if (primaryUserId[0] != null) {
// Create a new rawcontact with corresponding key if it does not exist yet
if (rawContactId == -1) {
diff --git a/OpenKeychain/src/main/res/layout/view_key_adv_user_id_item.xml b/OpenKeychain/src/main/res/layout/view_key_adv_user_id_item.xml
index 24e8e1a10..63c1ee70f 100644
--- a/OpenKeychain/src/main/res/layout/view_key_adv_user_id_item.xml
+++ b/OpenKeychain/src/main/res/layout/view_key_adv_user_id_item.xml
@@ -6,13 +6,6 @@
android:orientation="horizontal"
android:singleLine="true">
- <CheckBox
- android:id="@+id/user_id_item_check_box"
- android:layout_width="wrap_content"
- android:layout_height="match_parent"
- android:clickable="false"
- android:focusable="false" />
-
<LinearLayout
android:id="@+id/user_id_item_certified_layout"
android:layout_width="22dp"
@@ -25,6 +18,7 @@
android:id="@+id/user_id_item_certified"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
+ android:src="@drawable/status_signature_unverified_cutout_24px"
android:layout_gravity="center_horizontal" />
</LinearLayout>
diff --git a/OpenKeychain/src/main/res/layout/view_key_selectable_user_id_item.xml b/OpenKeychain/src/main/res/layout/view_key_selectable_user_id_item.xml
new file mode 100644
index 000000000..5d0edc916
--- /dev/null
+++ b/OpenKeychain/src/main/res/layout/view_key_selectable_user_id_item.xml
@@ -0,0 +1,18 @@
+<?xml version="1.0" encoding="utf-8"?>
+<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ android:layout_width="match_parent"
+ android:layout_height="wrap_content"
+ android:minHeight="?android:attr/listPreferredItemHeight"
+ android:orientation="horizontal"
+ android:singleLine="true">
+
+ <CheckBox
+ android:id="@+id/user_id_item_check_box"
+ android:layout_width="wrap_content"
+ android:layout_height="match_parent"
+ android:clickable="false"
+ android:focusable="false" />
+
+ <include layout="@layout/view_key_adv_user_id_item" />
+
+</LinearLayout>
diff --git a/OpenKeychain/src/main/res/values/colors.xml b/OpenKeychain/src/main/res/values/colors.xml
index c49e06b4b..8c5358fd0 100644
--- a/OpenKeychain/src/main/res/values/colors.xml
+++ b/OpenKeychain/src/main/res/values/colors.xml
@@ -13,6 +13,8 @@
<!-- Other colors -->
+ <color name="black">#000000</color>
+
<color name="primary_light">#C8E6C9</color>
<color name="fab">@color/accent</color>
<color name="fab_pressed">#1976D2</color>
diff --git a/OpenKeychain/src/main/res/values/strings.xml b/OpenKeychain/src/main/res/values/strings.xml
index b9f892687..4260cac3f 100644
--- a/OpenKeychain/src/main/res/values/strings.xml
+++ b/OpenKeychain/src/main/res/values/strings.xml
@@ -360,8 +360,7 @@
<string name="progress_con_reimport">"consolidate: reimporting…"</string>
<!-- action strings -->
- <string name="hint_keyserver_search_hint">"Name/Email/Key ID…"</string>
- <string name="hint_cloud_search_hint">"Name/Email/Proof/Key…"</string>
+ <string name="hint_cloud_search_hint">"Search via Name, Email…"</string>
<!-- key bit length selections -->
<string name="key_size_512">"512"</string>