260 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
		
		
			
		
	
	
			260 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
|  | /*
 | ||
|  |  * 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.DialogInterface;
 | ||
|  | import android.content.Intent;
 | ||
|  | import android.graphics.Bitmap;
 | ||
|  | import android.os.Bundle;
 | ||
|  | import android.support.v4.app.Fragment;
 | ||
|  | import android.support.v4.view.MenuItemCompat;
 | ||
|  | import android.support.v7.widget.ShareActionProvider;
 | ||
|  | 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.CompoundButton;
 | ||
|  | 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.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 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";
 | ||
|  | 
 | ||
|  |     private static final int QR_CODE_SIZE = 350;
 | ||
|  | 
 | ||
|  |     private ShareActionProvider shareActionProvider;
 | ||
|  |     /**
 | ||
|  |      * 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.delete, GoogleMaterial.Icon.gmd_delete);
 | ||
|  |         Drawables.addIcon(getActivity(), menu, R.id.share, GoogleMaterial.Icon.gmd_share);
 | ||
|  | 
 | ||
|  |         shareActionProvider = (ShareActionProvider) MenuItemCompat.getActionProvider(
 | ||
|  |                 menu.findItem(R.id.share));
 | ||
|  | 
 | ||
|  |         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, new
 | ||
|  |                                 DialogInterface.OnClickListener() {
 | ||
|  |                                     @Override
 | ||
|  |                                     public void onClick(DialogInterface dialog, int which) {
 | ||
|  |                                         Singleton.getAddressRepository(ctx).remove(item);
 | ||
|  |                                         item = null;
 | ||
|  |                                         ctx.onBackPressed();
 | ||
|  |                                     }
 | ||
|  |                                 })
 | ||
|  |                         .setNegativeButton(android.R.string.no, null)
 | ||
|  |                         .show();
 | ||
|  |                 return true;
 | ||
|  |             case R.id.share:
 | ||
|  |                 new AlertDialog.Builder(ctx)
 | ||
|  |                         .setMessage("I have no fucking clue.")
 | ||
|  |                         .show();
 | ||
|  |             default:
 | ||
|  |                 return false;
 | ||
|  |         }
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     private void setShareIntent(Intent shareIntent) {
 | ||
|  |         if (shareActionProvider != null) {
 | ||
|  |             shareActionProvider.setShareIntent(shareIntent);
 | ||
|  |         }
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     @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(new CompoundButton.OnCheckedChangeListener() {
 | ||
|  |                     @Override
 | ||
|  |                     public void onCheckedChanged(CompoundButton buttonView, boolean 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(encodeAsBitmap(item));
 | ||
|  |         }
 | ||
|  | 
 | ||
|  |         return rootView;
 | ||
|  |     }
 | ||
|  | 
 | ||
|  |     Bitmap encodeAsBitmap(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);
 | ||
|  |         }
 | ||
|  |         super.onPause();
 | ||
|  |     }
 | ||
|  | }
 |