ContactsController.java 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  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.support.annotation.NonNull;
  27. import android.support.annotation.Nullable;
  28. import android.support.design.widget.BottomNavigationView;
  29. import android.support.v4.view.MenuItemCompat;
  30. import android.support.v4.widget.SwipeRefreshLayout;
  31. import android.support.v7.widget.DividerItemDecoration;
  32. import android.support.v7.widget.LinearLayoutManager;
  33. import android.support.v7.widget.RecyclerView;
  34. import android.support.v7.widget.SearchView;
  35. import android.text.InputType;
  36. import android.text.TextUtils;
  37. import android.view.ActionMode;
  38. import android.view.LayoutInflater;
  39. import android.view.Menu;
  40. import android.view.MenuInflater;
  41. import android.view.MenuItem;
  42. import android.view.View;
  43. import android.view.ViewGroup;
  44. import android.view.ViewTreeObserver;
  45. import android.view.inputmethod.EditorInfo;
  46. import com.bluelinelabs.conductor.RouterTransaction;
  47. import com.bluelinelabs.conductor.changehandler.HorizontalChangeHandler;
  48. import com.bluelinelabs.conductor.changehandler.VerticalChangeHandler;
  49. import com.bluelinelabs.conductor.internal.NoOpControllerChangeHandler;
  50. import com.nextcloud.talk.R;
  51. import com.nextcloud.talk.activities.CallActivity;
  52. import com.nextcloud.talk.adapters.items.UserItem;
  53. import com.nextcloud.talk.api.NcApi;
  54. import com.nextcloud.talk.utils.ApiUtils;
  55. import com.nextcloud.talk.models.json.participants.Participant;
  56. import com.nextcloud.talk.models.json.rooms.RoomOverall;
  57. import com.nextcloud.talk.models.json.sharees.Sharee;
  58. import com.nextcloud.talk.application.NextcloudTalkApplication;
  59. import com.nextcloud.talk.controllers.base.BaseController;
  60. import com.nextcloud.talk.models.RetrofitBucket;
  61. import com.nextcloud.talk.persistence.entities.UserEntity;
  62. import com.nextcloud.talk.utils.database.user.UserUtils;
  63. import org.parceler.Parcels;
  64. import java.util.ArrayList;
  65. import java.util.Collections;
  66. import java.util.HashSet;
  67. import java.util.List;
  68. import java.util.Set;
  69. import javax.inject.Inject;
  70. import autodagger.AutoInjector;
  71. import butterknife.BindView;
  72. import eu.davidea.flexibleadapter.FlexibleAdapter;
  73. import eu.davidea.flexibleadapter.SelectableAdapter;
  74. import eu.davidea.flexibleadapter.common.SmoothScrollLinearLayoutManager;
  75. import io.reactivex.Observer;
  76. import io.reactivex.android.schedulers.AndroidSchedulers;
  77. import io.reactivex.disposables.Disposable;
  78. import io.reactivex.schedulers.Schedulers;
  79. import retrofit2.HttpException;
  80. @AutoInjector(NextcloudTalkApplication.class)
  81. public class ContactsController extends BaseController implements SearchView.OnQueryTextListener,
  82. ActionMode.Callback, FlexibleAdapter.OnItemClickListener {
  83. public static final String TAG = "ContactsController";
  84. private static final String KEY_SEARCH_QUERY = "ContactsController.searchQuery";
  85. @Inject
  86. UserUtils userUtils;
  87. @Inject
  88. NcApi ncApi;
  89. @BindView(R.id.recycler_view)
  90. RecyclerView recyclerView;
  91. @BindView(R.id.swipe_refresh_layout)
  92. SwipeRefreshLayout swipeRefreshLayout;
  93. private UserEntity userEntity;
  94. private Disposable contactsQueryDisposable;
  95. private Disposable cacheQueryDisposable;
  96. private FlexibleAdapter<UserItem> adapter;
  97. private List<UserItem> contactItems = new ArrayList<>();
  98. private MenuItem searchItem;
  99. private SearchView searchView;
  100. private String searchQuery;
  101. private ActionMode actionMode;
  102. public ContactsController() {
  103. super();
  104. setHasOptionsMenu(true);
  105. }
  106. @Override
  107. protected View inflateView(@NonNull LayoutInflater inflater, @NonNull ViewGroup container) {
  108. return inflater.inflate(R.layout.controller_generic_rv, container, false);
  109. }
  110. @Override
  111. protected void onViewBound(@NonNull View view) {
  112. super.onViewBound(view);
  113. NextcloudTalkApplication.getSharedApplication().getComponentApplication().inject(this);
  114. userEntity = userUtils.getCurrentUser();
  115. if (userEntity == null) {
  116. if (getParentController().getRouter() != null) {
  117. getParentController().getRouter().setRoot((RouterTransaction.with(new ServerSelectionController())
  118. .pushChangeHandler(new HorizontalChangeHandler())
  119. .popChangeHandler(new HorizontalChangeHandler())));
  120. }
  121. }
  122. if (adapter == null) {
  123. adapter = new FlexibleAdapter<>(contactItems, getActivity(), false);
  124. if (userEntity != null) {
  125. fetchData();
  126. }
  127. }
  128. adapter.addListener(this);
  129. prepareViews();
  130. }
  131. private void initSearchView() {
  132. if (getActivity() != null) {
  133. SearchManager searchManager = (SearchManager) getActivity().getSystemService(Context.SEARCH_SERVICE);
  134. if (searchItem != null) {
  135. searchView = (SearchView) MenuItemCompat.getActionView(searchItem);
  136. searchView.setMaxWidth(Integer.MAX_VALUE);
  137. searchView.setInputType(InputType.TYPE_TEXT_VARIATION_FILTER);
  138. searchView.setImeOptions(EditorInfo.IME_ACTION_DONE | EditorInfo.IME_FLAG_NO_FULLSCREEN);
  139. searchView.setQueryHint(getResources().getString(R.string.nc_search));
  140. if (searchManager != null) {
  141. searchView.setSearchableInfo(searchManager.getSearchableInfo(getActivity().getComponentName()));
  142. }
  143. searchView.setOnQueryTextListener(this);
  144. }
  145. }
  146. final View mSearchEditFrame = searchView
  147. .findViewById(android.support.v7.appcompat.R.id.search_edit_frame);
  148. BottomNavigationView bottomNavigationView = getParentController().getView().findViewById(R.id.navigation);
  149. Handler handler = new Handler();
  150. ViewTreeObserver vto = mSearchEditFrame.getViewTreeObserver();
  151. vto.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
  152. int oldVisibility = -1;
  153. @Override
  154. public void onGlobalLayout() {
  155. int currentVisibility = mSearchEditFrame.getVisibility();
  156. if (currentVisibility != oldVisibility) {
  157. if (currentVisibility == View.VISIBLE) {
  158. handler.postDelayed(() -> bottomNavigationView.setVisibility(View.GONE), 100);
  159. } else {
  160. handler.postDelayed(() -> {
  161. bottomNavigationView.setVisibility(View.VISIBLE);
  162. searchItem.setVisible(contactItems.size() > 0);
  163. }, 500);
  164. }
  165. oldVisibility = currentVisibility;
  166. }
  167. }
  168. });
  169. }
  170. @Override
  171. public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
  172. super.onCreateOptionsMenu(menu, inflater);
  173. inflater.inflate(R.menu.menu_filter, menu);
  174. searchItem = menu.findItem(R.id.action_search);
  175. initSearchView();
  176. }
  177. @Override
  178. public void onPrepareOptionsMenu(Menu menu) {
  179. super.onPrepareOptionsMenu(menu);
  180. searchItem.setVisible(contactItems.size() > 0);
  181. if (adapter.hasSearchText()) {
  182. searchItem.expandActionView();
  183. searchView.setQuery(adapter.getSearchText(), false);
  184. }
  185. }
  186. private void fetchData() {
  187. dispose(null);
  188. Set<Sharee> shareeHashSet = new HashSet<>();
  189. contactItems = new ArrayList<>();
  190. RetrofitBucket retrofitBucket = ApiUtils.getRetrofitBucketForContactsSearch(userEntity.getBaseUrl(),
  191. "");
  192. contactsQueryDisposable = ncApi.getContactsWithSearchParam(
  193. ApiUtils.getCredentials(userEntity.getUsername(), userEntity.getToken()),
  194. retrofitBucket.getUrl(), retrofitBucket.getQueryMap())
  195. .subscribeOn(Schedulers.newThread())
  196. .observeOn(AndroidSchedulers.mainThread())
  197. .subscribe(shareesOverall -> {
  198. if (shareesOverall != null) {
  199. if (shareesOverall.getOcs().getData().getUsers() != null) {
  200. shareeHashSet.addAll(shareesOverall.getOcs().getData().getUsers());
  201. }
  202. if (shareesOverall.getOcs().getData().getExactUsers() != null &&
  203. shareesOverall.getOcs().getData().getExactUsers().getExactSharees() != null) {
  204. shareeHashSet.addAll(shareesOverall.getOcs().getData().
  205. getExactUsers().getExactSharees());
  206. }
  207. Participant participant;
  208. for (Sharee sharee : shareeHashSet) {
  209. if (!sharee.getValue().getShareWith().equals(userEntity.getUsername())) {
  210. participant = new Participant();
  211. participant.setName(sharee.getLabel());
  212. participant.setUserId(sharee.getValue().getShareWith());
  213. contactItems.add(new UserItem(participant, userEntity));
  214. }
  215. }
  216. Collections.sort(contactItems, (userItem, t1) ->
  217. userItem.getModel().getName().compareToIgnoreCase(t1.getModel().getName()));
  218. adapter.updateDataSet(contactItems, true);
  219. searchItem.setVisible(contactItems.size() > 0);
  220. swipeRefreshLayout.setRefreshing(false);
  221. }
  222. }, throwable -> {
  223. if (searchItem != null) {
  224. searchItem.setVisible(false);
  225. }
  226. if (throwable instanceof HttpException) {
  227. HttpException exception = (HttpException) throwable;
  228. switch (exception.code()) {
  229. case 401:
  230. if (getParentController() != null &&
  231. getParentController().getRouter() != null) {
  232. getParentController().getRouter().pushController((RouterTransaction.with
  233. (new WebViewLoginController(userEntity.getBaseUrl(),
  234. true))
  235. .pushChangeHandler(new VerticalChangeHandler())
  236. .popChangeHandler(new VerticalChangeHandler())));
  237. }
  238. break;
  239. default:
  240. break;
  241. }
  242. }
  243. swipeRefreshLayout.setRefreshing(false);
  244. dispose(contactsQueryDisposable);
  245. }
  246. , () -> {
  247. swipeRefreshLayout.setRefreshing(false);
  248. dispose(contactsQueryDisposable);
  249. });
  250. }
  251. private void prepareViews() {
  252. LinearLayoutManager layoutManager = new SmoothScrollLinearLayoutManager(getActivity());
  253. recyclerView.setLayoutManager(layoutManager);
  254. recyclerView.setHasFixedSize(true);
  255. recyclerView.setAdapter(adapter);
  256. recyclerView.addItemDecoration(new DividerItemDecoration(
  257. recyclerView.getContext(),
  258. layoutManager.getOrientation()
  259. ));
  260. swipeRefreshLayout.setOnRefreshListener(this::fetchData);
  261. swipeRefreshLayout.setColorSchemeResources(R.color.colorPrimary);
  262. }
  263. private void dispose(@Nullable Disposable disposable) {
  264. if (disposable != null && !disposable.isDisposed()) {
  265. disposable.dispose();
  266. } else if (disposable == null) {
  267. if (contactsQueryDisposable != null && !contactsQueryDisposable.isDisposed()) {
  268. contactsQueryDisposable.dispose();
  269. contactsQueryDisposable = null;
  270. }
  271. if (cacheQueryDisposable != null && !cacheQueryDisposable.isDisposed()) {
  272. cacheQueryDisposable.dispose();
  273. cacheQueryDisposable = null;
  274. }
  275. }
  276. }
  277. @Override
  278. public void onSaveViewState(@NonNull View view, @NonNull Bundle outState) {
  279. super.onSaveViewState(view, outState);
  280. if (searchView != null && !TextUtils.isEmpty(searchView.getQuery())) {
  281. outState.putString(KEY_SEARCH_QUERY, searchView.getQuery().toString());
  282. }
  283. }
  284. @Override
  285. public void onRestoreViewState(@NonNull View view, @NonNull Bundle savedViewState) {
  286. super.onRestoreViewState(view, savedViewState);
  287. searchQuery = savedViewState.getString(KEY_SEARCH_QUERY, "");
  288. }
  289. @Override
  290. public void onDestroy() {
  291. super.onDestroy();
  292. dispose(null);
  293. }
  294. @Override
  295. public boolean onQueryTextChange(String newText) {
  296. if (adapter.hasNewSearchText(newText) || !TextUtils.isEmpty(searchQuery)) {
  297. if (!TextUtils.isEmpty(searchQuery)) {
  298. adapter.setSearchText(searchQuery);
  299. searchQuery = "";
  300. adapter.filterItems();
  301. } else {
  302. adapter.setSearchText(newText);
  303. adapter.filterItems(300);
  304. }
  305. }
  306. if (swipeRefreshLayout != null) {
  307. swipeRefreshLayout.setEnabled(!adapter.hasSearchText());
  308. }
  309. return true;
  310. }
  311. @Override
  312. public boolean onQueryTextSubmit(String query) {
  313. return onQueryTextChange(query);
  314. }
  315. @Override
  316. public boolean onCreateActionMode(ActionMode actionMode, Menu menu) {
  317. adapter.setMode(SelectableAdapter.Mode.MULTI);
  318. return true;
  319. }
  320. @Override
  321. public boolean onPrepareActionMode(ActionMode actionMode, Menu menu) {
  322. return false;
  323. }
  324. @Override
  325. public boolean onActionItemClicked(ActionMode actionMode, MenuItem menuItem) {
  326. return false;
  327. }
  328. @Override
  329. public void onDestroyActionMode(ActionMode actionMode) {
  330. adapter.setMode(SelectableAdapter.Mode.IDLE);
  331. actionMode = null;
  332. }
  333. /*@Override
  334. public boolean onItemClick(int position) {
  335. if (actionMode != null && position != RecyclerView.NO_POSITION) {
  336. // Mark the position selected
  337. toggleSelection(position);
  338. return true;
  339. } else {
  340. // Handle the item click listener
  341. // We don't need to activate anything
  342. return false;
  343. }
  344. }*/
  345. private void toggleSelection(int position) {
  346. adapter.toggleSelection(position);
  347. int count = adapter.getSelectedItemCount();
  348. if (count == 0) {
  349. actionMode.finish();
  350. } else {
  351. //setContextTitle(count);
  352. }
  353. }
  354. @Override
  355. public void onSaveInstanceState(@NonNull Bundle outState) {
  356. adapter.onSaveInstanceState(outState);
  357. super.onSaveInstanceState(outState);
  358. }
  359. @Override
  360. protected void onRestoreInstanceState(@NonNull Bundle savedInstanceState) {
  361. super.onRestoreInstanceState(savedInstanceState);
  362. if (adapter != null) {
  363. adapter.onRestoreInstanceState(savedInstanceState);
  364. }
  365. }
  366. @Override
  367. public boolean onItemClick(int position) {
  368. UserItem userItem = adapter.getItem(position);
  369. RetrofitBucket retrofitBucket = ApiUtils.getRetrofitBucketForCreateRoom(userEntity.getBaseUrl(), "1",
  370. userItem.getModel().getUserId());
  371. ncApi.createRoom(ApiUtils.getCredentials(userEntity.getUsername(), userEntity.getToken()),
  372. retrofitBucket.getUrl(), retrofitBucket.getQueryMap())
  373. .subscribeOn(Schedulers.newThread())
  374. .observeOn(AndroidSchedulers.mainThread())
  375. .subscribe(new Observer<RoomOverall>() {
  376. @Override
  377. public void onSubscribe(Disposable d) {
  378. }
  379. @Override
  380. public void onNext(RoomOverall roomOverall) {
  381. overridePushHandler(new NoOpControllerChangeHandler());
  382. overridePopHandler(new NoOpControllerChangeHandler());
  383. Intent callIntent = new Intent(getActivity(), CallActivity.class);
  384. Bundle bundle = new Bundle();
  385. bundle.putString(BundleKeys.KEY_ROOM_TOKEN, roomOverall.getOcs().getData().getToken());
  386. bundle.putParcelable(BundleKeys.KEY_USER_ENTITY, Parcels.wrap(userEntity));
  387. callIntent.putExtras(bundle);
  388. startActivity(callIntent);
  389. }
  390. @Override
  391. public void onError(Throwable e) {
  392. }
  393. @Override
  394. public void onComplete() {
  395. }
  396. });
  397. return true;
  398. }
  399. @Override
  400. protected String getTitle() {
  401. return getResources().getString(R.string.nc_app_name);
  402. }
  403. }