Switched to SuperSLiM on KeyListFragment and Fixed issue #1051

Changed scrollbar style
This commit is contained in:
Manoj Khanna 2015-03-28 00:02:06 +05:30
parent 6b48ddd717
commit adb8ca42ac
13 changed files with 804 additions and 515 deletions

View File

@ -2,7 +2,9 @@ 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
// NOTE: libraries are pinned to a specific build, see below
// from local Android SDK
@ -10,7 +12,6 @@ dependencies {
compile 'com.android.support:appcompat-v7:22.0.0'
compile 'com.android.support:recyclerview-v7:22.0.0'
compile 'com.android.support:cardview-v7:22.0.0'
// JCenter etc.
compile 'com.eftimoff:android-patternview:1.0.1@aar'
compile 'com.journeyapps:zxing-android-embedded:2.1.0@aar'
@ -20,7 +21,7 @@ dependencies {
compile 'it.neokree:MaterialNavigationDrawer:1.3.2'
compile 'com.getbase:floatingactionbutton:1.9.0'
compile 'org.commonjava.googlecode.markdown4j:markdown4j:2.2-cj-1.0'
compile 'com.tonicartos:superslim:0.4.8'
// libs as submodules
compile project(':extern:openpgp-api-lib')
compile project(':extern:openkeychain-api-lib')

View File

@ -25,8 +25,6 @@ import android.app.ProgressDialog;
import android.content.Context;
import android.content.Intent;
import android.database.Cursor;
import android.graphics.Color;
import android.graphics.PorterDuff;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
@ -37,9 +35,10 @@ import android.support.v4.app.LoaderManager;
import android.support.v4.content.CursorLoader;
import android.support.v4.content.Loader;
import android.support.v4.view.MenuItemCompat;
import android.support.v4.widget.CursorAdapter;
import android.support.v7.app.ActionBarActivity;
import android.support.v7.view.ActionMode;
import android.support.v7.widget.RecyclerView;
import android.support.v7.widget.SearchView;
import android.view.ActionMode;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuInflater;
@ -47,15 +46,11 @@ import android.view.MenuItem;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.AbsListView.MultiChoiceModeListener;
import android.widget.AdapterView;
import android.widget.ImageButton;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.LinearLayout;
import com.getbase.floatingactionbutton.FloatingActionButton;
import com.getbase.floatingactionbutton.FloatingActionsMenu;
import com.tonicartos.superslim.LayoutManager;
import org.sufficientlysecure.keychain.Constants;
import org.sufficientlysecure.keychain.R;
@ -64,20 +59,18 @@ import org.sufficientlysecure.keychain.operations.results.ConsolidateResult;
import org.sufficientlysecure.keychain.operations.results.DeleteResult;
import org.sufficientlysecure.keychain.operations.results.ImportKeyResult;
import org.sufficientlysecure.keychain.operations.results.OperationResult;
import org.sufficientlysecure.keychain.pgp.KeyRing;
import org.sufficientlysecure.keychain.provider.KeychainContract;
import org.sufficientlysecure.keychain.provider.KeychainContract.KeyRings;
import org.sufficientlysecure.keychain.provider.KeychainDatabase;
import org.sufficientlysecure.keychain.provider.ProviderHelper;
import org.sufficientlysecure.keychain.service.CloudImportService;
import org.sufficientlysecure.keychain.service.KeychainIntentService;
import org.sufficientlysecure.keychain.service.ServiceProgressHandler;
import org.sufficientlysecure.keychain.service.PassphraseCacheService;
import org.sufficientlysecure.keychain.service.ServiceProgressHandler;
import org.sufficientlysecure.keychain.ui.adapter.KeyListAdapter;
import org.sufficientlysecure.keychain.ui.dialog.DeleteKeyDialogFragment;
import org.sufficientlysecure.keychain.ui.dialog.ProgressDialogFragment;
import org.sufficientlysecure.keychain.ui.util.Highlighter;
import org.sufficientlysecure.keychain.ui.util.KeyFormattingUtils;
import org.sufficientlysecure.keychain.ui.util.KeyFormattingUtils.State;
import org.sufficientlysecure.keychain.ui.util.Notify;
import org.sufficientlysecure.keychain.util.ExportHelper;
import org.sufficientlysecure.keychain.util.FabContainer;
@ -86,26 +79,22 @@ import org.sufficientlysecure.keychain.util.Preferences;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import se.emilsjolander.stickylistheaders.StickyListHeadersAdapter;
import se.emilsjolander.stickylistheaders.StickyListHeadersListView;
/**
* Public key list with sticky list headers. It does _not_ extend ListFragment because it uses
* StickyListHeaders library which does not extend upon ListView.
*/
public class KeyListFragment extends LoaderFragment
implements SearchView.OnQueryTextListener, AdapterView.OnItemClickListener,
LoaderManager.LoaderCallbacks<Cursor>, FabContainer {
implements SearchView.OnQueryTextListener, KeyListAdapter.OnClickListener,
LoaderManager.LoaderCallbacks<Cursor>, FabContainer, ActionMode.Callback {
static final int REQUEST_REPEAT_PASSPHRASE = 1;
static final int REQUEST_ACTION = 2;
ExportHelper mExportHelper;
private RecyclerView mRecyclerView;
private KeyListAdapter mAdapter;
private StickyListHeadersListView mStickyList;
// saves the mode object for multiselect, needed for reset at some point
private ActionMode mActionMode = null;
@ -134,8 +123,9 @@ public class KeyListFragment extends LoaderFragment
View root = super.onCreateView(inflater, superContainer, savedInstanceState);
View view = inflater.inflate(R.layout.key_list_fragment, getContainer());
mStickyList = (StickyListHeadersListView) view.findViewById(R.id.key_list_list);
mStickyList.setOnItemClickListener(this);
mRecyclerView = (RecyclerView) view.findViewById(R.id.key_list_recycler);
mRecyclerView.setHasFixedSize(true);
mRecyclerView.setLayoutManager(new LayoutManager(getActivity()));
mFab = (FloatingActionsMenu) view.findViewById(R.id.fab_main);
@ -180,103 +170,6 @@ public class KeyListFragment extends LoaderFragment
// show app name instead of "keys" from nav drawer
getActivity().setTitle(R.string.app_name);
mStickyList.setOnItemClickListener(this);
mStickyList.setAreHeadersSticky(true);
mStickyList.setDrawingListUnderStickyHeader(false);
mStickyList.setFastScrollEnabled(true);
// Adds an empty footer view so that the Floating Action Button won't block content
// in last few rows.
View footer = new View(getActivity());
int spacing = (int) android.util.TypedValue.applyDimension(
android.util.TypedValue.COMPLEX_UNIT_DIP, 72, getResources().getDisplayMetrics()
);
android.widget.AbsListView.LayoutParams params = new android.widget.AbsListView.LayoutParams(
android.widget.AbsListView.LayoutParams.MATCH_PARENT,
spacing
);
footer.setLayoutParams(params);
mStickyList.addFooterView(footer, null, false);
/*
* Multi-selection
*/
mStickyList.setFastScrollAlwaysVisible(true);
mStickyList.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE_MODAL);
mStickyList.getWrappedList().setMultiChoiceModeListener(new MultiChoiceModeListener() {
@Override
public boolean onCreateActionMode(ActionMode mode, Menu menu) {
android.view.MenuInflater inflater = getActivity().getMenuInflater();
inflater.inflate(R.menu.key_list_multi, menu);
mActionMode = mode;
return true;
}
@Override
public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
return false;
}
@Override
public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
// get IDs for checked positions as long array
long[] ids;
switch (item.getItemId()) {
case R.id.menu_key_list_multi_encrypt: {
ids = mAdapter.getCurrentSelectedMasterKeyIds();
encrypt(mode, ids);
break;
}
case R.id.menu_key_list_multi_delete: {
ids = mAdapter.getCurrentSelectedMasterKeyIds();
showDeleteKeyDialog(mode, ids, mAdapter.isAnySecretSelected());
break;
}
case R.id.menu_key_list_multi_export: {
ids = mAdapter.getCurrentSelectedMasterKeyIds();
showMultiExportDialog(ids);
break;
}
case R.id.menu_key_list_multi_select_all: {
// select all
for (int i = 0; i < mStickyList.getCount(); i++) {
mStickyList.setItemChecked(i, true);
}
break;
}
}
return true;
}
@Override
public void onDestroyActionMode(ActionMode mode) {
mActionMode = null;
mAdapter.clearSelection();
}
@Override
public void onItemCheckedStateChanged(ActionMode mode, int position, long id,
boolean checked) {
if (checked) {
mAdapter.setNewSelection(position, true);
} else {
mAdapter.removeSelection(position);
}
int count = mStickyList.getCheckedItemCount();
String keysSelected = getResources().getQuantityString(
R.plurals.key_list_selected_keys, count, count);
mode.setTitle(keysSelected);
}
});
// We have a menu item to show in action bar.
setHasOptionsMenu(true);
@ -284,8 +177,9 @@ public class KeyListFragment extends LoaderFragment
setContentShown(false);
// Create an empty adapter we will use to display the loaded data.
mAdapter = new KeyListAdapter(getActivity(), null, 0);
mStickyList.setAdapter(mAdapter);
mAdapter = new KeyListAdapter(getActivity());
mAdapter.setOnClickListener(this);
mRecyclerView.setAdapter(mAdapter);
// Prepare the loader. Either re-connect with an existing one,
// or start a new one.
@ -304,14 +198,6 @@ public class KeyListFragment extends LoaderFragment
KeyRings.HAS_DUPLICATE_USER_ID,
};
static final int INDEX_MASTER_KEY_ID = 1;
static final int INDEX_USER_ID = 2;
static final int INDEX_IS_REVOKED = 3;
static final int INDEX_IS_EXPIRED = 4;
static final int INDEX_VERIFIED = 5;
static final int INDEX_HAS_ANY_SECRET = 6;
static final int INDEX_HAS_DUPLICATE_USER_ID = 7;
static final String ORDER =
KeyRings.HAS_ANY_SECRET + " DESC, UPPER(" + KeyRings.USER_ID + ") ASC";
@ -349,10 +235,17 @@ public class KeyListFragment extends LoaderFragment
mAdapter.setSearchQuery(mQuery);
mAdapter.swapCursor(data);
mStickyList.setAdapter(mAdapter);
mRecyclerView.setAdapter(mAdapter);
// this view is made visible if no data is available
mStickyList.setEmptyView(getActivity().findViewById(R.id.key_list_empty));
LinearLayout emptyLayout = (LinearLayout) getActivity().findViewById(R.id.key_list_empty);
if (mAdapter.getItemCount() == 0) {
mRecyclerView.setVisibility(View.GONE);
emptyLayout.setVisibility(View.VISIBLE);
} else {
emptyLayout.setVisibility(View.GONE);
mRecyclerView.setVisibility(View.VISIBLE);
}
// end action mode, if any
if (mActionMode != null) {
@ -375,15 +268,38 @@ public class KeyListFragment extends LoaderFragment
mAdapter.swapCursor(null);
}
/**
* On click on item, start key view activity
*/
@Override
public void onItemClick(AdapterView<?> adapterView, View view, int position, long id) {
Intent viewIntent = new Intent(getActivity(), ViewKeyActivity.class);
viewIntent.setData(
KeyRings.buildGenericKeyRingUri(mAdapter.getMasterKeyId(position)));
startActivity(viewIntent);
public void onImportClick() {
importFile();
}
@Override
public void onKeySlingerClick(KeyListAdapter.KeyHolder holder, int position) {
Intent safeSlingerIntent = new Intent(getActivity(), SafeSlingerActivity.class);
safeSlingerIntent.putExtra(SafeSlingerActivity.EXTRA_MASTER_KEY_ID, holder.getMasterKeyId());
startActivityForResult(safeSlingerIntent, 0);
}
@Override
public void onKeyClick(KeyListAdapter.KeyHolder holder, int position) {
if (mActionMode == null) {
Intent viewIntent = new Intent(getActivity(), ViewKeyActivity.class);
viewIntent.setData(KeychainContract.KeyRings.buildGenericKeyRingUri(holder.getMasterKeyId()));
startActivity(viewIntent);
} else {
mAdapter.toggleSelection(position);
updateSelectionCount();
}
}
@Override
public void onKeyLongClick(KeyListAdapter.KeyHolder holder, int position) {
if (mActionMode == null) {
((ActionBarActivity) getActivity()).startSupportActionMode(this);
}
mAdapter.toggleSelection(position);
updateSelectionCount();
}
protected void encrypt(ActionMode mode, long[] masterKeyIds) {
@ -787,292 +703,62 @@ public class KeyListFragment extends LoaderFragment
anim.start();
}
/**
* Implements StickyListHeadersAdapter from library
*/
private class KeyListAdapter extends CursorAdapter implements StickyListHeadersAdapter {
private String mQuery;
private LayoutInflater mInflater;
private HashMap<Integer, Boolean> mSelection = new HashMap<>();
public KeyListAdapter(Context context, Cursor c, int flags) {
super(context, c, flags);
mInflater = LayoutInflater.from(context);
}
public void setSearchQuery(String query) {
mQuery = query;
}
@Override
public Cursor swapCursor(Cursor newCursor) {
return super.swapCursor(newCursor);
}
private class ItemViewHolder {
Long mMasterKeyId;
TextView mMainUserId;
TextView mMainUserIdRest;
ImageView mStatus;
View mSlinger;
ImageButton mSlingerButton;
}
@Override
public View newView(Context context, Cursor cursor, ViewGroup parent) {
View view = mInflater.inflate(R.layout.key_list_item, parent, false);
final ItemViewHolder holder = new ItemViewHolder();
holder.mMainUserId = (TextView) view.findViewById(R.id.key_list_item_name);
holder.mMainUserIdRest = (TextView) view.findViewById(R.id.key_list_item_email);
holder.mStatus = (ImageView) view.findViewById(R.id.key_list_item_status_icon);
holder.mSlinger = view.findViewById(R.id.key_list_item_slinger_view);
holder.mSlingerButton = (ImageButton) view.findViewById(R.id.key_list_item_slinger_button);
holder.mSlingerButton.setColorFilter(context.getResources().getColor(R.color.tertiary_text_light),
PorterDuff.Mode.SRC_IN);
view.setTag(holder);
view.findViewById(R.id.key_list_item_slinger_button).setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (holder.mMasterKeyId != null) {
Intent safeSlingerIntent = new Intent(getActivity(), SafeSlingerActivity.class);
safeSlingerIntent.putExtra(SafeSlingerActivity.EXTRA_MASTER_KEY_ID, holder.mMasterKeyId);
startActivityForResult(safeSlingerIntent, REQUEST_ACTION);
}
}
});
return view;
}
/**
* Bind cursor data to the item list view
*/
@Override
public void bindView(View view, Context context, Cursor cursor) {
Highlighter highlighter = new Highlighter(context, mQuery);
ItemViewHolder h = (ItemViewHolder) view.getTag();
{ // set name and stuff, common to both key types
String userId = cursor.getString(INDEX_USER_ID);
KeyRing.UserId userIdSplit = KeyRing.splitUserId(userId);
if (userIdSplit.name != null) {
h.mMainUserId.setText(highlighter.highlight(userIdSplit.name));
} else {
h.mMainUserId.setText(R.string.user_id_no_name);
}
if (userIdSplit.email != null) {
h.mMainUserIdRest.setText(highlighter.highlight(userIdSplit.email));
h.mMainUserIdRest.setVisibility(View.VISIBLE);
} else {
h.mMainUserIdRest.setVisibility(View.GONE);
}
}
{ // set edit button and status, specific by key type
long masterKeyId = cursor.getLong(INDEX_MASTER_KEY_ID);
boolean isSecret = cursor.getInt(INDEX_HAS_ANY_SECRET) != 0;
boolean isRevoked = cursor.getInt(INDEX_IS_REVOKED) > 0;
boolean isExpired = cursor.getInt(INDEX_IS_EXPIRED) != 0;
boolean isVerified = cursor.getInt(INDEX_VERIFIED) > 0;
boolean hasDuplicate = cursor.getInt(INDEX_HAS_DUPLICATE_USER_ID) == 1;
h.mMasterKeyId = masterKeyId;
// Note: order is important!
if (isRevoked) {
KeyFormattingUtils.setStatusImage(getActivity(), h.mStatus, null, State.REVOKED, R.color.bg_gray);
h.mStatus.setVisibility(View.VISIBLE);
h.mSlinger.setVisibility(View.GONE);
h.mMainUserId.setTextColor(context.getResources().getColor(R.color.bg_gray));
h.mMainUserIdRest.setTextColor(context.getResources().getColor(R.color.bg_gray));
} else if (isExpired) {
KeyFormattingUtils.setStatusImage(getActivity(), h.mStatus, null, State.EXPIRED, R.color.bg_gray);
h.mStatus.setVisibility(View.VISIBLE);
h.mSlinger.setVisibility(View.GONE);
h.mMainUserId.setTextColor(context.getResources().getColor(R.color.bg_gray));
h.mMainUserIdRest.setTextColor(context.getResources().getColor(R.color.bg_gray));
} else if (isSecret) {
h.mStatus.setVisibility(View.GONE);
h.mSlinger.setVisibility(View.VISIBLE);
h.mMainUserId.setTextColor(context.getResources().getColor(R.color.black));
h.mMainUserIdRest.setTextColor(context.getResources().getColor(R.color.black));
} else {
// this is a public key - show if it's verified
if (isVerified) {
KeyFormattingUtils.setStatusImage(getActivity(), h.mStatus, State.VERIFIED);
h.mStatus.setVisibility(View.VISIBLE);
} else {
KeyFormattingUtils.setStatusImage(getActivity(), h.mStatus, State.UNVERIFIED);
h.mStatus.setVisibility(View.VISIBLE);
}
h.mSlinger.setVisibility(View.GONE);
h.mMainUserId.setTextColor(context.getResources().getColor(R.color.black));
h.mMainUserIdRest.setTextColor(context.getResources().getColor(R.color.black));
}
}
}
public boolean isSecretAvailable(int id) {
if (!mCursor.moveToPosition(id)) {
throw new IllegalStateException("couldn't move cursor to position " + id);
}
return mCursor.getInt(INDEX_HAS_ANY_SECRET) != 0;
}
public long getMasterKeyId(int id) {
if (!mCursor.moveToPosition(id)) {
throw new IllegalStateException("couldn't move cursor to position " + id);
}
return mCursor.getLong(INDEX_MASTER_KEY_ID);
}
/**
* Creates a new header view and binds the section headers to it. It uses the ViewHolder
* pattern. Most functionality is similar to getView() from Android's CursorAdapter.
* <p/>
* NOTE: The variables mDataValid and mCursor are available due to the super class
* CursorAdapter.
*/
@Override
public View getHeaderView(int position, View convertView, ViewGroup parent) {
HeaderViewHolder holder;
if (convertView == null) {
holder = new HeaderViewHolder();
convertView = mInflater.inflate(R.layout.key_list_header, parent, false);
holder.mText = (TextView) convertView.findViewById(R.id.stickylist_header_text);
holder.mCount = (TextView) convertView.findViewById(R.id.contacts_num);
convertView.setTag(holder);
} else {
holder = (HeaderViewHolder) convertView.getTag();
}
if (!mDataValid) {
// no data available at this point
Log.d(Constants.TAG, "getHeaderView: No data available at this point!");
return convertView;
}
if (!mCursor.moveToPosition(position)) {
throw new IllegalStateException("couldn't move cursor to position " + position);
}
if (mCursor.getInt(KeyListFragment.INDEX_HAS_ANY_SECRET) != 0) {
{ // set contact count
int num = mCursor.getCount();
String contactsTotal = getResources().getQuantityString(R.plurals.n_keys, num, num);
holder.mCount.setText(contactsTotal);
holder.mCount.setVisibility(View.VISIBLE);
}
holder.mText.setText(convertView.getResources().getString(R.string.my_keys));
return convertView;
}
// set header text as first char in user id
String userId = mCursor.getString(KeyListFragment.INDEX_USER_ID);
String headerText = convertView.getResources().getString(R.string.user_id_no_name);
if (userId != null && userId.length() > 0) {
headerText = "" + userId.charAt(0);
}
holder.mText.setText(headerText);
holder.mCount.setVisibility(View.GONE);
return convertView;
}
/**
* Header IDs should be static, position=1 should always return the same Id that is.
*/
@Override
public long getHeaderId(int position) {
if (!mDataValid) {
// no data available at this point
Log.d(Constants.TAG, "getHeaderView: No data available at this point!");
return -1;
}
if (!mCursor.moveToPosition(position)) {
throw new IllegalStateException("couldn't move cursor to position " + position);
}
// early breakout: all secret keys are assigned id 0
if (mCursor.getInt(KeyListFragment.INDEX_HAS_ANY_SECRET) != 0) {
return 1L;
}
// otherwise, return the first character of the name as ID
String userId = mCursor.getString(KeyListFragment.INDEX_USER_ID);
if (userId != null && userId.length() > 0) {
return Character.toUpperCase(userId.charAt(0));
} else {
return Long.MAX_VALUE;
}
}
private class HeaderViewHolder {
TextView mText;
TextView mCount;
}
/**
* -------------------------- MULTI-SELECTION METHODS --------------
*/
public void setNewSelection(int position, boolean value) {
mSelection.put(position, value);
notifyDataSetChanged();
}
public boolean isAnySecretSelected() {
for (int pos : mSelection.keySet()) {
if (mAdapter.isSecretAvailable(pos))
return true;
}
return false;
}
public long[] getCurrentSelectedMasterKeyIds() {
long[] ids = new long[mSelection.size()];
int i = 0;
// get master key ids
for (int pos : mSelection.keySet()) {
ids[i++] = mAdapter.getMasterKeyId(pos);
}
return ids;
}
public void removeSelection(int position) {
mSelection.remove(position);
notifyDataSetChanged();
}
public void clearSelection() {
mSelection.clear();
notifyDataSetChanged();
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
// let the adapter handle setting up the row views
View v = super.getView(position, convertView, parent);
/**
* Change color for multi-selection
*/
if (mSelection.get(position) != null) {
// selected position color
v.setBackgroundColor(parent.getResources().getColor(R.color.emphasis));
} else {
// default color
v.setBackgroundColor(Color.TRANSPARENT);
}
return v;
}
@Override
public boolean onCreateActionMode(ActionMode actionMode, Menu menu) {
MenuInflater inflater = getActivity().getMenuInflater();
inflater.inflate(R.menu.key_list_multi, menu);
mActionMode = actionMode;
return true;
}
@Override
public boolean onPrepareActionMode(ActionMode actionMode, Menu menu) {
return false;
}
@Override
public boolean onActionItemClicked(ActionMode actionMode, MenuItem menuItem) {
// get IDs for checked positions as long array
long[] ids;
switch (menuItem.getItemId()) {
case R.id.menu_key_list_multi_encrypt: {
ids = mAdapter.getCurrentSelectedMasterKeyIds();
encrypt(actionMode, ids);
break;
}
case R.id.menu_key_list_multi_delete: {
ids = mAdapter.getCurrentSelectedMasterKeyIds();
showDeleteKeyDialog(actionMode, ids, mAdapter.isAnySecretSelected());
break;
}
case R.id.menu_key_list_multi_export: {
ids = mAdapter.getCurrentSelectedMasterKeyIds();
showMultiExportDialog(ids);
break;
}
case R.id.menu_key_list_multi_select_all: {
mAdapter.selectAll();
updateSelectionCount();
break;
}
}
return true;
}
@Override
public void onDestroyActionMode(ActionMode actionMode) {
mActionMode = null;
mAdapter.clearSelection();
}
private void updateSelectionCount() {
int count = mAdapter.getSelectionCount();
if (count == 0) {
mActionMode.finish();
} else {
mActionMode.setTitle(getResources().getQuantityString(R.plurals.key_list_selected_keys, count, count));
}
}
}

View File

@ -0,0 +1,491 @@
/*
* Copyright (C) 2013-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.Color;
import android.graphics.PorterDuff;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageButton;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;
import com.tonicartos.superslim.LayoutManager;
import com.tonicartos.superslim.LinearSLM;
import org.sufficientlysecure.keychain.R;
import org.sufficientlysecure.keychain.pgp.KeyRing;
import org.sufficientlysecure.keychain.ui.util.Highlighter;
import org.sufficientlysecure.keychain.ui.util.KeyFormattingUtils;
import java.util.ArrayList;
public class KeyListAdapter extends RecyclerCursorAdapter {
static final int INDEX_MASTER_KEY_ID = 1;
static final int INDEX_USER_ID = 2;
static final int INDEX_IS_REVOKED = 3;
static final int INDEX_IS_EXPIRED = 4;
static final int INDEX_VERIFIED = 5;
static final int INDEX_HAS_ANY_SECRET = 6;
static final int INDEX_HAS_DUPLICATE_USER_ID = 7;
private Context mContext;
private ArrayList<Item> mItemList = new ArrayList<>();
private OnClickListener mOnClickListener;
private String mQuery;
public KeyListAdapter(Context context) {
super(null);
mContext = context;
setOnCursorSwappedListener(new OnCursorSwappedListener() {
@Override
public void onCursorSwapped(Cursor oldCursor, Cursor newCursor) {
mItemList = new ArrayList<>();
if (newCursor == null) {
return;
}
int count = newCursor.getCount();
if (count == 0) {
return;
}
mItemList.add(new Item(Item.TYPE_MY_KEYS_HEADER, 0));
int i;
for (i = 0; i < count; i++) {
if (newCursor.moveToPosition(i) && newCursor.getInt(INDEX_HAS_ANY_SECRET) != 0) {
mItemList.add(new Item(Item.TYPE_KEY, 0, i));
} else {
break;
}
}
if (mItemList.size() == 1) {
if (mQuery == null || mQuery.isEmpty()) {
mItemList.add(new Item(Item.TYPE_IMPORT, 0));
} else {
mItemList.clear();
}
}
char prevHeaderChar = '\0';
int prevHeaderIndex = 0;
for (; i < count; i++) {
if (newCursor.moveToPosition(i)) {
char headerChar = Character.toUpperCase(newCursor.getString(INDEX_USER_ID).charAt(0));
if (headerChar != prevHeaderChar) {
prevHeaderChar = headerChar;
prevHeaderIndex = mItemList.size();
mItemList.add(new Item(Item.TYPE_CHAR_HEADER, prevHeaderIndex));
}
mItemList.add(new Item(Item.TYPE_KEY, prevHeaderIndex, i));
}
}
mItemList.add(new Item(Item.TYPE_FOOTER, mItemList.size() - 1));
}
});
}
@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
RecyclerView.ViewHolder viewHolder = null;
switch (viewType) {
case Item.TYPE_MY_KEYS_HEADER:
case Item.TYPE_CHAR_HEADER:
viewHolder = new HeaderHolder(LayoutInflater.from(mContext)
.inflate(R.layout.key_list_header, parent, false));
break;
case Item.TYPE_IMPORT:
viewHolder = new ImportKeyHolder(LayoutInflater.from(mContext)
.inflate(R.layout.key_list_import, parent, false));
break;
case Item.TYPE_KEY:
viewHolder = new KeyHolder(LayoutInflater.from(mContext)
.inflate(R.layout.key_list_key, parent, false));
break;
case Item.TYPE_FOOTER:
viewHolder = new RecyclerView.ViewHolder(LayoutInflater.from(mContext)
.inflate(R.layout.key_list_footer, parent, false)) {
};
break;
}
return viewHolder;
}
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
Item item = mItemList.get(position);
switch (item.mType) {
case Item.TYPE_MY_KEYS_HEADER:
case Item.TYPE_CHAR_HEADER:
((HeaderHolder) holder).bind(position);
break;
case Item.TYPE_KEY:
((KeyHolder) holder).bind(position);
break;
}
View itemView = holder.itemView;
LayoutManager.LayoutParams layoutParams = new LayoutManager.LayoutParams(itemView.getLayoutParams());
layoutParams.setSlm(LinearSLM.ID);
layoutParams.setFirstPosition(item.mHeaderIndex);
itemView.setLayoutParams(layoutParams);
}
@Override
public int getItemViewType(int position) {
return mItemList.get(position).mType;
}
@Override
public int getItemCount() {
return mItemList.size();
}
public void toggleSelection(int position) {
Item item = mItemList.get(position);
item.mSelected = item.mSelectable && !item.mSelected;
notifyDataSetChanged();
}
public void selectAll() {
for (Item item : mItemList) {
item.mSelected = item.mSelectable;
}
notifyDataSetChanged();
}
public void clearSelection() {
for (Item item : mItemList) {
item.mSelected = false;
}
notifyDataSetChanged();
}
public int getSelectionCount() {
int count = 0;
for (Item item : mItemList) {
if (item.mSelected) {
count++;
}
}
return count;
}
public boolean isAnySecretSelected() {
for (Item item : mItemList) {
if (item.mSelected
&& mCursor.moveToPosition(item.mCursorPosition)
&& mCursor.getInt(INDEX_HAS_ANY_SECRET) != 0) {
return true;
}
}
return false;
}
public long[] getCurrentSelectedMasterKeyIds() {
long[] ids = new long[getSelectionCount()];
int i = 0;
for (Item item : mItemList) {
if (item.mSelected
&& mCursor.moveToPosition(item.mCursorPosition)) {
ids[i++] = mCursor.getLong(INDEX_MASTER_KEY_ID);
}
}
return ids;
}
public void setOnClickListener(OnClickListener onClickListener) {
mOnClickListener = onClickListener;
}
public void setSearchQuery(String query) {
mQuery = query;
}
private class Item {
public static final int TYPE_MY_KEYS_HEADER = 0;
public static final int TYPE_CHAR_HEADER = 1;
public static final int TYPE_IMPORT = 2;
public static final int TYPE_KEY = 3;
public static final int TYPE_FOOTER = 4;
private int mType, mHeaderIndex, mCursorPosition;
private boolean mSelectable, mSelected;
private Item(int type, int headerIndex, int cursorPosition) {
mType = type;
mHeaderIndex = headerIndex;
mCursorPosition = cursorPosition;
mSelectable = cursorPosition != -1;
}
private Item(int type, int headerIndex) {
this(type, headerIndex, -1);
}
}
public class HeaderHolder extends RecyclerView.ViewHolder {
private TextView mTextView, mCountTextView;
private HeaderHolder(View itemView) {
super(itemView);
mTextView = (TextView) itemView.findViewById(R.id.key_list_header_text);
mCountTextView = (TextView) itemView.findViewById(R.id.key_list_header_count);
}
private void bind(int position) {
Item item = mItemList.get(position);
switch (item.mType) {
case Item.TYPE_MY_KEYS_HEADER: {
mTextView.setText(mContext.getString(R.string.my_keys));
int count = mCursor.getCount();
mCountTextView.setText(mContext.getResources().getQuantityString(R.plurals.n_keys, count, count));
mCountTextView.setVisibility(View.VISIBLE);
break;
}
case Item.TYPE_CHAR_HEADER: {
Item nextItem = mItemList.get(position + 1);
mCursor.moveToPosition(nextItem.mCursorPosition);
String userId = mCursor.getString(INDEX_USER_ID),
text = mContext.getString(R.string.user_id_no_name);
if (userId != null && !userId.isEmpty()) {
text = String.valueOf(Character.toUpperCase(userId.charAt(0)));
}
mTextView.setText(text);
if (position == 0) {
int count = mCursor.getCount();
mCountTextView.setText(mContext.getResources().getQuantityString(R.plurals.n_keys, count, count));
mCountTextView.setVisibility(View.VISIBLE);
} else {
mCountTextView.setVisibility(View.GONE);
}
break;
}
}
}
}
public class ImportKeyHolder extends RecyclerView.ViewHolder {
public ImportKeyHolder(View itemView) {
super(itemView);
itemView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mOnClickListener.onImportClick();
}
});
}
}
public class KeyHolder extends RecyclerView.ViewHolder {
private View mDividerView;
private TextView mNameTextView, mEmailTextView;
private LinearLayout mSlingerLayout;
private ImageButton mSlingerImageButton;
private ImageView mStatusImageView;
private int mPosition;
private long mMasterKeyId;
private KeyHolder(View itemView) {
super(itemView);
mDividerView = itemView.findViewById(R.id.key_list_key_divider);
mNameTextView = (TextView) itemView.findViewById(R.id.key_list_key_name);
mEmailTextView = (TextView) itemView.findViewById(R.id.key_list_key_email);
mSlingerLayout = (LinearLayout) itemView.findViewById(R.id.key_list_key_slinger_view);
mSlingerImageButton = (ImageButton) itemView.findViewById(R.id.key_list_key_slinger_button);
mStatusImageView = (ImageView) itemView.findViewById(R.id.key_list_key_status_icon);
itemView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mOnClickListener.onKeyClick(KeyHolder.this, mPosition);
}
});
itemView.setOnLongClickListener(new View.OnLongClickListener() {
@Override
public boolean onLongClick(View v) {
mOnClickListener.onKeyLongClick(KeyHolder.this, mPosition);
return true;
}
});
mSlingerImageButton.setColorFilter(mContext.getResources().getColor(R.color.tertiary_text_light),
PorterDuff.Mode.SRC_IN);
mSlingerImageButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mOnClickListener.onKeySlingerClick(KeyHolder.this, mPosition);
}
});
}
private void bind(int position) {
mPosition = position;
Item item = mItemList.get(position);
mCursor.moveToPosition(item.mCursorPosition);
if (item.mSelected) {
// selected position color
itemView.setBackgroundColor(itemView.getResources().getColor(R.color.emphasis));
} else {
// default color
itemView.setBackgroundColor(Color.TRANSPARENT);
}
Item prevItem = mItemList.get(position - 1);
if (prevItem.mType == Item.TYPE_MY_KEYS_HEADER
|| prevItem.mType == Item.TYPE_CHAR_HEADER) {
mDividerView.setVisibility(View.GONE);
} else {
mDividerView.setVisibility(View.VISIBLE);
}
Highlighter highlighter = new Highlighter(mContext, mQuery);
// set name and stuff, common to both key types
String userId = mCursor.getString(INDEX_USER_ID);
KeyRing.UserId userIdSplit = KeyRing.splitUserId(userId);
if (userIdSplit.name != null) {
mNameTextView.setText(highlighter.highlight(userIdSplit.name));
} else {
mNameTextView.setText(R.string.user_id_no_name);
}
if (userIdSplit.email != null) {
mEmailTextView.setText(highlighter.highlight(userIdSplit.email));
mEmailTextView.setVisibility(View.VISIBLE);
} else {
mEmailTextView.setVisibility(View.GONE);
}
// set edit button and status, specific by key type
long masterKeyId = mCursor.getLong(INDEX_MASTER_KEY_ID);
boolean isSecret = mCursor.getInt(INDEX_HAS_ANY_SECRET) != 0;
boolean isRevoked = mCursor.getInt(INDEX_IS_REVOKED) > 0;
boolean isExpired = mCursor.getInt(INDEX_IS_EXPIRED) != 0;
boolean isVerified = mCursor.getInt(INDEX_VERIFIED) > 0;
boolean hasDuplicate = mCursor.getInt(INDEX_HAS_DUPLICATE_USER_ID) == 1;
mMasterKeyId = masterKeyId;
// Note: order is important!
if (isRevoked) {
KeyFormattingUtils.setStatusImage(mContext, mStatusImageView, null, KeyFormattingUtils.State.REVOKED, R.color.bg_gray);
mStatusImageView.setVisibility(View.VISIBLE);
mSlingerLayout.setVisibility(View.GONE);
mNameTextView.setTextColor(mContext.getResources().getColor(R.color.bg_gray));
mEmailTextView.setTextColor(mContext.getResources().getColor(R.color.bg_gray));
} else if (isExpired) {
KeyFormattingUtils.setStatusImage(mContext, mStatusImageView, null, KeyFormattingUtils.State.EXPIRED, R.color.bg_gray);
mStatusImageView.setVisibility(View.VISIBLE);
mSlingerLayout.setVisibility(View.GONE);
mNameTextView.setTextColor(mContext.getResources().getColor(R.color.bg_gray));
mEmailTextView.setTextColor(mContext.getResources().getColor(R.color.bg_gray));
} else if (isSecret) {
mStatusImageView.setVisibility(View.GONE);
mSlingerLayout.setVisibility(View.VISIBLE);
mNameTextView.setTextColor(mContext.getResources().getColor(R.color.black));
mEmailTextView.setTextColor(mContext.getResources().getColor(R.color.black));
} else {
// this is a public key - show if it's verified
if (isVerified) {
KeyFormattingUtils.setStatusImage(mContext, mStatusImageView, KeyFormattingUtils.State.VERIFIED);
mStatusImageView.setVisibility(View.VISIBLE);
} else {
KeyFormattingUtils.setStatusImage(mContext, mStatusImageView, KeyFormattingUtils.State.UNVERIFIED);
mStatusImageView.setVisibility(View.VISIBLE);
}
mSlingerLayout.setVisibility(View.GONE);
mNameTextView.setTextColor(mContext.getResources().getColor(R.color.black));
mEmailTextView.setTextColor(mContext.getResources().getColor(R.color.black));
}
}
public long getMasterKeyId() {
return mMasterKeyId;
}
}
public interface OnClickListener {
public void onImportClick();
public void onKeySlingerClick(KeyHolder holder, int position);
public void onKeyClick(KeyHolder holder, int position);
public void onKeyLongClick(KeyHolder holder, int position);
}
}

View File

@ -0,0 +1,67 @@
/*
* Copyright (C) 2013-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.database.Cursor;
import android.support.v7.widget.RecyclerView;
public abstract class RecyclerCursorAdapter<VH extends RecyclerView.ViewHolder> extends RecyclerView.Adapter<VH> {
protected Cursor mCursor;
private OnCursorSwappedListener mOnCursorSwappedListener;
public RecyclerCursorAdapter(Cursor cursor) {
mCursor = cursor;
notifyDataSetChanged();
}
/**
* Swap in a new Cursor, returning the old Cursor. The returned old Cursor is not closed.
*
* @param newCursor The new cursor to be used.
* @return Returns the previously set Cursor, or null if there was not one.
* If the given new Cursor is the same instance is the previously set
* Cursor, null is also returned.
*/
public Cursor swapCursor(Cursor newCursor) {
if (newCursor == mCursor) {
return null;
}
Cursor oldCursor = mCursor;
mCursor = newCursor;
mOnCursorSwappedListener.onCursorSwapped(oldCursor, newCursor);
notifyDataSetChanged();
return oldCursor;
}
public void setOnCursorSwappedListener(OnCursorSwappedListener mOnCursorSwappedListener) {
this.mOnCursorSwappedListener = mOnCursorSwappedListener;
}
public interface OnCursorSwappedListener {
public void onCursorSwapped(Cursor oldCursor, Cursor newCursor);
}
}

View File

@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="72dp" />

View File

@ -10,14 +10,13 @@
android:layout_width="match_parent"
android:layout_height="match_parent">
<se.emilsjolander.stickylistheaders.StickyListHeadersListView
android:id="@+id/key_list_list"
<android.support.v7.widget.RecyclerView
android:id="@+id/key_list_recycler"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:drawSelectorOnTop="true"
android:fastScrollEnabled="true"
android:paddingLeft="16dp"
android:paddingRight="32dp"
android:paddingRight="16dp"
android:scrollbars="vertical"
android:scrollbarStyle="outsideOverlay" />
<LinearLayout

View File

@ -1,22 +1,26 @@
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="wrap_content" >
android:layout_height="wrap_content"
android:background="@color/background_material_light"
app:slm_headerDisplay="inline|sticky"
app:slm_isHeader="true">
<TextView
style="@style/SectionHeader"
android:id="@+id/stickylist_header_text"
android:id="@+id/key_list_header_text"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="start|left"
android:text="header text" />
<TextView
android:id="@+id/key_list_header_count"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textAppearance="?android:attr/textAppearanceSmall"
android:text="key count"
android:id="@+id/contacts_num"
android:layout_centerVertical="true"
android:layout_alignParentRight="true"
android:layout_alignParentEnd="true"

View File

@ -0,0 +1,21 @@
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:padding="4dp"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:clickable="true"
android:minHeight="?android:attr/listPreferredItemHeight"
style="?android:attr/borderlessButtonStyle">
<TextView
android:paddingLeft="8dp"
android:paddingRight="8dp"
android:textAppearance="?android:attr/textAppearanceMedium"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_gravity="center"
android:text="@string/btn_import_from_file"
android:drawableLeft="@drawable/ic_folder_grey_24dp"
android:drawablePadding="8dp"
android:gravity="center" />
</FrameLayout>

View File

@ -1,75 +0,0 @@
<?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:gravity="center_vertical"
android:singleLine="true"
android:orientation="horizontal"
android:descendantFocusability="blocksDescendants"
android:focusable="false">
<LinearLayout
android:layout_width="0dip"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:layout_weight="1"
android:focusable="true"
android:orientation="vertical"
android:paddingLeft="8dp"
android:paddingRight="4dp"
android:paddingTop="4dp"
android:paddingBottom="4dp">
<TextView
android:id="@+id/key_list_item_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/label_main_user_id"
android:textAppearance="?android:attr/textAppearanceMedium" />
<TextView
android:id="@+id/key_list_item_email"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:singleLine="true"
android:ellipsize="end"
android:text="user@example.com"
android:textAppearance="?android:attr/textAppearanceSmall" />
</LinearLayout>
<LinearLayout
android:id="@+id/key_list_item_slinger_view"
android:layout_width="wrap_content"
android:layout_height="?android:attr/listPreferredItemHeight"
android:layout_gravity="center_vertical"
android:orientation="horizontal">
<View
android:layout_width="1dip"
android:layout_height="match_parent"
android:gravity="right"
android:layout_marginBottom="8dp"
android:layout_marginTop="8dp"
android:background="?android:attr/listDivider" />
<ImageButton
android:id="@+id/key_list_item_slinger_button"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_gravity="center"
android:src="@drawable/ic_repeat_grey_24dp"
android:padding="12dp"
android:background="?android:selectableItemBackground" />
</LinearLayout>
<ImageView
android:id="@+id/key_list_item_status_icon"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:src="@drawable/status_signature_revoked_cutout_24dp"
android:padding="16dp" />
</LinearLayout>

View File

@ -0,0 +1,89 @@
<?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:orientation="vertical">
<View
android:id="@+id/key_list_key_divider"
android:layout_width="match_parent"
android:layout_height="1dip"
android:background="?android:attr/listDivider" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="?android:attr/selectableItemBackground"
android:minHeight="?android:attr/listPreferredItemHeight"
android:gravity="center_vertical"
android:singleLine="true"
android:orientation="horizontal"
android:descendantFocusability="blocksDescendants"
android:focusable="false">
<LinearLayout
android:layout_width="0dip"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:layout_weight="1"
android:focusable="true"
android:orientation="vertical"
android:paddingLeft="8dp"
android:paddingRight="4dp"
android:paddingTop="4dp"
android:paddingBottom="4dp">
<TextView
android:id="@+id/key_list_key_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/label_main_user_id"
android:textAppearance="?android:attr/textAppearanceMedium" />
<TextView
android:id="@+id/key_list_key_email"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:singleLine="true"
android:ellipsize="end"
android:text="user@example.com"
android:textAppearance="?android:attr/textAppearanceSmall" />
</LinearLayout>
<LinearLayout
android:id="@+id/key_list_key_slinger_view"
android:layout_width="wrap_content"
android:layout_height="?android:attr/listPreferredItemHeight"
android:layout_gravity="center_vertical"
android:orientation="horizontal">
<View
android:layout_width="1dip"
android:layout_height="match_parent"
android:gravity="right"
android:layout_marginBottom="8dp"
android:layout_marginTop="8dp"
android:background="?android:attr/listDivider" />
<ImageButton
android:id="@+id/key_list_key_slinger_button"
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_gravity="center"
android:src="@drawable/ic_repeat_grey_24dp"
android:padding="12dp"
android:background="?android:selectableItemBackground" />
</LinearLayout>
<ImageView
android:id="@+id/key_list_key_status_icon"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:src="@drawable/status_signature_revoked_cutout_24dp"
android:padding="16dp" />
</LinearLayout>
</LinearLayout>

View File

@ -1,6 +1,7 @@
<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:tools="http://schemas.android.com/tools"
xmlns:android="http://schemas.android.com/apk/res/android">
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<item
android:id="@+id/menu_key_list_multi_encrypt"
@ -9,19 +10,19 @@
<item
android:id="@+id/menu_key_list_multi_export"
android:showAsAction="never"
app:showAsAction="never"
tools:ignore="AppCompatResource"
android:title="@string/menu_export_key" />
<item
android:id="@+id/menu_key_list_multi_delete"
android:showAsAction="never"
app:showAsAction="never"
tools:ignore="AppCompatResource"
android:title="@string/menu_delete_key" />
<item
android:id="@+id/menu_key_list_multi_select_all"
android:showAsAction="never"
app:showAsAction="never"
tools:ignore="AppCompatResource"
android:title="@string/menu_select_all" />

View File

@ -85,6 +85,7 @@
<string name="btn_encrypt_files">"Encrypt files"</string>
<string name="btn_encrypt_text">"Encrypt text"</string>
<string name="btn_add_email">"Add additional email address"</string>
<string name="btn_import_from_file">Import from file</string>
<!-- menu -->
<string name="menu_preferences">"Settings"</string>

@ -1 +0,0 @@
Subproject commit 70a2ed80632938540bf07b81270384f4e5a96f9e