ContactsController.java 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975
  1. /*
  2. * Nextcloud Talk application
  3. *
  4. * @author Mario Danic
  5. * Copyright (C) 2017 Mario Danic (mario@lovelyhq.com)
  6. *
  7. * This program is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation, either version 3 of the License, or
  10. * at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  19. */
  20. package com.nextcloud.talk.controllers;
  21. import android.app.SearchManager;
  22. import android.content.Context;
  23. import android.content.Intent;
  24. import android.os.Bundle;
  25. import android.os.Handler;
  26. import android.text.InputType;
  27. import android.text.TextUtils;
  28. import android.util.Log;
  29. import android.view.LayoutInflater;
  30. import android.view.Menu;
  31. import android.view.MenuInflater;
  32. import android.view.MenuItem;
  33. import android.view.View;
  34. import android.view.ViewGroup;
  35. import android.view.ViewTreeObserver;
  36. import android.view.inputmethod.EditorInfo;
  37. import android.widget.RelativeLayout;
  38. import com.bluelinelabs.conductor.RouterTransaction;
  39. import com.bluelinelabs.conductor.changehandler.HorizontalChangeHandler;
  40. import com.bluelinelabs.conductor.changehandler.VerticalChangeHandler;
  41. import com.bluelinelabs.logansquare.LoganSquare;
  42. import com.google.android.material.bottomnavigation.BottomNavigationView;
  43. import com.kennyc.bottomsheet.BottomSheet;
  44. import com.nextcloud.talk.R;
  45. import com.nextcloud.talk.activities.MagicCallActivity;
  46. import com.nextcloud.talk.adapters.items.ProgressItem;
  47. import com.nextcloud.talk.adapters.items.UserHeaderItem;
  48. import com.nextcloud.talk.adapters.items.UserItem;
  49. import com.nextcloud.talk.api.NcApi;
  50. import com.nextcloud.talk.application.NextcloudTalkApplication;
  51. import com.nextcloud.talk.controllers.base.BaseController;
  52. import com.nextcloud.talk.controllers.bottomsheet.OperationsMenuController;
  53. import com.nextcloud.talk.events.BottomSheetLockEvent;
  54. import com.nextcloud.talk.models.RetrofitBucket;
  55. import com.nextcloud.talk.models.database.UserEntity;
  56. import com.nextcloud.talk.models.json.autocomplete.AutocompleteOverall;
  57. import com.nextcloud.talk.models.json.autocomplete.AutocompleteUser;
  58. import com.nextcloud.talk.models.json.participants.Participant;
  59. import com.nextcloud.talk.models.json.rooms.Conversation;
  60. import com.nextcloud.talk.models.json.rooms.RoomOverall;
  61. import com.nextcloud.talk.models.json.sharees.Sharee;
  62. import com.nextcloud.talk.models.json.sharees.ShareesOverall;
  63. import com.nextcloud.talk.utils.ApiUtils;
  64. import com.nextcloud.talk.utils.bundle.BundleKeys;
  65. import com.nextcloud.talk.utils.database.user.UserUtils;
  66. import org.greenrobot.eventbus.EventBus;
  67. import org.greenrobot.eventbus.Subscribe;
  68. import org.greenrobot.eventbus.ThreadMode;
  69. import org.parceler.Parcels;
  70. import java.util.ArrayList;
  71. import java.util.Collections;
  72. import java.util.HashMap;
  73. import java.util.HashSet;
  74. import java.util.List;
  75. import java.util.Map;
  76. import java.util.Set;
  77. import javax.inject.Inject;
  78. import androidx.annotation.NonNull;
  79. import androidx.annotation.Nullable;
  80. import androidx.appcompat.widget.SearchView;
  81. import androidx.core.view.MenuItemCompat;
  82. import androidx.recyclerview.widget.RecyclerView;
  83. import androidx.swiperefreshlayout.widget.SwipeRefreshLayout;
  84. import autodagger.AutoInjector;
  85. import butterknife.BindView;
  86. import butterknife.OnClick;
  87. import butterknife.Optional;
  88. import eu.davidea.fastscroller.FastScroller;
  89. import eu.davidea.flexibleadapter.FlexibleAdapter;
  90. import eu.davidea.flexibleadapter.SelectableAdapter;
  91. import eu.davidea.flexibleadapter.common.SmoothScrollLinearLayoutManager;
  92. import eu.davidea.flexibleadapter.items.AbstractFlexibleItem;
  93. import eu.davidea.flexibleadapter.items.IFlexible;
  94. import eu.davidea.flipview.FlipView;
  95. import io.reactivex.Observer;
  96. import io.reactivex.android.schedulers.AndroidSchedulers;
  97. import io.reactivex.disposables.Disposable;
  98. import io.reactivex.schedulers.Schedulers;
  99. import okhttp3.ResponseBody;
  100. import retrofit2.HttpException;
  101. @AutoInjector(NextcloudTalkApplication.class)
  102. public class ContactsController extends BaseController implements SearchView.OnQueryTextListener,
  103. FlexibleAdapter.OnItemClickListener, FastScroller.OnScrollStateChangeListener, FlexibleAdapter.EndlessScrollListener {
  104. public static final String TAG = "ContactsController";
  105. private static final String KEY_SEARCH_QUERY = "ContactsController.searchQuery";
  106. @Nullable
  107. @BindView(R.id.initial_relative_layout)
  108. public RelativeLayout initialRelativeLayout;
  109. @Nullable
  110. @BindView(R.id.secondary_relative_layout)
  111. public RelativeLayout secondaryRelativeLayout;
  112. @Inject
  113. UserUtils userUtils;
  114. @Inject
  115. EventBus eventBus;
  116. @BindView(R.id.recycler_view)
  117. RecyclerView recyclerView;
  118. @BindView(R.id.swipe_refresh_layout)
  119. SwipeRefreshLayout swipeRefreshLayout;
  120. @BindView(R.id.fast_scroller)
  121. FastScroller fastScroller;
  122. @Inject
  123. NcApi ncApi;
  124. private String credentials;
  125. private UserEntity currentUser;
  126. private Disposable contactsQueryDisposable;
  127. private Disposable cacheQueryDisposable;
  128. private FlexibleAdapter adapter;
  129. private List<AbstractFlexibleItem> contactItems = new ArrayList<>();
  130. private BottomSheet bottomSheet;
  131. private View view;
  132. private int currentPage;
  133. private int currentSearchPage;
  134. private SmoothScrollLinearLayoutManager layoutManager;
  135. private MenuItem searchItem;
  136. private SearchView searchView;
  137. private String searchQuery;
  138. private boolean isNewConversationView;
  139. private boolean isPublicCall;
  140. private HashMap<String, UserHeaderItem> userHeaderItems = new HashMap<>();
  141. private boolean alreadyFetching = false;
  142. private boolean canFetchFurther = true;
  143. private boolean canFetchSearchFurther = true;
  144. private MenuItem doneMenuItem;
  145. public ContactsController() {
  146. super();
  147. setHasOptionsMenu(true);
  148. }
  149. public ContactsController(Bundle args) {
  150. super(args);
  151. setHasOptionsMenu(true);
  152. if (args.containsKey(BundleKeys.KEY_NEW_CONVERSATION)) {
  153. isNewConversationView = true;
  154. }
  155. }
  156. @Override
  157. protected View inflateView(@NonNull LayoutInflater inflater, @NonNull ViewGroup container) {
  158. if (isNewConversationView) {
  159. return inflater.inflate(R.layout.controller_contacts_rv, container, false);
  160. } else {
  161. return inflater.inflate(R.layout.controller_generic_rv, container, false);
  162. }
  163. }
  164. @Override
  165. protected void onAttach(@NonNull View view) {
  166. super.onAttach(view);
  167. eventBus.register(this);
  168. if (isNewConversationView) {
  169. toggleNewCallHeaderVisibility(!isPublicCall);
  170. checkAndHandleDoneMenuItem();
  171. if (getActionBar() != null) {
  172. getActionBar().setDisplayHomeAsUpEnabled(true);
  173. }
  174. }
  175. }
  176. @Override
  177. protected void onViewBound(@NonNull View view) {
  178. super.onViewBound(view);
  179. NextcloudTalkApplication.getSharedApplication().getComponentApplication().inject(this);
  180. FlipView.resetLayoutAnimationDelay(true, 1000L);
  181. FlipView.stopLayoutAnimation();
  182. currentUser = userUtils.getCurrentUser();
  183. if (currentUser != null) {
  184. credentials = ApiUtils.getCredentials(currentUser.getUsername(), currentUser.getToken());
  185. }
  186. if (adapter == null) {
  187. adapter = new FlexibleAdapter<>(contactItems, getActivity(), false);
  188. if (currentUser != null) {
  189. fetchData(true);
  190. }
  191. }
  192. setupAdapter();
  193. prepareViews();
  194. }
  195. private void setupAdapter() {
  196. adapter.setNotifyChangeOfUnfilteredItems(true)
  197. .setMode(SelectableAdapter.Mode.MULTI);
  198. adapter.setEndlessScrollListener(this, new ProgressItem());
  199. adapter.registerAdapterDataObserver(new androidx.recyclerview.widget.RecyclerView.AdapterDataObserver() {
  200. @Override
  201. public void onChanged() {
  202. super.onChanged();
  203. adapter.filterItems();
  204. adapter.onLoadMoreComplete(null);
  205. }
  206. });
  207. adapter.setStickyHeaderElevation(5)
  208. .setUnlinkAllItemsOnRemoveHeaders(true)
  209. .setDisplayHeadersAtStartUp(true)
  210. .setStickyHeaders(true);
  211. adapter.addListener(this);
  212. }
  213. private void selectionDone() {
  214. if (!isPublicCall && adapter.getSelectedPositions().size() == 1) {
  215. String roomType = "1";
  216. if ("groups".equals(((UserItem) adapter.getItem(adapter.getSelectedPositions().get(0))).getModel().getSource())) {
  217. roomType = "2";
  218. }
  219. RetrofitBucket retrofitBucket = ApiUtils.getRetrofitBucketForCreateRoom(currentUser.getBaseUrl(), roomType,
  220. ((UserItem) adapter.getItem(adapter.getSelectedPositions().get(0))).getModel().getUserId(), null);
  221. ncApi.createRoom(credentials,
  222. retrofitBucket.getUrl(), retrofitBucket.getQueryMap())
  223. .subscribeOn(Schedulers.newThread())
  224. .observeOn(AndroidSchedulers.mainThread())
  225. .subscribe(new Observer<RoomOverall>() {
  226. @Override
  227. public void onSubscribe(Disposable d) {
  228. }
  229. @Override
  230. public void onNext(RoomOverall roomOverall) {
  231. Intent conversationIntent = new Intent(getActivity(), MagicCallActivity.class);
  232. Bundle bundle = new Bundle();
  233. bundle.putString(BundleKeys.KEY_ROOM_TOKEN, roomOverall.getOcs().getData().getToken());
  234. bundle.putString(BundleKeys.KEY_ROOM_ID, roomOverall.getOcs().getData().getRoomId());
  235. if (currentUser.hasSpreedCapabilityWithName("chat-v2")) {
  236. bundle.putString(BundleKeys.KEY_CONVERSATION_NAME,
  237. roomOverall.getOcs().getData().getDisplayName());
  238. conversationIntent.putExtras(bundle);
  239. getRouter().pushController((RouterTransaction.with(new ChatController(bundle))
  240. .pushChangeHandler(new HorizontalChangeHandler())
  241. .popChangeHandler(new HorizontalChangeHandler())));
  242. } else {
  243. conversationIntent.putExtras(bundle);
  244. startActivity(conversationIntent);
  245. new Handler().postDelayed(() -> getRouter().popCurrentController(), 100);
  246. }
  247. }
  248. @Override
  249. public void onError(Throwable e) {
  250. }
  251. @Override
  252. public void onComplete() {
  253. }
  254. });
  255. } else {
  256. Bundle bundle = new Bundle();
  257. Conversation.RoomType roomType;
  258. if (isPublicCall) {
  259. roomType = Conversation.RoomType.ROOM_PUBLIC_CALL;
  260. } else {
  261. roomType = Conversation.RoomType.ROOM_GROUP_CALL;
  262. }
  263. bundle.putParcelable(BundleKeys.KEY_CONVERSATION_TYPE, Parcels.wrap(roomType));
  264. ArrayList<String> userIds = new ArrayList<>();
  265. Set<Integer> selectedPositions = adapter.getSelectedPositionsAsSet();
  266. ArrayList<String> groupIds = new ArrayList<>();
  267. for (int selectedPosition : selectedPositions) {
  268. if (adapter.getItem(selectedPosition) instanceof UserItem) {
  269. UserItem userItem = (UserItem) adapter.getItem(selectedPosition);
  270. if (!"groups".equals(userItem.getModel().getSource())) {
  271. userIds.add(userItem.getModel().getUserId());
  272. } else {
  273. groupIds.add(userItem.getModel().getUserId());
  274. }
  275. }
  276. }
  277. bundle.putStringArrayList(BundleKeys.KEY_INVITED_PARTICIPANTS, userIds);
  278. bundle.putStringArrayList(BundleKeys.KEY_INVITED_GROUP, groupIds);
  279. bundle.putInt(BundleKeys.KEY_OPERATION_CODE, 11);
  280. prepareAndShowBottomSheetWithBundle(bundle);
  281. }
  282. }
  283. private void initSearchView() {
  284. if (getActivity() != null) {
  285. SearchManager searchManager = (SearchManager) getActivity().getSystemService(Context.SEARCH_SERVICE);
  286. if (searchItem != null) {
  287. searchView = (SearchView) MenuItemCompat.getActionView(searchItem);
  288. searchView.setMaxWidth(Integer.MAX_VALUE);
  289. searchView.setInputType(InputType.TYPE_TEXT_VARIATION_FILTER);
  290. searchView.setImeOptions(EditorInfo.IME_ACTION_DONE | EditorInfo.IME_FLAG_NO_FULLSCREEN);
  291. searchView.setQueryHint(getResources().getString(R.string.nc_search));
  292. if (searchManager != null) {
  293. searchView.setSearchableInfo(searchManager.getSearchableInfo(getActivity().getComponentName()));
  294. }
  295. searchView.setOnQueryTextListener(this);
  296. }
  297. }
  298. final View mSearchEditFrame = searchView
  299. .findViewById(androidx.appcompat.R.id.search_edit_frame);
  300. BottomNavigationView bottomNavigationView = null;
  301. if (getParentController() != null && getParentController().getView() != null) {
  302. bottomNavigationView = getParentController().getView().findViewById(R.id.navigation);
  303. }
  304. Handler handler = new Handler();
  305. ViewTreeObserver vto = mSearchEditFrame.getViewTreeObserver();
  306. BottomNavigationView finalBottomNavigationView = bottomNavigationView;
  307. vto.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
  308. int oldVisibility = -1;
  309. @Override
  310. public void onGlobalLayout() {
  311. int currentVisibility = mSearchEditFrame.getVisibility();
  312. if (currentVisibility != oldVisibility) {
  313. if (currentVisibility == View.VISIBLE) {
  314. if (finalBottomNavigationView != null) {
  315. handler.postDelayed(() -> finalBottomNavigationView.setVisibility(View.GONE), 100);
  316. }
  317. } else {
  318. handler.postDelayed(() -> {
  319. if (finalBottomNavigationView != null) {
  320. finalBottomNavigationView.setVisibility(View.VISIBLE);
  321. }
  322. searchItem.setVisible(contactItems.size() > 0);
  323. }, 500);
  324. }
  325. oldVisibility = currentVisibility;
  326. }
  327. }
  328. });
  329. }
  330. @Override
  331. public boolean onOptionsItemSelected(@NonNull MenuItem item) {
  332. switch (item.getItemId()) {
  333. case android.R.id.home:
  334. getRouter().popCurrentController();
  335. return true;
  336. case R.id.contacts_selection_done:
  337. selectionDone();
  338. return true;
  339. default:
  340. return super.onOptionsItemSelected(item);
  341. }
  342. }
  343. @Override
  344. public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
  345. super.onCreateOptionsMenu(menu, inflater);
  346. inflater.inflate(R.menu.menu_conversation_plus_filter, menu);
  347. searchItem = menu.findItem(R.id.action_search);
  348. doneMenuItem = menu.findItem(R.id.contacts_selection_done);
  349. menu.findItem(R.id.action_new_conversation).setVisible(false);
  350. initSearchView();
  351. }
  352. @Override
  353. public void onPrepareOptionsMenu(Menu menu) {
  354. super.onPrepareOptionsMenu(menu);
  355. searchItem.setVisible(contactItems.size() > 0);
  356. if (adapter.hasFilter()) {
  357. searchItem.expandActionView();
  358. searchView.setQuery((CharSequence) adapter.getFilter(String.class), false);
  359. }
  360. }
  361. private void fetchData(boolean startFromScratch) {
  362. dispose(null);
  363. alreadyFetching = true;
  364. Set<Sharee> shareeHashSet = new HashSet<>();
  365. Set<AutocompleteUser> autocompleteUsersHashSet = new HashSet<>();
  366. userHeaderItems = new HashMap<>();
  367. String query = "";
  368. if (searchView != null && !TextUtils.isEmpty(searchView.getQuery())) {
  369. query = searchView.getQuery().toString();
  370. } else if (startFromScratch) {
  371. contactItems = new ArrayList<>();
  372. }
  373. RetrofitBucket retrofitBucket;
  374. boolean serverIs14OrUp = false;
  375. if (currentUser.hasSpreedCapabilityWithName("last-room-activity")) {
  376. // a hack to see if we're on 14 or not
  377. retrofitBucket = ApiUtils.getRetrofitBucketForContactsSearchFor14(currentUser.getBaseUrl(), query);
  378. serverIs14OrUp = true;
  379. } else {
  380. retrofitBucket = ApiUtils.getRetrofitBucketForContactsSearch(currentUser.getBaseUrl(), query);
  381. }
  382. int page = 1;
  383. if (!startFromScratch) {
  384. if (TextUtils.isEmpty(query)) {
  385. page = currentPage + 1;
  386. } else {
  387. page = currentSearchPage + 1;
  388. }
  389. }
  390. Map<String, Object> modifiedQueryMap = new HashMap<>(retrofitBucket.getQueryMap());
  391. modifiedQueryMap.put("page", page);
  392. modifiedQueryMap.put("perPage", 100);
  393. List<String> shareTypesList = null;
  394. if (serverIs14OrUp) {
  395. shareTypesList = new ArrayList<>();
  396. // users
  397. shareTypesList.add("0");
  398. // groups
  399. shareTypesList.add("1");
  400. modifiedQueryMap.put("shareTypes[]", shareTypesList);
  401. }
  402. boolean finalServerIs14OrUp = serverIs14OrUp;
  403. ncApi.getContactsWithSearchParam(
  404. credentials,
  405. retrofitBucket.getUrl(), shareTypesList, modifiedQueryMap)
  406. .subscribeOn(Schedulers.newThread())
  407. .observeOn(AndroidSchedulers.mainThread())
  408. .retry(3)
  409. .subscribe(new Observer<ResponseBody>() {
  410. @Override
  411. public void onSubscribe(Disposable d) {
  412. contactsQueryDisposable = d;
  413. }
  414. @Override
  415. public void onNext(ResponseBody responseBody) {
  416. if (responseBody != null) {
  417. Participant participant;
  418. List<AbstractFlexibleItem> newUserItemList = new ArrayList<>();
  419. newUserItemList.addAll(contactItems);
  420. try {
  421. if (!finalServerIs14OrUp) {
  422. ShareesOverall shareesOverall = LoganSquare.parse(responseBody.string(), ShareesOverall.class);
  423. if (shareesOverall.getOcs().getData().getUsers() != null) {
  424. shareeHashSet.addAll(shareesOverall.getOcs().getData().getUsers());
  425. }
  426. if (shareesOverall.getOcs().getData().getExactUsers() != null &&
  427. shareesOverall.getOcs().getData().getExactUsers().getExactSharees() != null) {
  428. shareeHashSet.addAll(shareesOverall.getOcs().getData().
  429. getExactUsers().getExactSharees());
  430. }
  431. for (Sharee sharee : shareeHashSet) {
  432. if (!sharee.getValue().getShareWith().equals(currentUser.getUserId())) {
  433. participant = new Participant();
  434. participant.setName(sharee.getLabel());
  435. String headerTitle;
  436. headerTitle = sharee.getLabel().substring(0, 1).toUpperCase();
  437. UserHeaderItem userHeaderItem;
  438. if (!userHeaderItems.containsKey(headerTitle)) {
  439. userHeaderItem = new UserHeaderItem(headerTitle);
  440. userHeaderItems.put(headerTitle, userHeaderItem);
  441. }
  442. participant.setUserId(sharee.getValue().getShareWith());
  443. UserItem newContactItem = new UserItem(participant, currentUser,
  444. userHeaderItems.get(headerTitle));
  445. if (!contactItems.contains(newContactItem)) {
  446. newUserItemList.add(newContactItem);
  447. }
  448. }
  449. }
  450. } else {
  451. AutocompleteOverall autocompleteOverall = LoganSquare.parse(responseBody.string(), AutocompleteOverall.class);
  452. autocompleteUsersHashSet.addAll(autocompleteOverall.getOcs().getData());
  453. for (AutocompleteUser autocompleteUser : autocompleteUsersHashSet) {
  454. if (!autocompleteUser.getId().equals(currentUser.getUserId())) {
  455. participant = new Participant();
  456. participant.setName(autocompleteUser.getLabel());
  457. participant.setUserId(autocompleteUser.getId());
  458. participant.setSource(autocompleteUser.getSource());
  459. String headerTitle;
  460. headerTitle = participant.getName().substring(0, 1).toUpperCase();
  461. UserHeaderItem userHeaderItem;
  462. if (!userHeaderItems.containsKey(headerTitle)) {
  463. userHeaderItem = new UserHeaderItem(headerTitle);
  464. userHeaderItems.put(headerTitle, userHeaderItem);
  465. }
  466. participant.setUserId(participant.getUserId());
  467. UserItem newContactItem = new UserItem(participant, currentUser,
  468. userHeaderItems.get(headerTitle));
  469. if (!contactItems.contains(newContactItem)) {
  470. newUserItemList.add(newContactItem);
  471. }
  472. }
  473. }
  474. }
  475. } catch (Exception exception) {
  476. Log.e(TAG, "Parsing response body failed while getting contacts");
  477. }
  478. if (TextUtils.isEmpty((CharSequence) modifiedQueryMap.get("search"))) {
  479. canFetchFurther = !shareeHashSet.isEmpty() || (finalServerIs14OrUp && !autocompleteUsersHashSet.isEmpty());
  480. currentPage = (int) modifiedQueryMap.get("page");
  481. } else {
  482. canFetchSearchFurther = !shareeHashSet.isEmpty() || (finalServerIs14OrUp && !autocompleteUsersHashSet.isEmpty());
  483. currentSearchPage = (int) modifiedQueryMap.get("page");
  484. }
  485. boolean shouldFilterManually = false;
  486. if (newUserItemList.size() == contactItems.size()) {
  487. shouldFilterManually = true;
  488. }
  489. contactItems = newUserItemList;
  490. userHeaderItems = new HashMap<>();
  491. Collections.sort(newUserItemList, (o1, o2) -> {
  492. String firstName;
  493. String secondName;
  494. if (o1 instanceof UserItem) {
  495. firstName = ((UserItem) o1).getModel().getName();
  496. } else {
  497. firstName = ((UserHeaderItem) o1).getModel();
  498. }
  499. if (o2 instanceof UserItem) {
  500. secondName = ((UserItem) o2).getModel().getName();
  501. } else {
  502. secondName = ((UserHeaderItem) o2).getModel();
  503. }
  504. return firstName.compareToIgnoreCase(secondName);
  505. });
  506. if (!shouldFilterManually) {
  507. adapter.updateDataSet(newUserItemList, false);
  508. } else {
  509. adapter.filterItems();
  510. adapter.onLoadMoreComplete(null);
  511. }
  512. if (searchItem != null) {
  513. searchItem.setVisible(newUserItemList.size() > 0);
  514. }
  515. if (swipeRefreshLayout != null) {
  516. swipeRefreshLayout.setRefreshing(false);
  517. }
  518. if (isNewConversationView) {
  519. checkAndHandleDoneMenuItem();
  520. }
  521. }
  522. }
  523. @Override
  524. public void onError(Throwable e) {
  525. if (searchItem != null) {
  526. searchItem.setVisible(false);
  527. }
  528. if (e instanceof HttpException) {
  529. HttpException exception = (HttpException) e;
  530. switch (exception.code()) {
  531. case 401:
  532. if (getParentController() != null &&
  533. getParentController().getRouter() != null) {
  534. getParentController().getRouter().pushController((RouterTransaction.with
  535. (new WebViewLoginController(currentUser.getBaseUrl(),
  536. true))
  537. .pushChangeHandler(new VerticalChangeHandler())
  538. .popChangeHandler(new VerticalChangeHandler())));
  539. }
  540. break;
  541. default:
  542. break;
  543. }
  544. if (swipeRefreshLayout != null) {
  545. swipeRefreshLayout.setRefreshing(false);
  546. }
  547. }
  548. dispose(contactsQueryDisposable);
  549. }
  550. @Override
  551. public void onComplete() {
  552. if (swipeRefreshLayout != null) {
  553. swipeRefreshLayout.setRefreshing(false);
  554. }
  555. dispose(contactsQueryDisposable);
  556. alreadyFetching = false;
  557. }
  558. });
  559. }
  560. private void prepareViews() {
  561. layoutManager = new SmoothScrollLinearLayoutManager(getActivity());
  562. recyclerView.setLayoutManager(new SmoothScrollLinearLayoutManager(getActivity()));
  563. recyclerView.setHasFixedSize(true);
  564. recyclerView.setAdapter(adapter);
  565. swipeRefreshLayout.setOnRefreshListener(() -> fetchData(true));
  566. swipeRefreshLayout.setColorSchemeResources(R.color.colorPrimary);
  567. fastScroller.addOnScrollStateChangeListener(this);
  568. adapter.setFastScroller(fastScroller);
  569. fastScroller.setBubbleTextCreator(position -> {
  570. IFlexible abstractFlexibleItem = adapter.getItem(position);
  571. if (abstractFlexibleItem instanceof UserItem) {
  572. return ((UserItem) adapter.getItem(position)).getHeader().getModel();
  573. } else if (abstractFlexibleItem instanceof UserHeaderItem) {
  574. return ((UserHeaderItem) adapter.getItem(position)).getModel();
  575. } else {
  576. return "";
  577. }
  578. });
  579. }
  580. private void dispose(@Nullable Disposable disposable) {
  581. if (disposable != null && !disposable.isDisposed()) {
  582. disposable.dispose();
  583. } else if (disposable == null) {
  584. if (contactsQueryDisposable != null && !contactsQueryDisposable.isDisposed()) {
  585. contactsQueryDisposable.dispose();
  586. contactsQueryDisposable = null;
  587. }
  588. if (cacheQueryDisposable != null && !cacheQueryDisposable.isDisposed()) {
  589. cacheQueryDisposable.dispose();
  590. cacheQueryDisposable = null;
  591. }
  592. }
  593. }
  594. @Override
  595. public void onSaveViewState(@NonNull View view, @NonNull Bundle outState) {
  596. adapter.onSaveInstanceState(outState);
  597. super.onSaveViewState(view, outState);
  598. if (searchView != null && !TextUtils.isEmpty(searchView.getQuery())) {
  599. outState.putString(KEY_SEARCH_QUERY, searchView.getQuery().toString());
  600. }
  601. }
  602. @Override
  603. public void onRestoreViewState(@NonNull View view, @NonNull Bundle savedViewState) {
  604. super.onRestoreViewState(view, savedViewState);
  605. searchQuery = savedViewState.getString(KEY_SEARCH_QUERY, "");
  606. if (adapter != null) {
  607. adapter.onRestoreInstanceState(savedViewState);
  608. }
  609. }
  610. @Override
  611. public void onDestroy() {
  612. super.onDestroy();
  613. dispose(null);
  614. }
  615. @Override
  616. public boolean onQueryTextChange(String newText) {
  617. if (adapter.hasNewFilter(newText) || !TextUtils.isEmpty(searchQuery)) {
  618. if (!TextUtils.isEmpty(searchQuery)) {
  619. adapter.setFilter(searchQuery);
  620. adapter.filterItems();
  621. searchQuery = "";
  622. } else {
  623. adapter.setFilter(newText);
  624. if (TextUtils.isEmpty(newText)) {
  625. adapter.filterItems();
  626. } else {
  627. fetchData(true);
  628. }
  629. }
  630. }
  631. if (swipeRefreshLayout != null) {
  632. swipeRefreshLayout.setEnabled(!adapter.hasFilter());
  633. }
  634. return true;
  635. }
  636. @Override
  637. public boolean onQueryTextSubmit(String query) {
  638. return onQueryTextChange(query);
  639. }
  640. private void checkAndHandleDoneMenuItem() {
  641. if (adapter != null && doneMenuItem != null) {
  642. if (adapter.getSelectedItemCount() > 0 || isPublicCall) {
  643. if (!doneMenuItem.isVisible()) {
  644. doneMenuItem.setVisible(true);
  645. }
  646. } else {
  647. doneMenuItem.setVisible(false);
  648. }
  649. } else if (doneMenuItem != null) {
  650. doneMenuItem.setVisible(false);
  651. }
  652. }
  653. @Override
  654. protected String getTitle() {
  655. if (!isNewConversationView) {
  656. return getResources().getString(R.string.nc_app_name);
  657. } else {
  658. return getResources().getString(R.string.nc_select_contacts);
  659. }
  660. }
  661. @Override
  662. public void onFastScrollerStateChange(boolean scrolling) {
  663. swipeRefreshLayout.setEnabled(!scrolling);
  664. }
  665. private void prepareAndShowBottomSheetWithBundle(Bundle bundle) {
  666. if (view == null) {
  667. view = getActivity().getLayoutInflater().inflate(R.layout.bottom_sheet, null, false);
  668. }
  669. getChildRouter((ViewGroup) view).setRoot(
  670. RouterTransaction.with(new OperationsMenuController(bundle))
  671. .popChangeHandler(new VerticalChangeHandler())
  672. .pushChangeHandler(new VerticalChangeHandler()));
  673. if (bottomSheet == null) {
  674. bottomSheet = new BottomSheet.Builder(getActivity()).setView(view).create();
  675. }
  676. bottomSheet.setOnCancelListener(dialog -> {
  677. if (getActionBar() != null) {
  678. getActionBar().setDisplayHomeAsUpEnabled(true);
  679. }
  680. });
  681. bottomSheet.setOnShowListener(dialog -> eventBus.post(new BottomSheetLockEvent(false, 0,
  682. false, false)));
  683. bottomSheet.show();
  684. }
  685. @Subscribe(threadMode = ThreadMode.MAIN)
  686. public void onMessageEvent(BottomSheetLockEvent bottomSheetLockEvent) {
  687. if (bottomSheet != null) {
  688. if (!bottomSheetLockEvent.isCancelable()) {
  689. bottomSheet.setCancelable(bottomSheetLockEvent.isCancelable());
  690. } else {
  691. bottomSheet.setCancelable(bottomSheetLockEvent.isCancelable());
  692. if (bottomSheet.isShowing() && bottomSheetLockEvent.isCancel()) {
  693. new Handler().postDelayed(() -> {
  694. bottomSheet.setOnCancelListener(null);
  695. bottomSheet.cancel();
  696. }, bottomSheetLockEvent.getDelay());
  697. }
  698. }
  699. }
  700. }
  701. @Override
  702. protected void onDetach(@NonNull View view) {
  703. super.onDetach(view);
  704. eventBus.unregister(this);
  705. }
  706. @Override
  707. public boolean onItemClick(View view, int position) {
  708. if (adapter.getItem(position) instanceof UserItem) {
  709. if (!isNewConversationView) {
  710. UserItem userItem = (UserItem) adapter.getItem(position);
  711. String roomType = "1";
  712. if ("groups".equals(userItem.getModel().getSource())) {
  713. roomType = "2";
  714. }
  715. RetrofitBucket retrofitBucket = ApiUtils.getRetrofitBucketForCreateRoom(currentUser.getBaseUrl(), roomType, userItem.getModel().getUserId(), null);
  716. ncApi.createRoom(credentials,
  717. retrofitBucket.getUrl(), retrofitBucket.getQueryMap())
  718. .subscribeOn(Schedulers.newThread())
  719. .observeOn(AndroidSchedulers.mainThread())
  720. .subscribe(new Observer<RoomOverall>() {
  721. @Override
  722. public void onSubscribe(Disposable d) {
  723. }
  724. @Override
  725. public void onNext(RoomOverall roomOverall) {
  726. if (getActivity() != null) {
  727. Intent conversationIntent = new Intent(getActivity(), MagicCallActivity.class);
  728. Bundle bundle = new Bundle();
  729. bundle.putString(BundleKeys.KEY_ROOM_TOKEN, roomOverall.getOcs().getData().getToken());
  730. bundle.putString(BundleKeys.KEY_ROOM_ID, roomOverall.getOcs().getData().getRoomId());
  731. conversationIntent.putExtras(bundle);
  732. if (currentUser.hasSpreedCapabilityWithName("chat-v2")) {
  733. bundle.putString(BundleKeys.KEY_CONVERSATION_NAME,
  734. roomOverall.getOcs().getData().getDisplayName());
  735. if (getParentController() != null) {
  736. getParentController().getRouter().pushController((RouterTransaction.with(new ChatController(bundle))
  737. .pushChangeHandler(new HorizontalChangeHandler())
  738. .popChangeHandler(new HorizontalChangeHandler())));
  739. }
  740. } else {
  741. startActivity(conversationIntent);
  742. }
  743. }
  744. }
  745. @Override
  746. public void onError(Throwable e) {
  747. }
  748. @Override
  749. public void onComplete() {
  750. }
  751. });
  752. } else {
  753. ((UserItem) adapter.getItem(position)).flipItemSelection();
  754. adapter.toggleSelection(position);
  755. if (currentUser.hasSpreedCapabilityWithName("last-room-activity") &&
  756. "groups".equals(((UserItem) adapter.getItem(position)).getModel().getSource()) &&
  757. adapter.getSelectedItemCount() > 1) {
  758. List<Integer> selectedPositions = adapter.getSelectedPositions();
  759. for (int i = 0; i < selectedPositions.size(); i++) {
  760. if (!selectedPositions.get(i).equals(position) && "groups".equals(((UserItem) adapter.getItem(selectedPositions.get(i))).getModel().getSource())) {
  761. ((UserItem) adapter.getItem(selectedPositions.get(i))).flipItemSelection();
  762. adapter.toggleSelection(selectedPositions.get(i));
  763. }
  764. }
  765. }
  766. checkAndHandleDoneMenuItem();
  767. }
  768. }
  769. return true;
  770. }
  771. @Optional
  772. @OnClick(R.id.call_header_layout)
  773. void toggleCallHeader() {
  774. toggleNewCallHeaderVisibility(isPublicCall);
  775. isPublicCall = !isPublicCall;
  776. checkAndHandleDoneMenuItem();
  777. }
  778. private void toggleNewCallHeaderVisibility(boolean showInitialLayout) {
  779. if (showInitialLayout) {
  780. initialRelativeLayout.setVisibility(View.VISIBLE);
  781. secondaryRelativeLayout.setVisibility(View.GONE);
  782. } else {
  783. initialRelativeLayout.setVisibility(View.GONE);
  784. secondaryRelativeLayout.setVisibility(View.VISIBLE);
  785. }
  786. }
  787. @Override
  788. public void noMoreLoad(int newItemsSize) {
  789. }
  790. @Override
  791. public void onLoadMore(int lastPosition, int currentPage) {
  792. String query = "";
  793. if (searchView != null && !TextUtils.isEmpty(searchView.getQuery())) {
  794. query = searchView.getQuery().toString();
  795. }
  796. if (!alreadyFetching && ((searchView != null && searchView.isIconified() && canFetchFurther)
  797. || (!TextUtils.isEmpty(query) && canFetchSearchFurther))) {
  798. fetchData(false);
  799. } else {
  800. adapter.onLoadMoreComplete(null);
  801. }
  802. }
  803. }