ConversationInfoController.java 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  1. /*
  2. * Nextcloud Talk application
  3. *
  4. * @author Mario Danic
  5. * Copyright (C) 2017-2018 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.os.Bundle;
  22. import android.text.TextUtils;
  23. import android.view.LayoutInflater;
  24. import android.view.MenuItem;
  25. import android.view.View;
  26. import android.view.ViewGroup;
  27. import android.widget.ImageView;
  28. import android.widget.ProgressBar;
  29. import android.widget.TextView;
  30. import com.bumptech.glide.load.engine.DiskCacheStrategy;
  31. import com.bumptech.glide.load.model.GlideUrl;
  32. import com.bumptech.glide.load.model.LazyHeaders;
  33. import com.bumptech.glide.load.resource.bitmap.CircleCrop;
  34. import com.bumptech.glide.request.RequestOptions;
  35. import com.nextcloud.talk.R;
  36. import com.nextcloud.talk.adapters.items.UserItem;
  37. import com.nextcloud.talk.api.NcApi;
  38. import com.nextcloud.talk.application.NextcloudTalkApplication;
  39. import com.nextcloud.talk.controllers.base.BaseController;
  40. import com.nextcloud.talk.models.database.UserEntity;
  41. import com.nextcloud.talk.models.json.converters.EnumNotificationLevelConverter;
  42. import com.nextcloud.talk.models.json.converters.EnumParticipantTypeConverter;
  43. import com.nextcloud.talk.models.json.participants.Participant;
  44. import com.nextcloud.talk.models.json.participants.ParticipantsOverall;
  45. import com.nextcloud.talk.models.json.rooms.Conversation;
  46. import com.nextcloud.talk.models.json.rooms.RoomOverall;
  47. import com.nextcloud.talk.utils.ApiUtils;
  48. import com.nextcloud.talk.utils.bundle.BundleKeys;
  49. import com.nextcloud.talk.utils.glide.GlideApp;
  50. import com.nextcloud.talk.utils.preferencestorage.DatabaseStorageModule;
  51. import com.yarolegovich.mp.MaterialChoicePreference;
  52. import com.yarolegovich.mp.MaterialPreferenceCategory;
  53. import com.yarolegovich.mp.MaterialPreferenceScreen;
  54. import org.parceler.Parcels;
  55. import java.util.ArrayList;
  56. import java.util.Collections;
  57. import java.util.Comparator;
  58. import java.util.HashMap;
  59. import java.util.List;
  60. import javax.inject.Inject;
  61. import androidx.annotation.NonNull;
  62. import androidx.recyclerview.widget.RecyclerView;
  63. import autodagger.AutoInjector;
  64. import butterknife.BindView;
  65. import eu.davidea.flexibleadapter.FlexibleAdapter;
  66. import eu.davidea.flexibleadapter.common.SmoothScrollLinearLayoutManager;
  67. import eu.davidea.flexibleadapter.items.AbstractFlexibleItem;
  68. import io.reactivex.Observer;
  69. import io.reactivex.android.schedulers.AndroidSchedulers;
  70. import io.reactivex.disposables.Disposable;
  71. import io.reactivex.schedulers.Schedulers;
  72. @AutoInjector(NextcloudTalkApplication.class)
  73. public class ConversationInfoController extends BaseController {
  74. private String baseUrl;
  75. private String conversationToken;
  76. private UserEntity conversationUser;
  77. private String credentials;
  78. @BindView(R.id.notification_settings)
  79. MaterialPreferenceScreen materialPreferenceScreen;
  80. @BindView(R.id.progressBar)
  81. ProgressBar progressBar;
  82. @BindView(R.id.conversation_info_message_notifications)
  83. MaterialChoicePreference messageNotificationLevel;
  84. @BindView(R.id.conversation_info_name)
  85. MaterialPreferenceCategory nameCategoryView;
  86. @BindView(R.id.avatar_image)
  87. ImageView conversationAvatarImageView;
  88. @BindView(R.id.display_name_text)
  89. TextView conversationDisplayName;
  90. @BindView(R.id.participants_list_category)
  91. MaterialPreferenceCategory participantsListCategory;
  92. @BindView(R.id.recycler_view)
  93. RecyclerView recyclerView;
  94. @Inject
  95. NcApi ncApi;
  96. private Disposable roomDisposable;
  97. private Disposable participantsDisposable;
  98. private Conversation conversation;
  99. private FlexibleAdapter<AbstractFlexibleItem> adapter;
  100. private List<AbstractFlexibleItem> recyclerViewItems = new ArrayList<>();
  101. public ConversationInfoController(Bundle args) {
  102. super(args);
  103. setHasOptionsMenu(true);
  104. NextcloudTalkApplication.getSharedApplication().getComponentApplication().inject(this);
  105. conversationUser = Parcels.unwrap(args.getParcelable(BundleKeys.KEY_USER_ENTITY));
  106. conversationToken = args.getString(BundleKeys.KEY_ROOM_TOKEN);
  107. baseUrl = args.getString(BundleKeys.KEY_BASE_URL);
  108. credentials = ApiUtils.getCredentials(conversationUser.getUsername(), conversationUser.getToken());
  109. }
  110. @Override
  111. public boolean onOptionsItemSelected(@NonNull MenuItem item) {
  112. switch (item.getItemId()) {
  113. case android.R.id.home:
  114. getRouter().popCurrentController();
  115. return true;
  116. default:
  117. return super.onOptionsItemSelected(item);
  118. }
  119. }
  120. @Override
  121. protected View inflateView(@NonNull LayoutInflater inflater, @NonNull ViewGroup container) {
  122. return inflater.inflate(R.layout.controller_conversation_info, container, false);
  123. }
  124. @Override
  125. protected void onViewBound(@NonNull View view) {
  126. super.onViewBound(view);
  127. materialPreferenceScreen.setStorageModule(new DatabaseStorageModule(conversationUser, conversationToken));
  128. if (adapter == null) {
  129. fetchRoomInfo();
  130. } else {
  131. loadConversationAvatar();
  132. materialPreferenceScreen.setVisibility(View.VISIBLE);
  133. nameCategoryView.setVisibility(View.VISIBLE);
  134. participantsListCategory.setVisibility(View.VISIBLE);
  135. progressBar.setVisibility(View.GONE);
  136. conversationDisplayName.setText(conversation.getDisplayName());
  137. setupAdapter();
  138. }
  139. }
  140. private void setupAdapter() {
  141. if (adapter == null) {
  142. adapter = new FlexibleAdapter<>(recyclerViewItems, getActivity(), true);
  143. }
  144. SmoothScrollLinearLayoutManager layoutManager =
  145. new SmoothScrollLinearLayoutManager(getActivity());
  146. recyclerView.setLayoutManager(layoutManager);
  147. recyclerView.setHasFixedSize(true);
  148. recyclerView.setAdapter(adapter);
  149. }
  150. private void handleParticipants(List<Participant> participants) {
  151. UserItem userItem;
  152. Participant participant;
  153. recyclerViewItems = new ArrayList<>();
  154. UserItem ownUserItem = null;
  155. for (int i = 0; i < participants.size(); i++) {
  156. participant = participants.get(i);
  157. userItem = new UserItem(participant, conversationUser, null);
  158. userItem.setEnabled(!participant.getSessionId().equals("0"));
  159. if (!TextUtils.isEmpty(participant.getUserId()) && !participant.getUserId().equals(conversationUser.getUserId())) {
  160. ownUserItem = userItem;
  161. } else {
  162. recyclerViewItems.add(userItem);
  163. }
  164. }
  165. if (ownUserItem != null) {
  166. recyclerViewItems.add(ownUserItem);
  167. }
  168. setupAdapter();
  169. participantsListCategory.setVisibility(View.VISIBLE);
  170. adapter.notifyDataSetChanged();
  171. }
  172. @Override
  173. protected void onAttach(@NonNull View view) {
  174. super.onAttach(view);
  175. if (getActionBar() != null) {
  176. getActionBar().setDisplayHomeAsUpEnabled(true);
  177. }
  178. }
  179. @Override
  180. protected String getTitle() {
  181. return getResources().getString(R.string.nc_conversation_menu_conversation_info);
  182. }
  183. private void getListOfParticipants() {
  184. ncApi.getPeersForCall(credentials, ApiUtils.getUrlForParticipants(conversationUser.getBaseUrl(), conversationToken))
  185. .subscribeOn(Schedulers.newThread())
  186. .observeOn(AndroidSchedulers.mainThread())
  187. .subscribe(new Observer<ParticipantsOverall>() {
  188. @Override
  189. public void onSubscribe(Disposable d) {
  190. participantsDisposable = d;
  191. }
  192. @Override
  193. public void onNext(ParticipantsOverall participantsOverall) {
  194. handleParticipants(participantsOverall.getOcs().getData());
  195. }
  196. @Override
  197. public void onError(Throwable e) {
  198. }
  199. @Override
  200. public void onComplete() {
  201. participantsDisposable.dispose();
  202. }
  203. });
  204. }
  205. private void fetchRoomInfo() {
  206. ncApi.getRoom(credentials, ApiUtils.getRoom(conversationUser.getBaseUrl(), conversationToken))
  207. .subscribeOn(Schedulers.newThread())
  208. .observeOn(AndroidSchedulers.mainThread())
  209. .subscribe(new Observer<RoomOverall>() {
  210. @Override
  211. public void onSubscribe(Disposable d) {
  212. roomDisposable = d;
  213. }
  214. @Override
  215. public void onNext(RoomOverall roomOverall) {
  216. conversation = roomOverall.getOcs().getData();
  217. getListOfParticipants();
  218. if (progressBar != null) {
  219. progressBar.setVisibility(View.GONE);
  220. }
  221. loadConversationAvatar();
  222. if (conversationUser.hasSpreedCapabilityWithName("notification-levels")) {
  223. messageNotificationLevel.setEnabled(true);
  224. messageNotificationLevel.setAlpha(1.0f);
  225. if (!conversation.getNotificationLevel().equals(Conversation.NotificationLevel.DEFAULT)) {
  226. String stringValue;
  227. switch (new EnumNotificationLevelConverter().convertToInt(conversation.getNotificationLevel())) {
  228. case 1:
  229. stringValue = "always";
  230. break;
  231. case 2:
  232. stringValue = "mention";
  233. break;
  234. case 3:
  235. stringValue = "never";
  236. break;
  237. default:
  238. stringValue = "mention";
  239. break;
  240. }
  241. messageNotificationLevel.setValue(stringValue);
  242. } else {
  243. setProperNotificationValue(conversation);
  244. }
  245. } else {
  246. messageNotificationLevel.setEnabled(false);
  247. messageNotificationLevel.setAlpha(0.38f);
  248. setProperNotificationValue(conversation);
  249. }
  250. materialPreferenceScreen.setVisibility(View.VISIBLE);
  251. nameCategoryView.setVisibility(View.VISIBLE);
  252. conversationDisplayName.setText(conversation.getDisplayName());
  253. }
  254. @Override
  255. public void onError(Throwable e) {
  256. }
  257. @Override
  258. public void onComplete() {
  259. roomDisposable.dispose();
  260. }
  261. });
  262. }
  263. private void setProperNotificationValue(Conversation conversation) {
  264. if (conversation.getType().equals(Conversation.RoomType.ROOM_TYPE_ONE_TO_ONE_CALL)) {
  265. // hack to see if we get mentioned always or just on mention
  266. if (conversationUser.hasSpreedCapabilityWithName("mention-flag")) {
  267. messageNotificationLevel.setValue("always");
  268. } else {
  269. messageNotificationLevel.setValue("mention");
  270. }
  271. } else {
  272. messageNotificationLevel.setValue("mention");
  273. }
  274. }
  275. private void loadConversationAvatar() {
  276. int avatarSize = getResources().getDimensionPixelSize(R.dimen.avatar_size_big);
  277. switch (conversation.getType()) {
  278. case ROOM_TYPE_ONE_TO_ONE_CALL:
  279. if (!TextUtils.isEmpty(conversation.getName())) {
  280. GlideUrl glideUrl = new GlideUrl(ApiUtils.getUrlForAvatarWithName(conversationUser.getBaseUrl(),
  281. conversation.getName(), R.dimen.avatar_size), new LazyHeaders.Builder()
  282. .setHeader("Accept", "image/*")
  283. .setHeader("User-Agent", ApiUtils.getUserAgent())
  284. .build());
  285. GlideApp.with(NextcloudTalkApplication.getSharedApplication().getApplicationContext())
  286. .asBitmap()
  287. .diskCacheStrategy(DiskCacheStrategy.NONE)
  288. .load(glideUrl)
  289. .centerInside()
  290. .override(avatarSize, avatarSize)
  291. .apply(RequestOptions.bitmapTransform(new CircleCrop()))
  292. .into(conversationAvatarImageView);
  293. }
  294. break;
  295. case ROOM_GROUP_CALL:
  296. GlideApp.with(NextcloudTalkApplication.getSharedApplication().getApplicationContext())
  297. .asBitmap()
  298. .diskCacheStrategy(DiskCacheStrategy.NONE)
  299. .load(R.drawable.ic_people_group_white_24px)
  300. .centerInside()
  301. .override(avatarSize, avatarSize)
  302. .apply(RequestOptions.bitmapTransform(new CircleCrop()))
  303. .into(conversationAvatarImageView);
  304. break;
  305. case ROOM_PUBLIC_CALL:
  306. GlideApp.with(NextcloudTalkApplication.getSharedApplication().getApplicationContext())
  307. .asBitmap()
  308. .diskCacheStrategy(DiskCacheStrategy.NONE)
  309. .load(R.drawable.ic_link_white_24px)
  310. .centerInside()
  311. .override(avatarSize, avatarSize)
  312. .apply(RequestOptions.bitmapTransform(new CircleCrop()))
  313. .into(conversationAvatarImageView);
  314. break;
  315. default:
  316. }
  317. }
  318. }