/* * Copyright 2015 Christian Basler * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package ch.dissem.apps.abit; import android.app.Activity; import android.app.AlertDialog; import android.content.Intent; import android.graphics.Bitmap; import android.os.Bundle; import android.support.v4.app.Fragment; import android.text.Editable; import android.text.TextWatcher; import android.view.LayoutInflater; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; import android.view.ViewGroup; import android.widget.ImageView; import android.widget.Switch; import android.widget.TextView; import com.google.zxing.BarcodeFormat; import com.google.zxing.MultiFormatWriter; import com.google.zxing.WriterException; import com.google.zxing.common.BitMatrix; import com.mikepenz.community_material_typeface_library.CommunityMaterial; import com.mikepenz.google_material_typeface_library.GoogleMaterial; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import ch.dissem.apps.abit.service.Singleton; import ch.dissem.apps.abit.util.Drawables; import ch.dissem.bitmessage.entity.BitmessageAddress; import ch.dissem.bitmessage.wif.WifExporter; import static android.graphics.Color.BLACK; import static android.graphics.Color.WHITE; /** * A fragment representing a single Message detail screen. * This fragment is either contained in a {@link MainActivity} * in two-pane mode (on tablets) or a {@link MessageDetailActivity} * on handsets. */ public class AddressDetailFragment extends Fragment { private static final Logger LOG = LoggerFactory.getLogger(AddressDetailFragment.class); /** * The fragment argument representing the item ID that this fragment * represents. */ public static final String ARG_ITEM = "item"; public static final String EXPORT_POSTFIX = ".keys.dat"; private static final int QR_CODE_SIZE = 350; /** * The content this fragment is presenting. */ private BitmessageAddress item; /** * Mandatory empty constructor for the fragment manager to instantiate the * fragment (e.g. upon screen orientation changes). */ public AddressDetailFragment() { } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); if (getArguments().containsKey(ARG_ITEM)) { // Load the dummy content specified by the fragment // arguments. In a real-world scenario, use a Loader // to load content from a content provider. item = (BitmessageAddress) getArguments().getSerializable(ARG_ITEM); } setHasOptionsMenu(true); } @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { inflater.inflate(R.menu.address, menu); Drawables.addIcon(getActivity(), menu, R.id.write_message, GoogleMaterial.Icon.gmd_mail); Drawables.addIcon(getActivity(), menu, R.id.share, GoogleMaterial.Icon.gmd_share); Drawables.addIcon(getActivity(), menu, R.id.delete, GoogleMaterial.Icon.gmd_delete); Drawables.addIcon(getActivity(), menu, R.id.export, CommunityMaterial.Icon.cmd_export) .setVisible(item != null && item.getPrivateKey() != null); super.onCreateOptionsMenu(menu, inflater); } @Override public boolean onOptionsItemSelected(MenuItem menuItem) { final Activity ctx = getActivity(); switch (menuItem.getItemId()) { case R.id.write_message: { Intent intent = new Intent(ctx, ComposeMessageActivity.class); intent.putExtra(ComposeMessageActivity.EXTRA_IDENTITY, Singleton.getIdentity(ctx)); intent.putExtra(ComposeMessageActivity.EXTRA_RECIPIENT, item); startActivity(intent); return true; } case R.id.delete: { int warning; if (item.getPrivateKey() != null) warning = R.string.delete_identity_warning; else warning = R.string.delete_contact_warning; new AlertDialog.Builder(ctx) .setMessage(warning) .setPositiveButton(android.R.string.yes, (dialog, which) -> { Singleton.getAddressRepository(ctx).remove(item); MainActivity mainActivity = MainActivity.getInstance(); if (item.getPrivateKey() != null && mainActivity != null) { mainActivity.removeIdentityEntry(item); } item = null; ctx.onBackPressed(); }) .setNegativeButton(android.R.string.no, null) .show(); return true; } case R.id.export: { new AlertDialog.Builder(ctx) .setMessage(R.string.confirm_export) .setPositiveButton(android.R.string.yes, (dialog, which) -> { Intent shareIntent = new Intent(Intent.ACTION_SEND); shareIntent.setType("text/plain"); shareIntent.putExtra(Intent.EXTRA_TITLE, item + EXPORT_POSTFIX); WifExporter exporter = new WifExporter(Singleton .getBitmessageContext(ctx)); exporter.addIdentity(item); shareIntent.putExtra(Intent.EXTRA_TEXT, exporter.toString ()); startActivity(Intent.createChooser(shareIntent, null)); }) .setNegativeButton(android.R.string.no, null) .show(); return true; } case R.id.share: { Intent shareIntent = new Intent(Intent.ACTION_SEND); shareIntent.setType("text/plain"); shareIntent.putExtra(Intent.EXTRA_TEXT, item.getAddress()); startActivity(Intent.createChooser(shareIntent, null)); } default: return false; } } @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View rootView = inflater.inflate(R.layout.fragment_address_detail, container, false); // Show the dummy content as text in a TextView. if (item != null) { ((ImageView) rootView.findViewById(R.id.avatar)).setImageDrawable(new Identicon(item)); TextView name = (TextView) rootView.findViewById(R.id.name); name.setText(item.toString()); name.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { // Nothing to do } @Override public void onTextChanged(CharSequence s, int start, int before, int count) { // Nothing to do } @Override public void afterTextChanged(Editable s) { item.setAlias(s.toString()); } }); TextView address = (TextView) rootView.findViewById(R.id.address); address.setText(item.getAddress()); address.setSelected(true); ((TextView) rootView.findViewById(R.id.stream_number)).setText(getActivity() .getString(R.string.stream_number, item.getStream())); if (item.getPrivateKey() == null) { Switch active = (Switch) rootView.findViewById(R.id.active); active.setChecked(item.isSubscribed()); active.setOnCheckedChangeListener((buttonView, isChecked) -> item.setSubscribed(isChecked)); ImageView pubkeyAvailableImg = (ImageView) rootView.findViewById(R.id .pubkey_available); if (item.getPubkey() == null) { pubkeyAvailableImg.setAlpha(0.3f); TextView pubkeyAvailableDesc = (TextView) rootView.findViewById(R.id .pubkey_available_desc); pubkeyAvailableDesc.setText(R.string.pubkey_not_available); } } else { rootView.findViewById(R.id.active).setVisibility(View.GONE); rootView.findViewById(R.id.pubkey_available).setVisibility(View.GONE); rootView.findViewById(R.id.pubkey_available_desc).setVisibility(View.GONE); } // QR code ImageView qrCode = (ImageView) rootView.findViewById(R.id.qr_code); qrCode.setImageBitmap(qrCode(item)); } return rootView; } Bitmap qrCode(BitmessageAddress address) { StringBuilder link = new StringBuilder("bitmessage:"); link.append(address.getAddress()); if (address.getAlias() != null) { link.append("?label=").append(address.getAlias()); } BitMatrix result; try { result = new MultiFormatWriter().encode(link.toString(), BarcodeFormat.QR_CODE, QR_CODE_SIZE, QR_CODE_SIZE, null); } catch (WriterException e) { LOG.error(e.getMessage(), e); return null; } int w = result.getWidth(); int h = result.getHeight(); int[] pixels = new int[w * h]; for (int y = 0; y < h; y++) { int offset = y * w; for (int x = 0; x < w; x++) { pixels[offset + x] = result.get(x, y) ? BLACK : WHITE; } } Bitmap bitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); bitmap.setPixels(pixels, 0, QR_CODE_SIZE, 0, 0, w, h); return bitmap; } @Override public void onPause() { if (item != null) { Singleton.getAddressRepository(getContext()).save(item); MainActivity mainActivity = MainActivity.getInstance(); if (mainActivity != null && item.getPrivateKey() != null) { mainActivity.updateIdentityEntry(item); } } super.onPause(); } }