StartConversationActivity.java

  1package eu.siacs.conversations.ui;
  2
  3import android.Manifest;
  4import android.annotation.SuppressLint;
  5import android.annotation.TargetApi;
  6import android.app.ActionBar;
  7import android.app.ActionBar.Tab;
  8import android.app.ActionBar.TabListener;
  9import android.app.AlertDialog;
 10import android.app.Fragment;
 11import android.app.FragmentTransaction;
 12import android.app.ListFragment;
 13import android.content.Context;
 14import android.content.DialogInterface;
 15import android.content.DialogInterface.OnClickListener;
 16import android.content.Intent;
 17import android.content.pm.PackageManager;
 18import android.net.Uri;
 19import android.nfc.NdefMessage;
 20import android.nfc.NdefRecord;
 21import android.nfc.NfcAdapter;
 22import android.os.Build;
 23import android.os.Bundle;
 24import android.os.Parcelable;
 25import android.support.v13.app.FragmentPagerAdapter;
 26import android.support.v4.view.ViewPager;
 27import android.text.Editable;
 28import android.text.TextWatcher;
 29import android.util.Log;
 30import android.view.ContextMenu;
 31import android.view.ContextMenu.ContextMenuInfo;
 32import android.view.KeyEvent;
 33import android.view.Menu;
 34import android.view.MenuItem;
 35import android.view.View;
 36import android.view.inputmethod.InputMethodManager;
 37import android.widget.AdapterView;
 38import android.widget.AdapterView.AdapterContextMenuInfo;
 39import android.widget.AdapterView.OnItemClickListener;
 40import android.widget.ArrayAdapter;
 41import android.widget.AutoCompleteTextView;
 42import android.widget.CheckBox;
 43import android.widget.Checkable;
 44import android.widget.EditText;
 45import android.widget.ListView;
 46import android.widget.Spinner;
 47import android.widget.TextView;
 48import android.widget.Toast;
 49
 50import com.google.zxing.integration.android.IntentIntegrator;
 51import com.google.zxing.integration.android.IntentResult;
 52
 53import java.util.ArrayList;
 54import java.util.Arrays;
 55import java.util.Collections;
 56import java.util.List;
 57import java.util.concurrent.atomic.AtomicBoolean;
 58
 59import eu.siacs.conversations.Config;
 60import eu.siacs.conversations.R;
 61import eu.siacs.conversations.entities.Account;
 62import eu.siacs.conversations.entities.Blockable;
 63import eu.siacs.conversations.entities.Bookmark;
 64import eu.siacs.conversations.entities.Contact;
 65import eu.siacs.conversations.entities.Conversation;
 66import eu.siacs.conversations.entities.ListItem;
 67import eu.siacs.conversations.entities.Presence;
 68import eu.siacs.conversations.services.XmppConnectionService.OnRosterUpdate;
 69import eu.siacs.conversations.ui.adapter.KnownHostsAdapter;
 70import eu.siacs.conversations.ui.adapter.ListItemAdapter;
 71import eu.siacs.conversations.utils.XmppUri;
 72import eu.siacs.conversations.xmpp.OnUpdateBlocklist;
 73import eu.siacs.conversations.xmpp.XmppConnection;
 74import eu.siacs.conversations.xmpp.jid.InvalidJidException;
 75import eu.siacs.conversations.xmpp.jid.Jid;
 76
 77public class StartConversationActivity extends XmppActivity implements OnRosterUpdate, OnUpdateBlocklist {
 78
 79	public int conference_context_id;
 80	public int contact_context_id;
 81	private Tab mContactsTab;
 82	private Tab mConferencesTab;
 83	private ViewPager mViewPager;
 84	private MyListFragment mContactsListFragment = new MyListFragment();
 85	private List<ListItem> contacts = new ArrayList<>();
 86	private ArrayAdapter<ListItem> mContactsAdapter;
 87	private MyListFragment mConferenceListFragment = new MyListFragment();
 88	private List<ListItem> conferences = new ArrayList<ListItem>();
 89	private ArrayAdapter<ListItem> mConferenceAdapter;
 90	private List<String> mActivatedAccounts = new ArrayList<String>();
 91	private List<String> mKnownHosts;
 92	private List<String> mKnownConferenceHosts;
 93	private Invite mPendingInvite = null;
 94	private Menu mOptionsMenu;
 95	private EditText mSearchEditText;
 96	private AtomicBoolean mRequestedContactsPermission = new AtomicBoolean(false);
 97	private final int REQUEST_SYNC_CONTACTS = 0x3b28cf;
 98
 99	private MenuItem.OnActionExpandListener mOnActionExpandListener = new MenuItem.OnActionExpandListener() {
100
101		@Override
102		public boolean onMenuItemActionExpand(MenuItem item) {
103			mSearchEditText.post(new Runnable() {
104
105				@Override
106				public void run() {
107					mSearchEditText.requestFocus();
108					InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
109					imm.showSoftInput(mSearchEditText,
110							InputMethodManager.SHOW_IMPLICIT);
111				}
112			});
113
114			return true;
115		}
116
117		@Override
118		public boolean onMenuItemActionCollapse(MenuItem item) {
119			InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
120			imm.hideSoftInputFromWindow(mSearchEditText.getWindowToken(),
121					InputMethodManager.HIDE_IMPLICIT_ONLY);
122			mSearchEditText.setText("");
123			filter(null);
124			return true;
125		}
126	};
127	private boolean mHideOfflineContacts = false;
128	private TabListener mTabListener = new TabListener() {
129
130		@Override
131		public void onTabUnselected(Tab tab, FragmentTransaction ft) {
132			return;
133		}
134
135		@Override
136		public void onTabSelected(Tab tab, FragmentTransaction ft) {
137			mViewPager.setCurrentItem(tab.getPosition());
138			onTabChanged();
139		}
140
141		@Override
142		public void onTabReselected(Tab tab, FragmentTransaction ft) {
143			return;
144		}
145	};
146	private ViewPager.SimpleOnPageChangeListener mOnPageChangeListener = new ViewPager.SimpleOnPageChangeListener() {
147		@Override
148		public void onPageSelected(int position) {
149			if (getActionBar() != null) {
150				getActionBar().setSelectedNavigationItem(position);
151			}
152			onTabChanged();
153		}
154	};
155	private TextWatcher mSearchTextWatcher = new TextWatcher() {
156
157		@Override
158		public void afterTextChanged(Editable editable) {
159			filter(editable.toString());
160		}
161
162		@Override
163		public void beforeTextChanged(CharSequence s, int start, int count,
164				int after) {
165		}
166
167		@Override
168		public void onTextChanged(CharSequence s, int start, int before,
169				int count) {
170		}
171	};
172	private MenuItem mMenuSearchView;
173	private ListItemAdapter.OnTagClickedListener mOnTagClickedListener = new ListItemAdapter.OnTagClickedListener() {
174		@Override
175		public void onTagClicked(String tag) {
176			if (mMenuSearchView != null) {
177				mMenuSearchView.expandActionView();
178				mSearchEditText.setText("");
179				mSearchEditText.append(tag);
180				filter(tag);
181			}
182		}
183	};
184	private String mInitialJid;
185
186	@Override
187	public void onRosterUpdate() {
188		this.refreshUi();
189	}
190
191	@Override
192	public void onCreate(Bundle savedInstanceState) {
193		super.onCreate(savedInstanceState);
194		setContentView(R.layout.activity_start_conversation);
195		mViewPager = (ViewPager) findViewById(R.id.start_conversation_view_pager);
196		ActionBar actionBar = getActionBar();
197		actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
198
199		mContactsTab = actionBar.newTab().setText(R.string.contacts)
200			.setTabListener(mTabListener);
201		mConferencesTab = actionBar.newTab().setText(R.string.conferences)
202			.setTabListener(mTabListener);
203		actionBar.addTab(mContactsTab);
204		actionBar.addTab(mConferencesTab);
205
206		mViewPager.setOnPageChangeListener(mOnPageChangeListener);
207		mViewPager.setAdapter(new FragmentPagerAdapter(getFragmentManager()) {
208
209			@Override
210			public int getCount() {
211				return 2;
212			}
213
214			@Override
215			public Fragment getItem(int position) {
216				if (position == 0) {
217					return mContactsListFragment;
218				} else {
219					return mConferenceListFragment;
220				}
221			}
222		});
223
224		mConferenceAdapter = new ListItemAdapter(this, conferences);
225		mConferenceListFragment.setListAdapter(mConferenceAdapter);
226		mConferenceListFragment.setContextMenu(R.menu.conference_context);
227		mConferenceListFragment
228			.setOnListItemClickListener(new OnItemClickListener() {
229
230				@Override
231				public void onItemClick(AdapterView<?> arg0, View arg1,
232						int position, long arg3) {
233					openConversationForBookmark(position);
234				}
235			});
236
237		mContactsAdapter = new ListItemAdapter(this, contacts);
238		((ListItemAdapter) mContactsAdapter).setOnTagClickedListener(this.mOnTagClickedListener);
239		mContactsListFragment.setListAdapter(mContactsAdapter);
240		mContactsListFragment.setContextMenu(R.menu.contact_context);
241		mContactsListFragment
242			.setOnListItemClickListener(new OnItemClickListener() {
243
244				@Override
245				public void onItemClick(AdapterView<?> arg0, View arg1,
246						int position, long arg3) {
247					openConversationForContact(position);
248				}
249			});
250
251		this.mHideOfflineContacts = getPreferences().getBoolean("hide_offline", false);
252
253	}
254
255	@Override
256	public void onStart() {
257		super.onStart();
258		askForContactsPermissions();
259	}
260
261	protected void openConversationForContact(int position) {
262		Contact contact = (Contact) contacts.get(position);
263		Conversation conversation = xmppConnectionService
264			.findOrCreateConversation(contact.getAccount(),
265					contact.getJid(), false);
266		switchToConversation(conversation);
267	}
268
269	protected void openConversationForContact() {
270		int position = contact_context_id;
271		openConversationForContact(position);
272	}
273
274	protected void openConversationForBookmark() {
275		openConversationForBookmark(conference_context_id);
276	}
277
278	protected void openConversationForBookmark(int position) {
279		Bookmark bookmark = (Bookmark) conferences.get(position);
280		Jid jid = bookmark.getJid();
281		if (jid == null) {
282			Toast.makeText(this,R.string.invalid_jid,Toast.LENGTH_SHORT).show();
283			return;
284		}
285		Conversation conversation = xmppConnectionService.findOrCreateConversation(bookmark.getAccount(),jid, true);
286		conversation.setBookmark(bookmark);
287		if (!conversation.getMucOptions().online()) {
288			xmppConnectionService.joinMuc(conversation);
289		}
290		if (!bookmark.autojoin() && getPreferences().getBoolean("autojoin", true)) {
291			bookmark.setAutojoin(true);
292			xmppConnectionService.pushBookmarks(bookmark.getAccount());
293		}
294		switchToConversation(conversation);
295	}
296
297	protected void openDetailsForContact() {
298		int position = contact_context_id;
299		Contact contact = (Contact) contacts.get(position);
300		switchToContactDetails(contact);
301	}
302
303	protected void toggleContactBlock() {
304		final int position = contact_context_id;
305		BlockContactDialog.show(this, xmppConnectionService, (Contact) contacts.get(position));
306	}
307
308	protected void deleteContact() {
309		final int position = contact_context_id;
310		final Contact contact = (Contact) contacts.get(position);
311		final AlertDialog.Builder builder = new AlertDialog.Builder(this);
312		builder.setNegativeButton(R.string.cancel, null);
313		builder.setTitle(R.string.action_delete_contact);
314		builder.setMessage(getString(R.string.remove_contact_text,
315				contact.getJid()));
316		builder.setPositiveButton(R.string.delete, new OnClickListener() {
317
318			@Override
319			public void onClick(DialogInterface dialog, int which) {
320				xmppConnectionService.deleteContactOnServer(contact);
321				filter(mSearchEditText.getText().toString());
322			}
323		});
324		builder.create().show();
325	}
326
327	protected void deleteConference() {
328		int position = conference_context_id;
329		final Bookmark bookmark = (Bookmark) conferences.get(position);
330
331		AlertDialog.Builder builder = new AlertDialog.Builder(this);
332		builder.setNegativeButton(R.string.cancel, null);
333		builder.setTitle(R.string.delete_bookmark);
334		builder.setMessage(getString(R.string.remove_bookmark_text,
335				bookmark.getJid()));
336		builder.setPositiveButton(R.string.delete, new OnClickListener() {
337
338			@Override
339			public void onClick(DialogInterface dialog, int which) {
340				bookmark.unregisterConversation();
341				Account account = bookmark.getAccount();
342				account.getBookmarks().remove(bookmark);
343				xmppConnectionService.pushBookmarks(account);
344				filter(mSearchEditText.getText().toString());
345			}
346		});
347		builder.create().show();
348
349	}
350
351	@SuppressLint("InflateParams")
352	protected void showCreateContactDialog(final String prefilledJid, final String fingerprint) {
353		EnterJidDialog dialog = new EnterJidDialog(
354			this, mKnownHosts, mActivatedAccounts,
355			getString(R.string.create_contact), getString(R.string.create),
356			prefilledJid, null, fingerprint == null
357		);
358
359		dialog.setOnEnterJidDialogPositiveListener(new EnterJidDialog.OnEnterJidDialogPositiveListener() {
360			@Override
361			public boolean onEnterJidDialogPositive(Jid accountJid, Jid contactJid) throws EnterJidDialog.JidError {
362				if (!xmppConnectionServiceBound) {
363					return false;
364				}
365
366				final Account account = xmppConnectionService.findAccountByJid(accountJid);
367				if (account == null) {
368					return true;
369				}
370
371				final Contact contact = account.getRoster().getContact(contactJid);
372				if (contact.showInRoster()) {
373					throw new EnterJidDialog.JidError(getString(R.string.contact_already_exists));
374				} else {
375					contact.addOtrFingerprint(fingerprint);
376					xmppConnectionService.createContact(contact);
377					switchToConversation(contact);
378					return true;
379				}
380			}
381		});
382
383		dialog.show();
384	}
385
386	@SuppressLint("InflateParams")
387	protected void showJoinConferenceDialog(final String prefilledJid) {
388		final AlertDialog.Builder builder = new AlertDialog.Builder(this);
389		builder.setTitle(R.string.join_conference);
390		final View dialogView = getLayoutInflater().inflate(R.layout.join_conference_dialog, null);
391		final Spinner spinner = (Spinner) dialogView.findViewById(R.id.account);
392		final AutoCompleteTextView jid = (AutoCompleteTextView) dialogView.findViewById(R.id.jid);
393		final boolean lock = Config.LOCK_DOMAINS_IN_CONVERSATIONS && Config.CONFERENCE_DOMAIN_LOCK != null;
394		final TextView jabberIdDesc = (TextView) dialogView.findViewById(R.id.jabber_id);
395		jabberIdDesc.setText(lock ? R.string.conference_name : R.string.conference_address);
396		jid.setHint(lock ? R.string.conference_name : R.string.conference_address_example);
397		if (!lock) {
398			jid.setAdapter(new KnownHostsAdapter(this, android.R.layout.simple_list_item_1, mKnownConferenceHosts));
399		}
400		if (prefilledJid != null) {
401			jid.append(prefilledJid);
402		}
403		populateAccountSpinner(this, mActivatedAccounts, spinner);
404		final Checkable bookmarkCheckBox = (CheckBox) dialogView
405			.findViewById(R.id.bookmark);
406		builder.setView(dialogView);
407		builder.setNegativeButton(R.string.cancel, null);
408		builder.setPositiveButton(R.string.join, null);
409		final AlertDialog dialog = builder.create();
410		dialog.show();
411		dialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(
412				new View.OnClickListener() {
413
414					@Override
415					public void onClick(final View v) {
416						if (!xmppConnectionServiceBound) {
417							return;
418						}
419						final Account account = getSelectedAccount(spinner);
420						if (account == null) {
421							return;
422						}
423						final Jid conferenceJid;
424						try {
425							if (lock) {
426								conferenceJid = Jid.fromParts(jid.getText().toString(),Config.CONFERENCE_DOMAIN_LOCK, null);
427							} else {
428								conferenceJid = Jid.fromString(jid.getText().toString());
429							}
430						} catch (final InvalidJidException e) {
431							jid.setError(getString(lock ? R.string.invalid_conference_name : R.string.invalid_jid));
432							return;
433						}
434
435						if (bookmarkCheckBox.isChecked()) {
436							if (account.hasBookmarkFor(conferenceJid)) {
437								jid.setError(getString(R.string.bookmark_already_exists));
438							} else {
439								final Bookmark bookmark = new Bookmark(account, conferenceJid.toBareJid());
440								bookmark.setAutojoin(getPreferences().getBoolean("autojoin", true));
441								String nick = conferenceJid.getResourcepart();
442								if (nick != null && !nick.isEmpty()) {
443									bookmark.setNick(nick);
444								}
445								account.getBookmarks().add(bookmark);
446								xmppConnectionService.pushBookmarks(account);
447								final Conversation conversation = xmppConnectionService
448										.findOrCreateConversation(account,
449												conferenceJid, true);
450								conversation.setBookmark(bookmark);
451								if (!conversation.getMucOptions().online()) {
452									xmppConnectionService.joinMuc(conversation);
453								}
454								dialog.dismiss();
455								switchToConversation(conversation);
456							}
457						} else {
458							final Conversation conversation = xmppConnectionService
459									.findOrCreateConversation(account,
460											conferenceJid, true);
461							if (!conversation.getMucOptions().online()) {
462								xmppConnectionService.joinMuc(conversation);
463							}
464							dialog.dismiss();
465							switchToConversation(conversation);
466						}
467					}
468				});
469	}
470
471	private Account getSelectedAccount(Spinner spinner) {
472		if (!spinner.isEnabled()) {
473			return null;
474		}
475		Jid jid;
476		try {
477			if (Config.DOMAIN_LOCK != null) {
478				jid = Jid.fromParts((String) spinner.getSelectedItem(), Config.DOMAIN_LOCK, null);
479			} else {
480				jid = Jid.fromString((String) spinner.getSelectedItem());
481			}
482		} catch (final InvalidJidException e) {
483			return null;
484		}
485		return xmppConnectionService.findAccountByJid(jid);
486	}
487
488	protected void switchToConversation(Contact contact) {
489		Conversation conversation = xmppConnectionService
490			.findOrCreateConversation(contact.getAccount(),
491					contact.getJid(), false);
492		switchToConversation(conversation);
493	}
494
495	public static void populateAccountSpinner(Context context, List<String> accounts, Spinner spinner) {
496		if (accounts.size() > 0) {
497			ArrayAdapter<String> adapter = new ArrayAdapter<>(context,
498					android.R.layout.simple_spinner_item, accounts);
499			adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
500			spinner.setAdapter(adapter);
501			spinner.setEnabled(true);
502		} else {
503			ArrayAdapter<String> adapter = new ArrayAdapter<>(context,
504					android.R.layout.simple_spinner_item,
505					Arrays.asList(new String[]{context.getString(R.string.no_accounts)}));
506			adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
507			spinner.setAdapter(adapter);
508			spinner.setEnabled(false);
509		}
510	}
511
512	@Override
513	public boolean onCreateOptionsMenu(Menu menu) {
514		this.mOptionsMenu = menu;
515		getMenuInflater().inflate(R.menu.start_conversation, menu);
516		MenuItem menuCreateContact = menu.findItem(R.id.action_create_contact);
517		MenuItem menuCreateConference = menu.findItem(R.id.action_join_conference);
518		MenuItem menuHideOffline = menu.findItem(R.id.action_hide_offline);
519		menuHideOffline.setChecked(this.mHideOfflineContacts);
520		mMenuSearchView = menu.findItem(R.id.action_search);
521		mMenuSearchView.setOnActionExpandListener(mOnActionExpandListener);
522		View mSearchView = mMenuSearchView.getActionView();
523		mSearchEditText = (EditText) mSearchView
524			.findViewById(R.id.search_field);
525		mSearchEditText.addTextChangedListener(mSearchTextWatcher);
526		if (getActionBar().getSelectedNavigationIndex() == 0) {
527			menuCreateConference.setVisible(false);
528		} else {
529			menuCreateContact.setVisible(false);
530		}
531		if (mInitialJid != null) {
532			mMenuSearchView.expandActionView();
533			mSearchEditText.append(mInitialJid);
534			filter(mInitialJid);
535		}
536		return true;
537	}
538
539	@Override
540	public boolean onOptionsItemSelected(MenuItem item) {
541		switch (item.getItemId()) {
542			case R.id.action_create_contact:
543				showCreateContactDialog(null,null);
544				return true;
545			case R.id.action_join_conference:
546				showJoinConferenceDialog(null);
547				return true;
548			case R.id.action_scan_qr_code:
549				new IntentIntegrator(this).initiateScan();
550				return true;
551			case R.id.action_hide_offline:
552				mHideOfflineContacts = !item.isChecked();
553				getPreferences().edit().putBoolean("hide_offline", mHideOfflineContacts).commit();
554				if (mSearchEditText != null) {
555					filter(mSearchEditText.getText().toString());
556				}
557				invalidateOptionsMenu();
558		}
559		return super.onOptionsItemSelected(item);
560	}
561
562	@Override
563	public boolean onKeyUp(int keyCode, KeyEvent event) {
564		if (keyCode == KeyEvent.KEYCODE_SEARCH && !event.isLongPress()) {
565			mOptionsMenu.findItem(R.id.action_search).expandActionView();
566			return true;
567		}
568		return super.onKeyUp(keyCode, event);
569	}
570
571	@Override
572	public void onActivityResult(int requestCode, int resultCode, Intent intent) {
573		if ((requestCode & 0xFFFF) == IntentIntegrator.REQUEST_CODE) {
574			IntentResult scanResult = IntentIntegrator.parseActivityResult(requestCode, resultCode, intent);
575			if (scanResult != null && scanResult.getFormatName() != null) {
576				String data = scanResult.getContents();
577				Invite invite = new Invite(data);
578				if (xmppConnectionServiceBound) {
579					invite.invite();
580				} else if (invite.getJid() != null) {
581					this.mPendingInvite = invite;
582				} else {
583					this.mPendingInvite = null;
584				}
585			}
586		}
587		super.onActivityResult(requestCode, requestCode, intent);
588	}
589
590	private void askForContactsPermissions() {
591		if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
592			if (checkSelfPermission(Manifest.permission.READ_CONTACTS) != PackageManager.PERMISSION_GRANTED) {
593				if (mRequestedContactsPermission.compareAndSet(false, true)) {
594					if (shouldShowRequestPermissionRationale(Manifest.permission.READ_CONTACTS)) {
595						AlertDialog.Builder builder = new AlertDialog.Builder(this);
596						builder.setTitle(R.string.sync_with_contacts);
597						builder.setMessage(R.string.sync_with_contacts_long);
598						builder.setPositiveButton(R.string.next, new OnClickListener() {
599							@Override
600							public void onClick(DialogInterface dialog, int which) {
601								if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
602									requestPermissions(new String[]{Manifest.permission.READ_CONTACTS}, REQUEST_SYNC_CONTACTS);
603								}
604							}
605						});
606						builder.create().show();
607					} else {
608						requestPermissions(new String[]{Manifest.permission.READ_CONTACTS}, 0);
609					}
610				}
611			}
612		}
613	}
614
615	@Override
616	public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
617		if (grantResults.length > 0)
618			if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
619				if (requestCode == REQUEST_SYNC_CONTACTS && xmppConnectionServiceBound) {
620					xmppConnectionService.loadPhoneContacts();
621				}
622			}
623	}
624
625	@Override
626	protected void onBackendConnected() {
627		this.mActivatedAccounts.clear();
628		for (Account account : xmppConnectionService.getAccounts()) {
629			if (account.getStatus() != Account.State.DISABLED) {
630				if (Config.DOMAIN_LOCK != null) {
631					this.mActivatedAccounts.add(account.getJid().getLocalpart());
632				} else {
633					this.mActivatedAccounts.add(account.getJid().toBareJid().toString());
634				}
635			}
636		}
637		final Intent intent = getIntent();
638		final ActionBar ab = getActionBar();
639		if (intent != null && intent.getBooleanExtra("init",false) && ab != null) {
640			ab.setDisplayShowHomeEnabled(false);
641			ab.setDisplayHomeAsUpEnabled(false);
642			ab.setHomeButtonEnabled(false);
643		}
644		this.mKnownHosts = xmppConnectionService.getKnownHosts();
645		this.mKnownConferenceHosts = xmppConnectionService.getKnownConferenceHosts();
646		if (this.mPendingInvite != null) {
647			mPendingInvite.invite();
648			this.mPendingInvite = null;
649		} else if (!handleIntent(getIntent())) {
650			if (mSearchEditText != null) {
651				filter(mSearchEditText.getText().toString());
652			} else {
653				filter(null);
654			}
655		}
656		setIntent(null);
657	}
658
659	@TargetApi(Build.VERSION_CODES.JELLY_BEAN)
660	Invite getInviteJellyBean(NdefRecord record) {
661		return new Invite(record.toUri());
662	}
663
664	protected boolean handleIntent(Intent intent) {
665		if (intent == null || intent.getAction() == null) {
666			return false;
667		}
668		switch (intent.getAction()) {
669			case Intent.ACTION_SENDTO:
670			case Intent.ACTION_VIEW:
671				Log.d(Config.LOGTAG, "received uri=" + intent.getData());
672				return new Invite(intent.getData()).invite();
673			case NfcAdapter.ACTION_NDEF_DISCOVERED:
674				for (Parcelable message : getIntent().getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES)) {
675					if (message instanceof NdefMessage) {
676						Log.d(Config.LOGTAG, "received message=" + message);
677						for (NdefRecord record : ((NdefMessage) message).getRecords()) {
678							switch (record.getTnf()) {
679								case NdefRecord.TNF_WELL_KNOWN:
680									if (Arrays.equals(record.getType(), NdefRecord.RTD_URI)) {
681										if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
682											return getInviteJellyBean(record).invite();
683										} else {
684											byte[] payload = record.getPayload();
685											if (payload[0] == 0) {
686												return new Invite(Uri.parse(new String(Arrays.copyOfRange(
687																	payload, 1, payload.length)))).invite();
688											}
689										}
690									}
691							}
692						}
693					}
694				}
695		}
696		return false;
697	}
698
699	private boolean handleJid(Invite invite) {
700		List<Contact> contacts = xmppConnectionService.findContacts(invite.getJid());
701		if (contacts.size() == 0) {
702			showCreateContactDialog(invite.getJid().toString(),invite.getFingerprint());
703			return false;
704		} else if (contacts.size() == 1) {
705			Contact contact = contacts.get(0);
706			if (invite.getFingerprint() != null) {
707				if (contact.addOtrFingerprint(invite.getFingerprint())) {
708					Log.d(Config.LOGTAG,"added new fingerprint");
709					xmppConnectionService.syncRosterToDisk(contact.getAccount());
710				}
711			}
712			switchToConversation(contact);
713			return true;
714		} else {
715			if (mMenuSearchView != null) {
716				mMenuSearchView.expandActionView();
717				mSearchEditText.setText("");
718				mSearchEditText.append(invite.getJid().toString());
719				filter(invite.getJid().toString());
720			} else {
721				mInitialJid = invite.getJid().toString();
722			}
723			return true;
724		}
725	}
726
727	protected void filter(String needle) {
728		if (xmppConnectionServiceBound) {
729			this.filterContacts(needle);
730			this.filterConferences(needle);
731		}
732	}
733
734	protected void filterContacts(String needle) {
735		this.contacts.clear();
736		for (Account account : xmppConnectionService.getAccounts()) {
737			if (account.getStatus() != Account.State.DISABLED) {
738				for (Contact contact : account.getRoster().getContacts()) {
739					Presence p = contact.getPresences().getMostAvailablePresence();
740					Presence.Status s = p == null ? Presence.Status.OFFLINE : p.getStatus();
741					if (contact.showInRoster() && contact.match(needle)
742							&& (!this.mHideOfflineContacts
743							|| s.compareTo(Presence.Status.OFFLINE) < 0)) {
744						this.contacts.add(contact);
745					}
746				}
747			}
748		}
749		Collections.sort(this.contacts);
750		mContactsAdapter.notifyDataSetChanged();
751	}
752
753	protected void filterConferences(String needle) {
754		this.conferences.clear();
755		for (Account account : xmppConnectionService.getAccounts()) {
756			if (account.getStatus() != Account.State.DISABLED) {
757				for (Bookmark bookmark : account.getBookmarks()) {
758					if (bookmark.match(needle)) {
759						this.conferences.add(bookmark);
760					}
761				}
762			}
763		}
764		Collections.sort(this.conferences);
765		mConferenceAdapter.notifyDataSetChanged();
766	}
767
768	private void onTabChanged() {
769		invalidateOptionsMenu();
770	}
771
772	@Override
773	public void OnUpdateBlocklist(final Status status) {
774		refreshUi();
775	}
776
777	@Override
778	protected void refreshUiReal() {
779		if (mSearchEditText != null) {
780			filter(mSearchEditText.getText().toString());
781		}
782	}
783
784	public static class MyListFragment extends ListFragment {
785		private AdapterView.OnItemClickListener mOnItemClickListener;
786		private int mResContextMenu;
787
788		public void setContextMenu(final int res) {
789			this.mResContextMenu = res;
790		}
791
792		@Override
793		public void onListItemClick(final ListView l, final View v, final int position, final long id) {
794			if (mOnItemClickListener != null) {
795				mOnItemClickListener.onItemClick(l, v, position, id);
796			}
797		}
798
799		public void setOnListItemClickListener(AdapterView.OnItemClickListener l) {
800			this.mOnItemClickListener = l;
801		}
802
803		@Override
804		public void onViewCreated(final View view, final Bundle savedInstanceState) {
805			super.onViewCreated(view, savedInstanceState);
806			registerForContextMenu(getListView());
807			getListView().setFastScrollEnabled(true);
808		}
809
810		@Override
811		public void onCreateContextMenu(final ContextMenu menu, final View v,
812				final ContextMenuInfo menuInfo) {
813			super.onCreateContextMenu(menu, v, menuInfo);
814			final StartConversationActivity activity = (StartConversationActivity) getActivity();
815			activity.getMenuInflater().inflate(mResContextMenu, menu);
816			final AdapterView.AdapterContextMenuInfo acmi = (AdapterContextMenuInfo) menuInfo;
817			if (mResContextMenu == R.menu.conference_context) {
818				activity.conference_context_id = acmi.position;
819			} else if (mResContextMenu == R.menu.contact_context){
820				activity.contact_context_id = acmi.position;
821				final Blockable contact = (Contact) activity.contacts.get(acmi.position);
822				final MenuItem blockUnblockItem = menu.findItem(R.id.context_contact_block_unblock);
823				XmppConnection xmpp = contact.getAccount().getXmppConnection();
824				if (xmpp != null && xmpp.getFeatures().blocking()) {
825					if (contact.isBlocked()) {
826						blockUnblockItem.setTitle(R.string.unblock_contact);
827					} else {
828						blockUnblockItem.setTitle(R.string.block_contact);
829					}
830				} else {
831					blockUnblockItem.setVisible(false);
832				}
833			}
834		}
835
836		@Override
837		public boolean onContextItemSelected(final MenuItem item) {
838			StartConversationActivity activity = (StartConversationActivity) getActivity();
839			switch (item.getItemId()) {
840				case R.id.context_start_conversation:
841					activity.openConversationForContact();
842					break;
843				case R.id.context_contact_details:
844					activity.openDetailsForContact();
845					break;
846				case R.id.context_contact_block_unblock:
847					activity.toggleContactBlock();
848					break;
849				case R.id.context_delete_contact:
850					activity.deleteContact();
851					break;
852				case R.id.context_join_conference:
853					activity.openConversationForBookmark();
854					break;
855				case R.id.context_delete_conference:
856					activity.deleteConference();
857			}
858			return true;
859		}
860	}
861
862	private class Invite extends XmppUri {
863
864		public Invite(final Uri uri) {
865			super(uri);
866		}
867
868		public Invite(final String uri) {
869			super(uri);
870		}
871
872		boolean invite() {
873			if (jid != null) {
874				if (muc) {
875					showJoinConferenceDialog(jid);
876				} else {
877					return handleJid(this);
878				}
879			}
880			return false;
881		}
882	}
883}