ConversationFragment.java

  1package eu.siacs.conversations.ui;
  2
  3import java.io.FileNotFoundException;
  4import java.util.ArrayList;
  5import java.util.HashMap;
  6import java.util.Hashtable;
  7import java.util.LinkedList;
  8import java.util.List;
  9import java.util.Set;
 10
 11import net.java.otr4j.session.SessionStatus;
 12
 13import eu.siacs.conversations.R;
 14import eu.siacs.conversations.crypto.PgpEngine.OpenPgpException;
 15import eu.siacs.conversations.crypto.PgpEngine.UserInputRequiredException;
 16import eu.siacs.conversations.entities.Contact;
 17import eu.siacs.conversations.entities.Conversation;
 18import eu.siacs.conversations.entities.Message;
 19import eu.siacs.conversations.entities.MucOptions;
 20import eu.siacs.conversations.entities.MucOptions.OnRenameListener;
 21import eu.siacs.conversations.services.XmppConnectionService;
 22import eu.siacs.conversations.utils.PhoneHelper;
 23import eu.siacs.conversations.utils.UIHelper;
 24import android.app.AlertDialog;
 25import android.app.Fragment;
 26import android.content.DialogInterface;
 27import android.content.Intent;
 28import android.content.IntentSender;
 29import android.content.SharedPreferences;
 30import android.content.IntentSender.SendIntentException;
 31import android.graphics.Bitmap;
 32import android.graphics.BitmapFactory;
 33import android.graphics.Typeface;
 34import android.net.Uri;
 35import android.os.AsyncTask;
 36import android.os.Bundle;
 37import android.preference.PreferenceManager;
 38import android.util.Log;
 39import android.view.LayoutInflater;
 40import android.view.View;
 41import android.view.View.OnClickListener;
 42import android.view.ViewGroup;
 43import android.widget.ArrayAdapter;
 44import android.widget.EditText;
 45import android.widget.LinearLayout;
 46import android.widget.ListView;
 47import android.widget.ImageButton;
 48import android.widget.ImageView;
 49import android.widget.TextView;
 50import android.widget.Toast;
 51
 52public class ConversationFragment extends Fragment {
 53
 54	protected Conversation conversation;
 55	protected ListView messagesView;
 56	protected LayoutInflater inflater;
 57	protected List<Message> messageList = new ArrayList<Message>();
 58	protected ArrayAdapter<Message> messageListAdapter;
 59	protected Contact contact;
 60	protected BitmapCache mBitmapCache = new BitmapCache();
 61
 62	protected String queuedPqpMessage = null;
 63
 64	private EditText chatMsg;
 65
 66	protected Bitmap selfBitmap;
 67	
 68	private IntentSender askForPassphraseIntent = null;
 69
 70	private OnClickListener sendMsgListener = new OnClickListener() {
 71
 72		@Override
 73		public void onClick(View v) {
 74			if (chatMsg.getText().length() < 1)
 75				return;
 76			Message message = new Message(conversation, chatMsg.getText()
 77					.toString(), conversation.nextMessageEncryption);
 78			if (conversation.nextMessageEncryption == Message.ENCRYPTION_OTR) {
 79				sendOtrMessage(message);
 80			} else if (conversation.nextMessageEncryption == Message.ENCRYPTION_PGP) {
 81				sendPgpMessage(message);
 82			} else {
 83				sendPlainTextMessage(message);
 84			}
 85		}
 86	};
 87	protected OnClickListener clickToDecryptListener = new OnClickListener() {
 88		
 89		@Override
 90		public void onClick(View v) {
 91			Log.d("gultsch","clicked to decrypt");
 92			if (askForPassphraseIntent!=null) {
 93				try {
 94					getActivity().startIntentSenderForResult(askForPassphraseIntent, ConversationActivity.REQUEST_DECRYPT_PGP, null, 0, 0, 0);
 95				} catch (SendIntentException e) {
 96					Log.d("gultsch","couldnt fire intent");
 97				}
 98			}
 99		}
100	};
101	
102	private LinearLayout pgpInfo;
103	private LinearLayout mucError;
104	private TextView mucErrorText;
105	private OnClickListener clickToMuc = new OnClickListener() {
106		
107		@Override
108		public void onClick(View v) {
109			Intent intent = new Intent(getActivity(),MucDetailsActivity.class);
110			intent.setAction(MucDetailsActivity.ACTION_VIEW_MUC);
111			intent.putExtra("uuid", conversation.getUuid());
112			startActivity(intent);
113		}
114	};
115	
116	public void hidePgpPassphraseBox() {
117		pgpInfo.setVisibility(View.GONE);
118	}
119
120	public void updateChatMsgHint() {
121		if (conversation.getMode() == Conversation.MODE_MULTI) {
122			chatMsg.setHint("Send message to conference");
123		} else {
124			switch (conversation.nextMessageEncryption) {
125			case Message.ENCRYPTION_NONE:
126				chatMsg.setHint("Send plain text message");
127				break;
128			case Message.ENCRYPTION_OTR:
129				chatMsg.setHint("Send OTR encrypted message");
130				break;
131			case Message.ENCRYPTION_PGP:
132				chatMsg.setHint("Send openPGP encryted messeage");
133				break;
134			case Message.ENCRYPTION_DECRYPTED:
135				chatMsg.setHint("Send openPGP encryted messeage");
136				break;
137			default:
138				break;
139			}
140		}
141	}
142
143	@Override
144	public View onCreateView(final LayoutInflater inflater,
145			ViewGroup container, Bundle savedInstanceState) {
146
147		this.inflater = inflater;
148
149		final View view = inflater.inflate(R.layout.fragment_conversation,
150				container, false);
151		chatMsg = (EditText) view.findViewById(R.id.textinput);
152		ImageButton sendButton = (ImageButton) view
153				.findViewById(R.id.textSendButton);
154		sendButton.setOnClickListener(this.sendMsgListener);
155		
156		pgpInfo = (LinearLayout) view.findViewById(R.id.pgp_keyentry);
157		pgpInfo.setOnClickListener(clickToDecryptListener);
158		mucError = (LinearLayout) view.findViewById(R.id.muc_error);
159		mucError.setOnClickListener(clickToMuc );
160		mucErrorText = (TextView) view.findViewById(R.id.muc_error_msg);
161		
162		messagesView = (ListView) view.findViewById(R.id.messages_view);
163
164		messageListAdapter = new ArrayAdapter<Message>(this.getActivity()
165				.getApplicationContext(), R.layout.message_sent,
166				this.messageList) {
167
168			private static final int SENT = 0;
169			private static final int RECIEVED = 1;
170			private static final int ERROR = 2;
171
172			@Override
173			public int getViewTypeCount() {
174				return 3;
175			}
176
177			@Override
178			public int getItemViewType(int position) {
179				if (getItem(position).getStatus() == Message.STATUS_RECIEVED) {
180					return RECIEVED;
181				} else if (getItem(position).getStatus() == Message.STATUS_ERROR) {
182					return ERROR;
183				} else {
184					return SENT;
185				}
186			}
187
188			@Override
189			public View getView(int position, View view, ViewGroup parent) {
190				Message item = getItem(position);
191				int type = getItemViewType(position);
192				ViewHolder viewHolder;
193				if (view == null) {
194					viewHolder = new ViewHolder();
195					switch (type) {
196					case SENT:
197						view = (View) inflater.inflate(R.layout.message_sent,
198								null);
199						viewHolder.imageView = (ImageView) view
200								.findViewById(R.id.message_photo);
201						viewHolder.imageView.setImageBitmap(selfBitmap);
202						break;
203					case RECIEVED:
204						view = (View) inflater.inflate(
205								R.layout.message_recieved, null);
206						viewHolder.imageView = (ImageView) view
207								.findViewById(R.id.message_photo);
208						if (item.getConversation().getMode() == Conversation.MODE_SINGLE) {
209							Uri uri = item.getConversation()
210									.getProfilePhotoUri();
211							if (uri != null) {
212								viewHolder.imageView
213										.setImageBitmap(mBitmapCache.get(item
214												.getConversation().getName(),
215												uri));
216							} else {
217								viewHolder.imageView
218										.setImageBitmap(mBitmapCache.get(item
219												.getConversation().getName(),
220												null));
221							}
222						}
223						break;
224					case ERROR:
225						view = (View) inflater.inflate(R.layout.message_error,
226								null);
227						viewHolder.imageView = (ImageView) view
228								.findViewById(R.id.message_photo);
229						viewHolder.imageView.setImageBitmap(mBitmapCache
230								.getError());
231						break;
232					default:
233						viewHolder = null;
234						break;
235					}
236					viewHolder.messageBody = (TextView) view
237							.findViewById(R.id.message_body);
238					viewHolder.time = (TextView) view
239							.findViewById(R.id.message_time);
240					view.setTag(viewHolder);
241				} else {
242					viewHolder = (ViewHolder) view.getTag();
243				}
244				if (type == RECIEVED) {
245					if (item.getConversation().getMode() == Conversation.MODE_MULTI) {
246						if (item.getCounterpart() != null) {
247							viewHolder.imageView.setImageBitmap(mBitmapCache
248									.get(item.getCounterpart(), null));
249						} else {
250							viewHolder.imageView
251									.setImageBitmap(mBitmapCache.get(item
252											.getConversation().getName(), null));
253						}
254					}
255				}
256				String body = item.getBody();
257				if (body != null) {
258					if (item.getEncryption() == Message.ENCRYPTION_PGP) {
259						viewHolder.messageBody.setText(getString(R.string.encrypted_message));
260						viewHolder.messageBody.setTextColor(0xff33B5E5);
261						viewHolder.messageBody.setTypeface(null,Typeface.ITALIC);
262					} else {
263						viewHolder.messageBody.setText(body.trim());
264						viewHolder.messageBody.setTextColor(0xff000000);
265						viewHolder.messageBody.setTypeface(null, Typeface.NORMAL);
266					}
267				}
268				if (item.getStatus() == Message.STATUS_UNSEND) {
269					viewHolder.time.setTypeface(null, Typeface.ITALIC);
270					viewHolder.time.setText("sending\u2026");
271				} else {
272					viewHolder.time.setTypeface(null, Typeface.NORMAL);
273					if ((item.getConversation().getMode() == Conversation.MODE_SINGLE)
274							|| (type != RECIEVED)) {
275						viewHolder.time.setText(UIHelper
276								.readableTimeDifference(item.getTimeSent()));
277					} else {
278						viewHolder.time.setText(item.getCounterpart()
279								+ " \u00B7 "
280								+ UIHelper.readableTimeDifference(item
281										.getTimeSent()));
282					}
283				}
284				return view;
285			}
286		};
287		messagesView.setAdapter(messageListAdapter);
288
289		return view;
290	}
291
292	protected Bitmap findSelfPicture() {
293		SharedPreferences sharedPref = PreferenceManager
294				.getDefaultSharedPreferences(getActivity()
295						.getApplicationContext());
296		boolean showPhoneSelfContactPicture = sharedPref.getBoolean(
297				"show_phone_selfcontact_picture", true);
298
299		Bitmap self = null;
300
301		if (showPhoneSelfContactPicture) {
302			Uri selfiUri = PhoneHelper.getSefliUri(getActivity());
303			if (selfiUri != null) {
304				try {
305					self = BitmapFactory.decodeStream(getActivity()
306							.getContentResolver().openInputStream(selfiUri));
307				} catch (FileNotFoundException e) {
308					self = null;
309				}
310			}
311		}
312		if (self == null) {
313			self = UIHelper.getUnknownContactPicture(conversation.getAccount()
314					.getJid(), 200);
315		}
316
317		final Bitmap selfBitmap = self;
318		return selfBitmap;
319	}
320
321	@Override
322	public void onStart() {
323		super.onStart();
324		ConversationActivity activity = (ConversationActivity) getActivity();
325
326		if (activity.xmppConnectionServiceBound) {
327			this.onBackendConnected();
328		}
329	}
330
331	public void onBackendConnected() {
332		final ConversationActivity activity = (ConversationActivity) getActivity();
333		activity.registerListener();
334		this.conversation = activity.getSelectedConversation();
335		this.selfBitmap = findSelfPicture();
336		updateMessages();
337		// rendering complete. now go tell activity to close pane
338		if (activity.getSlidingPaneLayout().isSlideable()) {
339			if (!activity.shouldPaneBeOpen()) {
340				activity.getSlidingPaneLayout().closePane();
341				activity.getActionBar().setDisplayHomeAsUpEnabled(true);
342				activity.getActionBar().setTitle(conversation.getName());
343				activity.invalidateOptionsMenu();
344				
345			}
346		}
347		if (queuedPqpMessage != null) {
348			this.conversation.nextMessageEncryption = Message.ENCRYPTION_PGP;
349			Message message = new Message(conversation, queuedPqpMessage,
350					Message.ENCRYPTION_PGP);
351			sendPgpMessage(message);
352		}
353		if (conversation.getMode() == Conversation.MODE_MULTI) {
354			activity.xmppConnectionService.setOnRenameListener(new OnRenameListener() {
355				
356				@Override
357				public void onRename(final boolean success) {
358					getActivity().runOnUiThread(new Runnable() {
359						
360						@Override
361						public void run() {
362							if (success) {
363								Toast.makeText(getActivity(), "Your nickname has been changed",Toast.LENGTH_SHORT).show();
364							} else {
365								Toast.makeText(getActivity(), "Nichname is already in use",Toast.LENGTH_SHORT).show();
366							}
367						}
368					});
369				}
370			});
371		}
372	}
373
374	public void updateMessages() {
375		ConversationActivity activity = (ConversationActivity) getActivity();
376		List<Message> encryptedMessages = new LinkedList<Message>();
377		// TODO this.conversation could be null?!
378		for(Message message : this.conversation.getMessages()) {
379			if (message.getEncryption() == Message.ENCRYPTION_PGP) {
380				encryptedMessages.add(message);
381			}
382		}
383		if (encryptedMessages.size() > 0) {
384			DecryptMessage task = new DecryptMessage();
385			Message[] msgs = new Message[encryptedMessages.size()];
386			task.execute(encryptedMessages.toArray(msgs));
387		}
388		this.messageList.clear();
389		this.messageList.addAll(this.conversation.getMessages());
390		this.messageListAdapter.notifyDataSetChanged();
391		if (conversation.getMode() == Conversation.MODE_SINGLE) {
392			if (messageList.size() >= 1) {
393				int latestEncryption = this.conversation.getLatestMessage()
394						.getEncryption();
395				if (latestEncryption== Message.ENCRYPTION_DECRYPTED) {
396					conversation.nextMessageEncryption = Message.ENCRYPTION_PGP;
397				} else {
398					conversation.nextMessageEncryption = latestEncryption;
399				}
400				makeFingerprintWarning(latestEncryption);
401			}
402		} else {
403			if (conversation.getMucOptions().getError() != 0) {
404				mucError.setVisibility(View.VISIBLE);
405				if (conversation.getMucOptions().getError() == MucOptions.ERROR_NICK_IN_USE) {
406					mucErrorText.setText(getString(R.string.nick_in_use));
407				}
408			} else {
409				mucError.setVisibility(View.GONE);
410			}
411		}
412		getActivity().invalidateOptionsMenu();
413		updateChatMsgHint();
414		int size = this.messageList.size();
415		if (size >= 1)
416			messagesView.setSelection(size - 1);
417		if (!activity.shouldPaneBeOpen()) {
418			conversation.markRead();
419			//TODO update notifications
420			UIHelper.updateNotification(getActivity(), activity.getConversationList(), false);
421			activity.updateConversationList();
422		}
423	}
424
425	protected void makeFingerprintWarning(int latestEncryption) {
426		final LinearLayout fingerprintWarning = (LinearLayout) getView()
427				.findViewById(R.id.new_fingerprint);
428		if (conversation.getContact() != null) {
429			Set<String> knownFingerprints = conversation.getContact()
430					.getOtrFingerprints();
431			if ((latestEncryption == Message.ENCRYPTION_OTR)
432					&& (conversation.hasValidOtrSession()
433							&& (conversation.getOtrSession().getSessionStatus() == SessionStatus.ENCRYPTED) && (!knownFingerprints
434								.contains(conversation.getOtrFingerprint())))) {
435				fingerprintWarning.setVisibility(View.VISIBLE);
436				TextView fingerprint = (TextView) getView().findViewById(
437						R.id.otr_fingerprint);
438				fingerprint.setText(conversation.getOtrFingerprint());
439				fingerprintWarning.setOnClickListener(new OnClickListener() {
440
441					@Override
442					public void onClick(View v) {
443						AlertDialog dialog = UIHelper
444								.getVerifyFingerprintDialog(
445										(ConversationActivity) getActivity(),
446										conversation, fingerprintWarning);
447						dialog.show();
448					}
449				});
450			} else {
451				fingerprintWarning.setVisibility(View.GONE);
452			}
453		} else {
454			fingerprintWarning.setVisibility(View.GONE);
455		}
456	}
457
458	protected void sendPlainTextMessage(Message message) {
459		ConversationActivity activity = (ConversationActivity) getActivity();
460		activity.xmppConnectionService.sendMessage(message, null);
461		chatMsg.setText("");
462	}
463
464	protected void sendPgpMessage(final Message message) {
465		ConversationActivity activity = (ConversationActivity) getActivity();
466		final XmppConnectionService xmppService = activity.xmppConnectionService;
467		Contact contact = message.getConversation().getContact();
468		if (activity.hasPgp()) {
469			if (contact.getPgpKeyId() != 0) {
470				xmppService.sendMessage(message, null);
471				chatMsg.setText("");
472			} else {
473				AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
474				builder.setTitle("No openPGP key found");
475				builder.setIconAttribute(android.R.attr.alertDialogIcon);
476				builder.setMessage("There is no openPGP key assoziated with this contact");
477				builder.setNegativeButton("Cancel", null);
478				builder.setPositiveButton("Send plain text",
479						new DialogInterface.OnClickListener() {
480	
481							@Override
482							public void onClick(DialogInterface dialog, int which) {
483								conversation.nextMessageEncryption = Message.ENCRYPTION_NONE;
484								message.setEncryption(Message.ENCRYPTION_NONE);
485								xmppService.sendMessage(message, null);
486								chatMsg.setText("");
487							}
488						});
489				builder.create().show();
490			}
491		}
492	}
493	
494	protected void sendOtrMessage(final Message message) {
495		ConversationActivity activity = (ConversationActivity) getActivity();
496		final XmppConnectionService xmppService = activity.xmppConnectionService;
497		if (conversation.hasValidOtrSession()) {
498			activity.xmppConnectionService.sendMessage(message, null);
499			chatMsg.setText("");
500		} else {
501			Hashtable<String, Integer> presences;
502			if (conversation.getContact() != null) {
503				presences = conversation.getContact().getPresences();
504			} else {
505				presences = null;
506			}
507			if ((presences == null) || (presences.size() == 0)) {
508				AlertDialog.Builder builder = new AlertDialog.Builder(
509						getActivity());
510				builder.setTitle("Contact is offline");
511				builder.setIconAttribute(android.R.attr.alertDialogIcon);
512				builder.setMessage("Sending OTR encrypted messages to an offline contact is impossible.");
513				builder.setPositiveButton("Send plain text",
514						new DialogInterface.OnClickListener() {
515
516							@Override
517							public void onClick(DialogInterface dialog,
518									int which) {
519								conversation.nextMessageEncryption = Message.ENCRYPTION_NONE;
520								message.setEncryption(Message.ENCRYPTION_NONE);
521								xmppService.sendMessage(message, null);
522								chatMsg.setText("");
523							}
524						});
525				builder.setNegativeButton("Cancel", null);
526				builder.create().show();
527			} else if (presences.size() == 1) {
528				xmppService.sendMessage(message, (String) presences.keySet()
529						.toArray()[0]);
530				chatMsg.setText("");
531			} else {
532				AlertDialog.Builder builder = new AlertDialog.Builder(
533						getActivity());
534				builder.setTitle("Choose Presence");
535				final String[] presencesArray = new String[presences.size()];
536				presences.keySet().toArray(presencesArray);
537				builder.setItems(presencesArray,
538						new DialogInterface.OnClickListener() {
539
540							@Override
541							public void onClick(DialogInterface dialog,
542									int which) {
543								xmppService.sendMessage(message,
544										presencesArray[which]);
545								chatMsg.setText("");
546							}
547						});
548				builder.create().show();
549			}
550		}
551	}
552
553	private static class ViewHolder {
554
555		protected TextView time;
556		protected TextView messageBody;
557		protected ImageView imageView;
558
559	}
560
561	private class BitmapCache {
562		private HashMap<String, Bitmap> bitmaps = new HashMap<String, Bitmap>();
563		private Bitmap error = null;
564
565		public Bitmap get(String name, Uri uri) {
566			if (bitmaps.containsKey(name)) {
567				return bitmaps.get(name);
568			} else {
569				Bitmap bm;
570				if (uri != null) {
571					try {
572						bm = BitmapFactory.decodeStream(getActivity()
573								.getContentResolver().openInputStream(uri));
574					} catch (FileNotFoundException e) {
575						bm = UIHelper.getUnknownContactPicture(name, 200);
576					}
577				} else {
578					bm = UIHelper.getUnknownContactPicture(name, 200);
579				}
580				bitmaps.put(name, bm);
581				return bm;
582			}
583		}
584
585		public Bitmap getError() {
586			if (error == null) {
587				error = UIHelper.getErrorPicture(200);
588			}
589			return error;
590		}
591	}
592	
593	class DecryptMessage extends AsyncTask<Message, Void, Boolean> {
594
595		@Override
596		protected Boolean doInBackground(Message... params) {
597			final ConversationActivity activity = (ConversationActivity) getActivity();
598			askForPassphraseIntent = null;
599			for(int i = 0; i < params.length; ++i) {
600				if (params[i].getEncryption() == Message.ENCRYPTION_PGP) {
601					String body = params[i].getBody();
602					String decrypted = null;
603					if (activity==null) {
604						return false;
605					} else if (!activity.xmppConnectionServiceBound) {
606						return false;
607					}
608					try {
609						decrypted = activity.xmppConnectionService.getPgpEngine().decrypt(body);
610					} catch (UserInputRequiredException e) {
611						askForPassphraseIntent = e.getPendingIntent().getIntentSender();
612						activity.runOnUiThread(new Runnable() {
613							
614							@Override
615							public void run() {
616								pgpInfo.setVisibility(View.VISIBLE);
617							}
618						});
619						
620						return false;
621		
622					} catch (OpenPgpException e) {
623						Log.d("gultsch","error decrypting pgp");
624					}
625					if (decrypted!=null) {
626						params[i].setBody(decrypted);
627						params[i].setEncryption(Message.ENCRYPTION_DECRYPTED);
628						activity.xmppConnectionService.updateMessage(params[i]);
629					}
630					if (activity!=null) {
631						activity.runOnUiThread(new Runnable() {
632							
633							@Override
634							public void run() {
635								messageListAdapter.notifyDataSetChanged();
636							}
637						});
638					}
639				}
640				if (activity!=null) {
641					activity.runOnUiThread(new Runnable() {
642						
643						@Override
644						public void run() {
645							activity.updateConversationList();
646						}
647					});
648				}
649			}
650			return true;
651		}
652		
653	}
654}