diff options
| author | Vincent Breitmoser <valodim@mugenguild.com> | 2015-03-02 18:34:51 +0100 | 
|---|---|---|
| committer | Vincent Breitmoser <valodim@mugenguild.com> | 2015-03-02 18:34:51 +0100 | 
| commit | e6af789bf50f9636d3455c4bd1b662ab5d04e3e2 (patch) | |
| tree | 35e7560cd785aa20c1fa814332d16bd094921282 /OpenKeychain/src/main/java/org | |
| parent | 5e53a5417c11f54f437d9f220d3e2e8579b8bf1b (diff) | |
| parent | 145e793db1d732bdd230be92750123cd20c9502d (diff) | |
| download | open-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/src/main/java/org')
7 files changed, 323 insertions, 279 deletions
| 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) { | 
