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}