1package eu.siacs.conversations.ui;
2
3import android.Manifest;
4import android.annotation.SuppressLint;
5import android.content.res.TypedArray;
6import android.databinding.DataBindingUtil;
7import android.databinding.ViewDataBinding;
8import android.graphics.drawable.Drawable;
9import android.support.annotation.AttrRes;
10import android.support.annotation.DrawableRes;
11import android.support.annotation.NonNull;
12import android.support.v4.content.ContextCompat;
13import android.support.v7.app.AlertDialog;
14import android.app.Dialog;
15import android.app.PendingIntent;
16import android.content.ActivityNotFoundException;
17import android.content.Context;
18import android.content.DialogInterface;
19import android.content.DialogInterface.OnClickListener;
20import android.content.Intent;
21import android.content.pm.PackageManager;
22import android.net.Uri;
23import android.os.Build;
24import android.os.Bundle;
25import android.support.v4.app.Fragment;
26import android.support.v4.app.FragmentManager;
27import android.support.v4.app.FragmentTransaction;
28import android.support.v4.app.ListFragment;
29import android.support.v4.view.MenuItemCompat;
30import android.support.v4.view.PagerAdapter;
31import android.support.v4.view.ViewPager;
32import android.support.v7.app.ActionBar;
33import android.text.Editable;
34import android.text.SpannableString;
35import android.text.Spanned;
36import android.text.TextWatcher;
37import android.text.style.TypefaceSpan;
38import android.util.Pair;
39import android.view.ContextMenu;
40import android.view.ContextMenu.ContextMenuInfo;
41import android.view.KeyEvent;
42import android.view.Menu;
43import android.view.MenuItem;
44import android.view.View;
45import android.view.ViewGroup;
46import android.view.inputmethod.InputMethodManager;
47import android.widget.AdapterView;
48import android.widget.AdapterView.AdapterContextMenuInfo;
49import android.widget.AdapterView.OnItemClickListener;
50import android.widget.ArrayAdapter;
51import android.widget.AutoCompleteTextView;
52import android.widget.CheckBox;
53import android.widget.Checkable;
54import android.widget.EditText;
55import android.widget.ListView;
56import android.widget.Spinner;
57import android.widget.TextView;
58import android.widget.Toast;
59
60import java.util.ArrayList;
61import java.util.Arrays;
62import java.util.Collection;
63import java.util.Collections;
64import java.util.List;
65import java.util.concurrent.atomic.AtomicBoolean;
66
67import eu.siacs.conversations.Config;
68import eu.siacs.conversations.R;
69import eu.siacs.conversations.databinding.ActivityStartConversationBinding;
70import eu.siacs.conversations.entities.Account;
71import eu.siacs.conversations.entities.Bookmark;
72import eu.siacs.conversations.entities.Contact;
73import eu.siacs.conversations.entities.Conversation;
74import eu.siacs.conversations.entities.ListItem;
75import eu.siacs.conversations.entities.Presence;
76import eu.siacs.conversations.services.XmppConnectionService.OnRosterUpdate;
77import eu.siacs.conversations.ui.adapter.KnownHostsAdapter;
78import eu.siacs.conversations.ui.adapter.ListItemAdapter;
79import eu.siacs.conversations.ui.service.EmojiService;
80import eu.siacs.conversations.ui.util.DelayedHintHelper;
81import eu.siacs.conversations.utils.XmppUri;
82import eu.siacs.conversations.xmpp.OnUpdateBlocklist;
83import eu.siacs.conversations.xmpp.XmppConnection;
84import rocks.xmpp.addr.Jid;
85
86public class StartConversationActivity extends XmppActivity implements OnRosterUpdate, OnUpdateBlocklist {
87
88 private final int REQUEST_SYNC_CONTACTS = 0x28cf;
89 private final int REQUEST_CREATE_CONFERENCE = 0x39da;
90 public int conference_context_id;
91 public int contact_context_id;
92 private ListPagerAdapter mListPagerAdapter;
93 private List<ListItem> contacts = new ArrayList<>();
94 private ListItemAdapter mContactsAdapter;
95 private List<ListItem> conferences = new ArrayList<>();
96 private ListItemAdapter mConferenceAdapter;
97 private List<String> mActivatedAccounts = new ArrayList<>();
98 private Collection<String> mKnownHosts;
99 private Collection<String> mKnownConferenceHosts;
100 private Invite mPendingInvite = null;
101 private EditText mSearchEditText;
102 private AtomicBoolean mRequestedContactsPermission = new AtomicBoolean(false);
103 private Dialog mCurrentDialog = null;
104 private boolean mHideOfflineContacts = false;
105 private MenuItem.OnActionExpandListener mOnActionExpandListener = new MenuItem.OnActionExpandListener() {
106
107 @Override
108 public boolean onMenuItemActionExpand(MenuItem item) {
109 mSearchEditText.post(() -> {
110 mSearchEditText.requestFocus();
111 InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
112 imm.showSoftInput(mSearchEditText, InputMethodManager.SHOW_IMPLICIT);
113 });
114
115 return true;
116 }
117
118 @Override
119 public boolean onMenuItemActionCollapse(MenuItem item) {
120 hideKeyboard();
121 mSearchEditText.setText("");
122 filter(null);
123 return true;
124 }
125 };
126 private TextWatcher mSearchTextWatcher = new TextWatcher() {
127
128 @Override
129 public void afterTextChanged(Editable editable) {
130 filter(editable.toString());
131 }
132
133 @Override
134 public void beforeTextChanged(CharSequence s, int start, int count,
135 int after) {
136 }
137
138 @Override
139 public void onTextChanged(CharSequence s, int start, int before, int count) {
140 }
141 };
142 private TextView.OnEditorActionListener mSearchDone = new TextView.OnEditorActionListener() {
143 @Override
144 public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
145 int pos = getSupportActionBar().getSelectedNavigationIndex();
146 if (pos == 0) {
147 if (contacts.size() == 1) {
148 openConversationForContact((Contact) contacts.get(0));
149 return true;
150 }
151 } else {
152 if (conferences.size() == 1) {
153 openConversationsForBookmark((Bookmark) conferences.get(0));
154 return true;
155 }
156 }
157 hideKeyboard();
158 mListPagerAdapter.requestFocus(pos);
159 return true;
160 }
161 };
162 private MenuItem mMenuSearchView;
163 private ListItemAdapter.OnTagClickedListener mOnTagClickedListener = new ListItemAdapter.OnTagClickedListener() {
164 @Override
165 public void onTagClicked(String tag) {
166 if (mMenuSearchView != null) {
167 mMenuSearchView.expandActionView();
168 mSearchEditText.setText("");
169 mSearchEditText.append(tag);
170 filter(tag);
171 }
172 }
173 };
174 private String mInitialJid;
175 private Pair<Integer, Intent> mPostponedActivityResult;
176 private Toast mToast;
177 private UiCallback<Conversation> mAdhocConferenceCallback = new UiCallback<Conversation>() {
178 @Override
179 public void success(final Conversation conversation) {
180 runOnUiThread(() -> {
181 hideToast();
182 switchToConversation(conversation);
183 });
184 }
185
186 @Override
187 public void error(final int errorCode, Conversation object) {
188 runOnUiThread(() -> replaceToast(getString(errorCode)));
189 }
190
191 @Override
192 public void userInputRequried(PendingIntent pi, Conversation object) {
193
194 }
195 };
196 private ActivityStartConversationBinding binding;
197 private ActionBar.TabListener mTabListener = new ActionBar.TabListener() {
198
199 @Override
200 public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction ft) {
201 return;
202 }
203
204 @Override
205 public void onTabSelected(ActionBar.Tab tab, FragmentTransaction ft) {
206 binding.startConversationViewPager.setCurrentItem(tab.getPosition());
207 onTabChanged();
208 }
209
210 @Override
211 public void onTabReselected(ActionBar.Tab tab, FragmentTransaction ft) {
212 return;
213 }
214 };
215 private ViewPager.SimpleOnPageChangeListener mOnPageChangeListener = new ViewPager.SimpleOnPageChangeListener() {
216 @Override
217 public void onPageSelected(int position) {
218 ActionBar actionBar = getSupportActionBar();
219 if (actionBar != null) {
220 actionBar.setSelectedNavigationItem(position);
221 }
222 onTabChanged();
223 }
224 };
225
226 public static void populateAccountSpinner(Context context, List<String> accounts, Spinner spinner) {
227 if (accounts.size() > 0) {
228 ArrayAdapter<String> adapter = new ArrayAdapter<>(context, R.layout.simple_list_item, accounts);
229 adapter.setDropDownViewResource(R.layout.simple_list_item);
230 spinner.setAdapter(adapter);
231 spinner.setEnabled(true);
232 } else {
233 ArrayAdapter<String> adapter = new ArrayAdapter<>(context,
234 R.layout.simple_list_item,
235 Arrays.asList(context.getString(R.string.no_accounts)));
236 adapter.setDropDownViewResource(R.layout.simple_list_item);
237 spinner.setAdapter(adapter);
238 spinner.setEnabled(false);
239 }
240 }
241
242 public static void launch(Context context) {
243 final Intent intent = new Intent(context, StartConversationActivity.class);
244 context.startActivity(intent);
245 }
246
247 protected void hideToast() {
248 if (mToast != null) {
249 mToast.cancel();
250 }
251 }
252
253 protected void replaceToast(String msg) {
254 hideToast();
255 mToast = Toast.makeText(this, msg, Toast.LENGTH_LONG);
256 mToast.show();
257 }
258
259 @Override
260 public void onRosterUpdate() {
261 this.refreshUi();
262 }
263
264 @Override
265 public void onCreate(Bundle savedInstanceState) {
266 super.onCreate(savedInstanceState);
267 new EmojiService(this).init();
268 this.binding = DataBindingUtil.setContentView(this, R.layout.activity_start_conversation);
269 this.binding.fab.setOnClickListener((v) -> {
270 if (getSupportActionBar().getSelectedNavigationIndex() == 0) {
271 String searchString = mSearchEditText != null ? mSearchEditText.getText().toString() : null;
272 if (searchString != null && !searchString.trim().isEmpty()) {
273 try {
274 Jid jid = Jid.of(searchString);
275 if (jid.getLocal() != null && jid.isBareJid() && jid.getDomain().contains(".")) {
276 showCreateContactDialog(jid.toString(), null);
277 return;
278 }
279 } catch (IllegalArgumentException ignored) {
280 //ignore and fall through
281 }
282 }
283 showCreateContactDialog(null, null);
284 } else {
285 showCreateConferenceDialog();
286 }
287 });
288 ActionBar actionBar = getSupportActionBar();
289 actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
290
291 ActionBar.Tab mContactsTab = actionBar.newTab().setText(R.string.contacts).setTabListener(mTabListener);
292 ActionBar.Tab mConferencesTab = actionBar.newTab().setText(R.string.conferences).setTabListener(mTabListener);
293 actionBar.addTab(mContactsTab);
294 actionBar.addTab(mConferencesTab);
295
296 binding.startConversationViewPager.setOnPageChangeListener(mOnPageChangeListener);
297 mListPagerAdapter = new ListPagerAdapter(getSupportFragmentManager());
298 binding.startConversationViewPager.setAdapter(mListPagerAdapter);
299
300 mConferenceAdapter = new ListItemAdapter(this, conferences);
301 mContactsAdapter = new ListItemAdapter(this, contacts);
302 mContactsAdapter.setOnTagClickedListener(this.mOnTagClickedListener);
303 this.mHideOfflineContacts = getPreferences().getBoolean("hide_offline", false);
304
305 }
306
307 @Override
308 public void onStart() {
309 super.onStart();
310 final int theme = findTheme();
311 if (this.mTheme != theme) {
312 recreate();
313 } else {
314 Intent i = getIntent();
315 if (i == null || !i.hasExtra(WelcomeActivity.EXTRA_INVITE_URI)) {
316 askForContactsPermissions();
317 }
318 }
319 mConferenceAdapter.refreshSettings();
320 mContactsAdapter.refreshSettings();
321 }
322
323 @Override
324 public void onStop() {
325 if (mCurrentDialog != null) {
326 mCurrentDialog.dismiss();
327 }
328 super.onStop();
329 }
330
331 @Override
332 public void onNewIntent(Intent intent) {
333 if (xmppConnectionServiceBound) {
334 handleIntent(intent);
335 } else {
336 setIntent(intent);
337 }
338 }
339
340 protected void openConversationForContact(int position) {
341 Contact contact = (Contact) contacts.get(position);
342 openConversationForContact(contact);
343 }
344
345 protected void openConversationForContact(Contact contact) {
346 Conversation conversation = xmppConnectionService.findOrCreateConversation(contact.getAccount(), contact.getJid(), false, true);
347 switchToConversation(conversation);
348 }
349
350 protected void openConversationForContact() {
351 int position = contact_context_id;
352 openConversationForContact(position);
353 }
354
355 protected void openConversationForBookmark() {
356 openConversationForBookmark(conference_context_id);
357 }
358
359 protected void openConversationForBookmark(int position) {
360 Bookmark bookmark = (Bookmark) conferences.get(position);
361 openConversationsForBookmark(bookmark);
362 }
363
364 protected void shareBookmarkUri() {
365 shareBookmarkUri(conference_context_id);
366 }
367
368 protected void shareBookmarkUri(int position) {
369 Bookmark bookmark = (Bookmark) conferences.get(position);
370 Intent shareIntent = new Intent();
371 shareIntent.setAction(Intent.ACTION_SEND);
372 shareIntent.putExtra(Intent.EXTRA_TEXT, "xmpp:" + bookmark.getJid().asBareJid().toEscapedString() + "?join");
373 shareIntent.setType("text/plain");
374 try {
375 startActivity(Intent.createChooser(shareIntent, getText(R.string.share_uri_with)));
376 } catch (ActivityNotFoundException e) {
377 Toast.makeText(this, R.string.no_application_to_share_uri, Toast.LENGTH_SHORT).show();
378 }
379 }
380
381 protected void openConversationsForBookmark(Bookmark bookmark) {
382 Jid jid = bookmark.getJid();
383 if (jid == null) {
384 Toast.makeText(this, R.string.invalid_jid, Toast.LENGTH_SHORT).show();
385 return;
386 }
387 Conversation conversation = xmppConnectionService.findOrCreateConversation(bookmark.getAccount(), jid, true, true, true);
388 bookmark.setConversation(conversation);
389 if (!bookmark.autojoin() && getPreferences().getBoolean("autojoin", getResources().getBoolean(R.bool.autojoin))) {
390 bookmark.setAutojoin(true);
391 xmppConnectionService.pushBookmarks(bookmark.getAccount());
392 }
393 switchToConversation(conversation);
394 }
395
396 protected void openDetailsForContact() {
397 int position = contact_context_id;
398 Contact contact = (Contact) contacts.get(position);
399 switchToContactDetails(contact);
400 }
401
402 protected void toggleContactBlock() {
403 final int position = contact_context_id;
404 BlockContactDialog.show(this, (Contact) contacts.get(position));
405 }
406
407 protected void deleteContact() {
408 final int position = contact_context_id;
409 final Contact contact = (Contact) contacts.get(position);
410 final AlertDialog.Builder builder = new AlertDialog.Builder(this);
411 builder.setNegativeButton(R.string.cancel, null);
412 builder.setTitle(R.string.action_delete_contact);
413 builder.setMessage(getString(R.string.remove_contact_text,
414 contact.getJid()));
415 builder.setPositiveButton(R.string.delete, new OnClickListener() {
416
417 @Override
418 public void onClick(DialogInterface dialog, int which) {
419 xmppConnectionService.deleteContactOnServer(contact);
420 filter(mSearchEditText.getText().toString());
421 }
422 });
423 builder.create().show();
424 }
425
426 protected void deleteConference() {
427 int position = conference_context_id;
428 final Bookmark bookmark = (Bookmark) conferences.get(position);
429
430 AlertDialog.Builder builder = new AlertDialog.Builder(this);
431 builder.setNegativeButton(R.string.cancel, null);
432 builder.setTitle(R.string.delete_bookmark);
433 builder.setMessage(getString(R.string.remove_bookmark_text,
434 bookmark.getJid()));
435 builder.setPositiveButton(R.string.delete, new OnClickListener() {
436
437 @Override
438 public void onClick(DialogInterface dialog, int which) {
439 bookmark.setConversation(null);
440 Account account = bookmark.getAccount();
441 account.getBookmarks().remove(bookmark);
442 xmppConnectionService.pushBookmarks(account);
443 filter(mSearchEditText.getText().toString());
444 }
445 });
446 builder.create().show();
447
448 }
449
450 @SuppressLint("InflateParams")
451 protected void showCreateContactDialog(final String prefilledJid, final Invite invite) {
452 EnterJidDialog dialog = new EnterJidDialog(
453 this, mKnownHosts, mActivatedAccounts,
454 getString(R.string.dialog_title_create_contact), getString(R.string.create),
455 prefilledJid, null, invite == null || !invite.hasFingerprints()
456 );
457
458 dialog.setOnEnterJidDialogPositiveListener((accountJid, contactJid) -> {
459 if (!xmppConnectionServiceBound) {
460 return false;
461 }
462
463 final Account account = xmppConnectionService.findAccountByJid(accountJid);
464 if (account == null) {
465 return true;
466 }
467
468 final Contact contact = account.getRoster().getContact(contactJid);
469 if (invite != null && invite.getName() != null) {
470 contact.setServerName(invite.getName());
471 }
472 if (contact.isSelf()) {
473 switchToConversation(contact, null);
474 return true;
475 } else if (contact.showInRoster()) {
476 throw new EnterJidDialog.JidError(getString(R.string.contact_already_exists));
477 } else {
478 xmppConnectionService.createContact(contact, true);
479 if (invite != null && invite.hasFingerprints()) {
480 xmppConnectionService.verifyFingerprints(contact, invite.getFingerprints());
481 }
482 switchToConversation(contact, invite == null ? null : invite.getBody());
483 return true;
484 }
485 });
486
487 mCurrentDialog = dialog.show();
488 }
489
490 @SuppressLint("InflateParams")
491 protected void showJoinConferenceDialog(final String prefilledJid) {
492 final AlertDialog.Builder builder = new AlertDialog.Builder(this);
493 builder.setTitle(R.string.dialog_title_join_conference);
494 final View dialogView = getLayoutInflater().inflate(R.layout.join_conference_dialog, null);
495 final Spinner spinner = dialogView.findViewById(R.id.account);
496 final AutoCompleteTextView jid = dialogView.findViewById(R.id.jid);
497 DelayedHintHelper.setHint(R.string.conference_address_example, jid);
498 jid.setAdapter(new KnownHostsAdapter(this, R.layout.simple_list_item, mKnownConferenceHosts));
499 if (prefilledJid != null) {
500 jid.append(prefilledJid);
501 }
502 populateAccountSpinner(this, mActivatedAccounts, spinner);
503 final Checkable bookmarkCheckBox = (CheckBox) dialogView
504 .findViewById(R.id.bookmark);
505 builder.setView(dialogView);
506 builder.setNegativeButton(R.string.cancel, null);
507 builder.setPositiveButton(R.string.join, null);
508 final AlertDialog dialog = builder.create();
509 dialog.show();
510 mCurrentDialog = dialog;
511 dialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(v -> {
512 if (!xmppConnectionServiceBound) {
513 return;
514 }
515 final Account account = getSelectedAccount(spinner);
516 if (account == null) {
517 return;
518 }
519 final Jid conferenceJid;
520 try {
521 conferenceJid = Jid.of(jid.getText().toString());
522 } catch (final IllegalArgumentException e) {
523 jid.setError(getString(R.string.invalid_jid));
524 return;
525 }
526
527 if (bookmarkCheckBox.isChecked()) {
528 if (account.hasBookmarkFor(conferenceJid)) {
529 jid.setError(getString(R.string.bookmark_already_exists));
530 } else {
531 final Bookmark bookmark = new Bookmark(account, conferenceJid.asBareJid());
532 bookmark.setAutojoin(getPreferences().getBoolean("autojoin", getResources().getBoolean(R.bool.autojoin)));
533 String nick = conferenceJid.getResource();
534 if (nick != null && !nick.isEmpty()) {
535 bookmark.setNick(nick);
536 }
537 account.getBookmarks().add(bookmark);
538 xmppConnectionService.pushBookmarks(account);
539 final Conversation conversation = xmppConnectionService
540 .findOrCreateConversation(account, conferenceJid, true, true, true);
541 bookmark.setConversation(conversation);
542 dialog.dismiss();
543 mCurrentDialog = null;
544 switchToConversation(conversation);
545 }
546 } else {
547 final Conversation conversation = xmppConnectionService
548 .findOrCreateConversation(account, conferenceJid, true, true, true);
549 dialog.dismiss();
550 mCurrentDialog = null;
551 switchToConversation(conversation);
552 }
553 });
554 }
555
556 private void showCreateConferenceDialog() {
557 final AlertDialog.Builder builder = new AlertDialog.Builder(this);
558 builder.setTitle(R.string.dialog_title_create_conference);
559 final View dialogView = getLayoutInflater().inflate(R.layout.create_conference_dialog, null);
560 final Spinner spinner = dialogView.findViewById(R.id.account);
561 final EditText subject = dialogView.findViewById(R.id.subject);
562 populateAccountSpinner(this, mActivatedAccounts, spinner);
563 builder.setView(dialogView);
564 builder.setPositiveButton(R.string.choose_participants, new OnClickListener() {
565 @Override
566 public void onClick(DialogInterface dialog, int which) {
567 if (!xmppConnectionServiceBound) {
568 return;
569 }
570 final Account account = getSelectedAccount(spinner);
571 if (account == null) {
572 return;
573 }
574 Intent intent = new Intent(getApplicationContext(), ChooseContactActivity.class);
575 intent.putExtra("multiple", true);
576 intent.putExtra("show_enter_jid", true);
577 intent.putExtra("subject", subject.getText().toString());
578 intent.putExtra(EXTRA_ACCOUNT, account.getJid().asBareJid().toString());
579 intent.putExtra(ChooseContactActivity.EXTRA_TITLE_RES_ID, R.string.choose_participants);
580 startActivityForResult(intent, REQUEST_CREATE_CONFERENCE);
581 }
582 });
583 builder.setNegativeButton(R.string.cancel, null);
584 mCurrentDialog = builder.create();
585 mCurrentDialog.show();
586 }
587
588 private Account getSelectedAccount(Spinner spinner) {
589 if (!spinner.isEnabled()) {
590 return null;
591 }
592 Jid jid;
593 try {
594 if (Config.DOMAIN_LOCK != null) {
595 jid = Jid.of((String) spinner.getSelectedItem(), Config.DOMAIN_LOCK, null);
596 } else {
597 jid = Jid.of((String) spinner.getSelectedItem());
598 }
599 } catch (final IllegalArgumentException e) {
600 return null;
601 }
602 return xmppConnectionService.findAccountByJid(jid);
603 }
604
605 protected void switchToConversation(Contact contact, String body) {
606 Conversation conversation = xmppConnectionService
607 .findOrCreateConversation(contact.getAccount(),
608 contact.getJid(), false, true);
609 switchToConversation(conversation, body, false);
610 }
611
612 @Override
613 public boolean onCreateOptionsMenu(Menu menu) {
614 getMenuInflater().inflate(R.menu.start_conversation, menu);
615 MenuItem menuHideOffline = menu.findItem(R.id.action_hide_offline);
616 MenuItem joinGroupChat = menu.findItem(R.id.action_join_conference);
617 MenuItem qrCodeScanMenuItem = menu.findItem(R.id.action_scan_qr_code);
618 ActionBar bar = getSupportActionBar();
619 joinGroupChat.setVisible(bar != null && bar.getSelectedNavigationIndex() == 1);
620 qrCodeScanMenuItem.setVisible(isCameraFeatureAvailable());
621 menuHideOffline.setChecked(this.mHideOfflineContacts);
622 mMenuSearchView = menu.findItem(R.id.action_search);
623 mMenuSearchView.setOnActionExpandListener(mOnActionExpandListener);
624 View mSearchView = mMenuSearchView.getActionView();
625 mSearchEditText = mSearchView.findViewById(R.id.search_field);
626 mSearchEditText.addTextChangedListener(mSearchTextWatcher);
627 mSearchEditText.setOnEditorActionListener(mSearchDone);
628 if (mInitialJid != null) {
629 mMenuSearchView.expandActionView();
630 mSearchEditText.append(mInitialJid);
631 filter(mInitialJid);
632 }
633 return super.onCreateOptionsMenu(menu);
634 }
635
636 @Override
637 public boolean onOptionsItemSelected(MenuItem item) {
638 switch (item.getItemId()) {
639 case R.id.action_join_conference:
640 showJoinConferenceDialog(null);
641 return true;
642 case R.id.action_scan_qr_code:
643 UriHandlerActivity.scan(this);
644 return true;
645 case R.id.action_hide_offline:
646 mHideOfflineContacts = !item.isChecked();
647 getPreferences().edit().putBoolean("hide_offline", mHideOfflineContacts).commit();
648 if (mSearchEditText != null) {
649 filter(mSearchEditText.getText().toString());
650 }
651 invalidateOptionsMenu();
652 }
653 return super.onOptionsItemSelected(item);
654 }
655
656 @Override
657 public boolean onKeyUp(int keyCode, KeyEvent event) {
658 if (keyCode == KeyEvent.KEYCODE_SEARCH && !event.isLongPress()) {
659 openSearch();
660 return true;
661 }
662 int c = event.getUnicodeChar();
663 if (c > 32) {
664 if (mSearchEditText != null && !mSearchEditText.isFocused()) {
665 openSearch();
666 mSearchEditText.append(Character.toString((char) c));
667 return true;
668 }
669 }
670 return super.onKeyUp(keyCode, event);
671 }
672
673 private void openSearch() {
674 if (mMenuSearchView != null) {
675 mMenuSearchView.expandActionView();
676 }
677 }
678
679 @Override
680 public void onActivityResult(int requestCode, int resultCode, Intent intent) {
681 if (resultCode == RESULT_OK) {
682 if (xmppConnectionServiceBound) {
683 this.mPostponedActivityResult = null;
684 if (requestCode == REQUEST_CREATE_CONFERENCE) {
685 Account account = extractAccount(intent);
686 final String subject = intent.getStringExtra("subject");
687 List<Jid> jids = new ArrayList<>();
688 if (intent.getBooleanExtra("multiple", false)) {
689 String[] toAdd = intent.getStringArrayExtra("contacts");
690 for (String item : toAdd) {
691 try {
692 jids.add(Jid.of(item));
693 } catch (IllegalArgumentException e) {
694 //ignored
695 }
696 }
697 } else {
698 try {
699 jids.add(Jid.of(intent.getStringExtra("contact")));
700 } catch (Exception e) {
701 //ignored
702 }
703 }
704 if (account != null && jids.size() > 0) {
705 if (xmppConnectionService.createAdhocConference(account, subject, jids, mAdhocConferenceCallback)) {
706 mToast = Toast.makeText(this, R.string.creating_conference, Toast.LENGTH_LONG);
707 mToast.show();
708 }
709 }
710 }
711 } else {
712 this.mPostponedActivityResult = new Pair<>(requestCode, intent);
713 }
714 }
715 super.onActivityResult(requestCode, requestCode, intent);
716 }
717
718 private void askForContactsPermissions() {
719 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
720 if (checkSelfPermission(Manifest.permission.READ_CONTACTS) != PackageManager.PERMISSION_GRANTED) {
721 if (mRequestedContactsPermission.compareAndSet(false, true)) {
722 if (shouldShowRequestPermissionRationale(Manifest.permission.READ_CONTACTS)) {
723 AlertDialog.Builder builder = new AlertDialog.Builder(this);
724 builder.setTitle(R.string.sync_with_contacts);
725 builder.setMessage(R.string.sync_with_contacts_long);
726 builder.setPositiveButton(R.string.next, new OnClickListener() {
727 @Override
728 public void onClick(DialogInterface dialog, int which) {
729 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
730 requestPermissions(new String[]{Manifest.permission.READ_CONTACTS}, REQUEST_SYNC_CONTACTS);
731 }
732 }
733 });
734 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {
735 builder.setOnDismissListener(new DialogInterface.OnDismissListener() {
736 @Override
737 public void onDismiss(DialogInterface dialog) {
738 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
739 requestPermissions(new String[]{Manifest.permission.READ_CONTACTS}, REQUEST_SYNC_CONTACTS);
740 }
741 }
742 });
743 }
744 builder.create().show();
745 } else {
746 requestPermissions(new String[]{Manifest.permission.READ_CONTACTS}, 0);
747 }
748 }
749 }
750 }
751 }
752
753 @Override
754 public void onRequestPermissionsResult(int requestCode, @NonNull String permissions[], @NonNull int[] grantResults) {
755 if (grantResults.length > 0)
756 if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
757 ScanActivity.onRequestPermissionResult(this, requestCode, grantResults);
758 if (requestCode == REQUEST_SYNC_CONTACTS && xmppConnectionServiceBound) {
759 xmppConnectionService.loadPhoneContacts();
760 }
761 }
762 }
763
764 @Override
765 protected void onBackendConnected() {
766 if (mPostponedActivityResult != null) {
767 onActivityResult(mPostponedActivityResult.first, RESULT_OK, mPostponedActivityResult.second);
768 this.mPostponedActivityResult = null;
769 }
770 this.mActivatedAccounts.clear();
771 for (Account account : xmppConnectionService.getAccounts()) {
772 if (account.getStatus() != Account.State.DISABLED) {
773 if (Config.DOMAIN_LOCK != null) {
774 this.mActivatedAccounts.add(account.getJid().getLocal());
775 } else {
776 this.mActivatedAccounts.add(account.getJid().asBareJid().toString());
777 }
778 }
779 }
780 final Intent intent = getIntent();
781 final ActionBar ab = getSupportActionBar();
782 boolean init = intent != null && intent.getBooleanExtra("init", false);
783 boolean noConversations = xmppConnectionService.getConversations().size() == 0;
784 if ((init || noConversations) && ab != null) {
785 ab.setDisplayShowHomeEnabled(false);
786 ab.setDisplayHomeAsUpEnabled(false);
787 ab.setHomeButtonEnabled(false);
788 }
789 this.mKnownHosts = xmppConnectionService.getKnownHosts();
790 this.mKnownConferenceHosts = xmppConnectionService.getKnownConferenceHosts();
791 if (this.mPendingInvite != null) {
792 mPendingInvite.invite();
793 this.mPendingInvite = null;
794 filter(null);
795 } else if (!handleIntent(getIntent())) {
796 if (mSearchEditText != null) {
797 filter(mSearchEditText.getText().toString());
798 } else {
799 filter(null);
800 }
801 } else {
802 filter(null);
803 }
804 setIntent(null);
805 }
806
807 protected boolean handleIntent(Intent intent) {
808 if (intent == null) {
809 return false;
810 }
811 final String inviteUri = intent.getStringExtra(WelcomeActivity.EXTRA_INVITE_URI);
812 if (inviteUri != null) {
813 Invite invite = new Invite(inviteUri);
814 if (invite.isJidValid()) {
815 return invite.invite();
816 }
817 }
818 if (intent.getAction() == null) {
819 return false;
820 }
821 switch (intent.getAction()) {
822 case Intent.ACTION_SENDTO:
823 case Intent.ACTION_VIEW:
824 Uri uri = intent.getData();
825 if (uri != null) {
826 Invite invite = new Invite(intent.getData(), intent.getBooleanExtra("scanned", false));
827 invite.account = intent.getStringExtra("account");
828 return invite.invite();
829 } else {
830 return false;
831 }
832 }
833 return false;
834 }
835
836 private boolean handleJid(Invite invite) {
837 List<Contact> contacts = xmppConnectionService.findContacts(invite.getJid(), invite.account);
838 if (invite.isAction(XmppUri.ACTION_JOIN)) {
839 Conversation muc = xmppConnectionService.findFirstMuc(invite.getJid());
840 if (muc != null) {
841 switchToConversation(muc, invite.getBody(), false);
842 return true;
843 } else {
844 showJoinConferenceDialog(invite.getJid().asBareJid().toString());
845 return false;
846 }
847 } else if (contacts.size() == 0) {
848 showCreateContactDialog(invite.getJid().toString(), invite);
849 return false;
850 } else if (contacts.size() == 1) {
851 Contact contact = contacts.get(0);
852 if (!invite.isSafeSource() && invite.hasFingerprints()) {
853 displayVerificationWarningDialog(contact, invite);
854 } else {
855 if (invite.hasFingerprints()) {
856 if (xmppConnectionService.verifyFingerprints(contact, invite.getFingerprints())) {
857 Toast.makeText(this, R.string.verified_fingerprints, Toast.LENGTH_SHORT).show();
858 }
859 }
860 if (invite.account != null) {
861 xmppConnectionService.getShortcutService().report(contact);
862 }
863 switchToConversation(contact, invite.getBody());
864 }
865 return true;
866 } else {
867 if (mMenuSearchView != null) {
868 mMenuSearchView.expandActionView();
869 mSearchEditText.setText("");
870 mSearchEditText.append(invite.getJid().toString());
871 filter(invite.getJid().toString());
872 } else {
873 mInitialJid = invite.getJid().toString();
874 }
875 return true;
876 }
877 }
878
879 private void displayVerificationWarningDialog(final Contact contact, final Invite invite) {
880 AlertDialog.Builder builder = new AlertDialog.Builder(this);
881 builder.setTitle(R.string.verify_omemo_keys);
882 View view = getLayoutInflater().inflate(R.layout.dialog_verify_fingerprints, null);
883 final CheckBox isTrustedSource = (CheckBox) view.findViewById(R.id.trusted_source);
884 TextView warning = (TextView) view.findViewById(R.id.warning);
885 String jid = contact.getJid().asBareJid().toString();
886 SpannableString spannable = new SpannableString(getString(R.string.verifying_omemo_keys_trusted_source, jid, contact.getDisplayName()));
887 int start = spannable.toString().indexOf(jid);
888 if (start >= 0) {
889 spannable.setSpan(new TypefaceSpan("monospace"), start, start + jid.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
890 }
891 warning.setText(spannable);
892 builder.setView(view);
893 builder.setPositiveButton(R.string.confirm, (dialog, which) -> {
894 if (isTrustedSource.isChecked() && invite.hasFingerprints()) {
895 xmppConnectionService.verifyFingerprints(contact, invite.getFingerprints());
896 }
897 switchToConversation(contact, invite.getBody());
898 });
899 builder.setNegativeButton(R.string.cancel, (dialog, which) -> StartConversationActivity.this.finish());
900 AlertDialog dialog = builder.create();
901 dialog.setCanceledOnTouchOutside(false);
902 dialog.setOnCancelListener(dialog1 -> StartConversationActivity.this.finish());
903 dialog.show();
904 }
905
906 protected void filter(String needle) {
907 if (xmppConnectionServiceBound) {
908 this.filterContacts(needle);
909 this.filterConferences(needle);
910 }
911 }
912
913 protected void filterContacts(String needle) {
914 this.contacts.clear();
915 for (Account account : xmppConnectionService.getAccounts()) {
916 if (account.getStatus() != Account.State.DISABLED) {
917 for (Contact contact : account.getRoster().getContacts()) {
918 Presence.Status s = contact.getShownStatus();
919 if (contact.showInRoster() && contact.match(this, needle)
920 && (!this.mHideOfflineContacts
921 || (needle != null && !needle.trim().isEmpty())
922 || s.compareTo(Presence.Status.OFFLINE) < 0)) {
923 this.contacts.add(contact);
924 }
925 }
926 }
927 }
928 Collections.sort(this.contacts);
929 mContactsAdapter.notifyDataSetChanged();
930 }
931
932 protected void filterConferences(String needle) {
933 this.conferences.clear();
934 for (Account account : xmppConnectionService.getAccounts()) {
935 if (account.getStatus() != Account.State.DISABLED) {
936 for (Bookmark bookmark : account.getBookmarks()) {
937 if (bookmark.match(this, needle)) {
938 this.conferences.add(bookmark);
939 }
940 }
941 }
942 }
943 Collections.sort(this.conferences);
944 mConferenceAdapter.notifyDataSetChanged();
945 }
946
947 private void onTabChanged() {
948 @DrawableRes final int fabDrawable;
949 if (getSupportActionBar().getSelectedNavigationIndex() == 0) {
950 fabDrawable = R.drawable.ic_person_add_white_24dp;
951 } else {
952 fabDrawable = R.drawable.ic_group_add_white_24dp;
953 }
954 binding.fab.setImageResource(fabDrawable);
955 invalidateOptionsMenu();
956 }
957
958 @Override
959 public void OnUpdateBlocklist(final Status status) {
960 refreshUi();
961 }
962
963 @Override
964 protected void refreshUiReal() {
965 if (mSearchEditText != null) {
966 filter(mSearchEditText.getText().toString());
967 }
968 }
969
970 public static class MyListFragment extends ListFragment {
971 private AdapterView.OnItemClickListener mOnItemClickListener;
972 private int mResContextMenu;
973
974 public void setContextMenu(final int res) {
975 this.mResContextMenu = res;
976 }
977
978 @Override
979 public void onListItemClick(final ListView l, final View v, final int position, final long id) {
980 if (mOnItemClickListener != null) {
981 mOnItemClickListener.onItemClick(l, v, position, id);
982 }
983 }
984
985 public void setOnListItemClickListener(AdapterView.OnItemClickListener l) {
986 this.mOnItemClickListener = l;
987 }
988
989 @Override
990 public void onViewCreated(@NonNull final View view, final Bundle savedInstanceState) {
991 super.onViewCreated(view, savedInstanceState);
992 registerForContextMenu(getListView());
993 getListView().setFastScrollEnabled(true);
994 getListView().setDivider(null);
995 getListView().setDividerHeight(0);
996 }
997
998 @Override
999 public void onCreateContextMenu(final ContextMenu menu, final View v,
1000 final ContextMenuInfo menuInfo) {
1001 super.onCreateContextMenu(menu, v, menuInfo);
1002 final StartConversationActivity activity = (StartConversationActivity) getActivity();
1003 if (activity == null) {
1004 return;
1005 }
1006 activity.getMenuInflater().inflate(mResContextMenu, menu);
1007 final AdapterView.AdapterContextMenuInfo acmi = (AdapterContextMenuInfo) menuInfo;
1008 if (mResContextMenu == R.menu.conference_context) {
1009 activity.conference_context_id = acmi.position;
1010 } else if (mResContextMenu == R.menu.contact_context) {
1011 activity.contact_context_id = acmi.position;
1012 final Contact contact = (Contact) activity.contacts.get(acmi.position);
1013 final MenuItem blockUnblockItem = menu.findItem(R.id.context_contact_block_unblock);
1014 final MenuItem showContactDetailsItem = menu.findItem(R.id.context_contact_details);
1015 if (contact.isSelf()) {
1016 showContactDetailsItem.setVisible(false);
1017 }
1018 XmppConnection xmpp = contact.getAccount().getXmppConnection();
1019 if (xmpp != null && xmpp.getFeatures().blocking() && !contact.isSelf()) {
1020 if (contact.isBlocked()) {
1021 blockUnblockItem.setTitle(R.string.unblock_contact);
1022 } else {
1023 blockUnblockItem.setTitle(R.string.block_contact);
1024 }
1025 } else {
1026 blockUnblockItem.setVisible(false);
1027 }
1028 }
1029 }
1030
1031 @Override
1032 public boolean onContextItemSelected(final MenuItem item) {
1033 StartConversationActivity activity = (StartConversationActivity) getActivity();
1034 if (activity == null) {
1035 return true;
1036 }
1037 switch (item.getItemId()) {
1038 case R.id.context_start_conversation:
1039 activity.openConversationForContact();
1040 break;
1041 case R.id.context_contact_details:
1042 activity.openDetailsForContact();
1043 break;
1044 case R.id.context_contact_block_unblock:
1045 activity.toggleContactBlock();
1046 break;
1047 case R.id.context_delete_contact:
1048 activity.deleteContact();
1049 break;
1050 case R.id.context_join_conference:
1051 activity.openConversationForBookmark();
1052 break;
1053 case R.id.context_share_uri:
1054 activity.shareBookmarkUri();
1055 break;
1056 case R.id.context_delete_conference:
1057 activity.deleteConference();
1058 }
1059 return true;
1060 }
1061 }
1062
1063 public class ListPagerAdapter extends PagerAdapter {
1064 FragmentManager fragmentManager;
1065 MyListFragment[] fragments;
1066
1067 public ListPagerAdapter(FragmentManager fm) {
1068 fragmentManager = fm;
1069 fragments = new MyListFragment[2];
1070 }
1071
1072 public void requestFocus(int pos) {
1073 if (fragments.length > pos) {
1074 fragments[pos].getListView().requestFocus();
1075 }
1076 }
1077
1078 @Override
1079 public void destroyItem(@NonNull ViewGroup container, int position, @NonNull Object object) {
1080 assert (0 <= position && position < fragments.length);
1081 FragmentTransaction trans = fragmentManager.beginTransaction();
1082 trans.remove(fragments[position]);
1083 trans.commit();
1084 fragments[position] = null;
1085 }
1086
1087 @Override
1088 public Fragment instantiateItem(@NonNull ViewGroup container, int position) {
1089 Fragment fragment = getItem(position);
1090 FragmentTransaction trans = fragmentManager.beginTransaction();
1091 trans.add(container.getId(), fragment, "fragment:" + position);
1092 trans.commit();
1093 return fragment;
1094 }
1095
1096 @Override
1097 public int getCount() {
1098 return fragments.length;
1099 }
1100
1101 @Override
1102 public boolean isViewFromObject(@NonNull View view, @NonNull Object fragment) {
1103 return ((Fragment) fragment).getView() == view;
1104 }
1105
1106 public Fragment getItem(int position) {
1107 assert (0 <= position && position < fragments.length);
1108 if (fragments[position] == null) {
1109 final MyListFragment listFragment = new MyListFragment();
1110 if (position == 1) {
1111 listFragment.setListAdapter(mConferenceAdapter);
1112 listFragment.setContextMenu(R.menu.conference_context);
1113 listFragment.setOnListItemClickListener((arg0, arg1, p, arg3) -> openConversationForBookmark(p));
1114 } else {
1115
1116 listFragment.setListAdapter(mContactsAdapter);
1117 listFragment.setContextMenu(R.menu.contact_context);
1118 listFragment.setOnListItemClickListener((arg0, arg1, p, arg3) -> openConversationForContact(p));
1119 }
1120 fragments[position] = listFragment;
1121 }
1122 return fragments[position];
1123 }
1124 }
1125
1126 private class Invite extends XmppUri {
1127
1128 public String account;
1129
1130 public Invite(final Uri uri) {
1131 super(uri);
1132 }
1133
1134 public Invite(final String uri) {
1135 super(uri);
1136 }
1137
1138 public Invite(Uri uri, boolean safeSource) {
1139 super(uri, safeSource);
1140 }
1141
1142 boolean invite() {
1143 if (!isJidValid()) {
1144 Toast.makeText(StartConversationActivity.this, R.string.invalid_jid, Toast.LENGTH_SHORT).show();
1145 return false;
1146 }
1147 if (getJid() != null) {
1148 return handleJid(this);
1149 }
1150 return false;
1151 }
1152 }
1153}