1/*
2 * Copyright (c) 2018, Daniel Gultsch All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without modification,
5 * are permitted provided that the following conditions are met:
6 *
7 * 1. Redistributions of source code must retain the above copyright notice, this
8 * list of conditions and the following disclaimer.
9 *
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation and/or
12 * other materials provided with the distribution.
13 *
14 * 3. Neither the name of the copyright holder nor the names of its contributors
15 * may be used to endorse or promote products derived from this software without
16 * specific prior written permission.
17 *
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
19 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
22 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
25 * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29
30package eu.siacs.conversations.ui;
31
32import android.app.Activity;
33import android.app.AlertDialog;
34import android.app.Fragment;
35import android.content.Intent;
36import android.graphics.Canvas;
37import android.graphics.Paint;
38import android.os.Bundle;
39import android.util.Log;
40import android.view.LayoutInflater;
41import android.view.Menu;
42import android.view.MenuInflater;
43import android.view.MenuItem;
44import android.view.View;
45import android.view.ViewGroup;
46import android.widget.Toast;
47
48import androidx.databinding.DataBindingUtil;
49import androidx.recyclerview.widget.ItemTouchHelper;
50import androidx.recyclerview.widget.LinearLayoutManager;
51import androidx.recyclerview.widget.RecyclerView;
52
53import com.google.android.material.color.MaterialColors;
54import com.google.android.material.dialog.MaterialAlertDialogBuilder;
55import com.google.android.material.snackbar.Snackbar;
56import com.google.common.collect.Collections2;
57
58import java.util.ArrayList;
59import java.util.List;
60import java.util.concurrent.atomic.AtomicReference;
61
62import eu.siacs.conversations.BuildConfig;
63import eu.siacs.conversations.Config;
64import eu.siacs.conversations.R;
65import eu.siacs.conversations.databinding.FragmentConversationsOverviewBinding;
66import eu.siacs.conversations.entities.Account;
67import eu.siacs.conversations.entities.Conversation;
68import eu.siacs.conversations.entities.Conversational;
69import eu.siacs.conversations.services.QuickConversationsService;
70import eu.siacs.conversations.ui.adapter.ConversationAdapter;
71import eu.siacs.conversations.ui.interfaces.OnConversationArchived;
72import eu.siacs.conversations.ui.interfaces.OnConversationSelected;
73import eu.siacs.conversations.ui.util.MenuDoubleTabUtil;
74import eu.siacs.conversations.ui.util.PendingActionHelper;
75import eu.siacs.conversations.ui.util.PendingItem;
76import eu.siacs.conversations.ui.util.ScrollState;
77import eu.siacs.conversations.utils.AccountUtils;
78import eu.siacs.conversations.utils.EasyOnboardingInvite;
79
80import static androidx.recyclerview.widget.ItemTouchHelper.LEFT;
81import static androidx.recyclerview.widget.ItemTouchHelper.RIGHT;
82
83public class ConversationsOverviewFragment extends XmppFragment {
84
85 private static final String STATE_SCROLL_POSITION = ConversationsOverviewFragment.class.getName()+".scroll_state";
86
87 private final List<Conversation> conversations = new ArrayList<>();
88 private final PendingItem<Conversation> swipedConversation = new PendingItem<>();
89 private final PendingItem<ScrollState> pendingScrollState = new PendingItem<>();
90 private FragmentConversationsOverviewBinding binding;
91 private ConversationAdapter conversationsAdapter;
92 private XmppActivity activity;
93 private float mSwipeEscapeVelocity = 0f;
94 private final PendingActionHelper pendingActionHelper = new PendingActionHelper();
95
96 private final ItemTouchHelper.SimpleCallback callback = new ItemTouchHelper.SimpleCallback(0,LEFT|RIGHT) {
97 @Override
98 public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) {
99 //todo maybe we can manually changing the position of the conversation
100 return false;
101 }
102
103 @Override
104 public float getSwipeEscapeVelocity (float defaultValue) {
105 return mSwipeEscapeVelocity;
106 }
107
108 @Override
109 public void onChildDraw(Canvas c, RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder,
110 float dX, float dY, int actionState, boolean isCurrentlyActive) {
111 super.onChildDraw(c, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive);
112 if(actionState != ItemTouchHelper.ACTION_STATE_IDLE){
113 Paint paint = new Paint();
114 paint.setColor(MaterialColors.getColor(viewHolder.itemView, com.google.android.material.R.attr.colorSecondaryFixedDim));
115 paint.setStyle(Paint.Style.FILL);
116 c.drawRect(viewHolder.itemView.getLeft(),viewHolder.itemView.getTop()
117 ,viewHolder.itemView.getRight(),viewHolder.itemView.getBottom(), paint);
118 }
119 }
120
121 @Override
122 public void clearView(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {
123 super.clearView(recyclerView, viewHolder);
124 viewHolder.itemView.setAlpha(1f);
125 }
126
127 @Override
128 public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {
129 pendingActionHelper.execute();
130 int position = viewHolder.getLayoutPosition();
131 try {
132 swipedConversation.push(conversations.get(position));
133 } catch (IndexOutOfBoundsException e) {
134 return;
135 }
136 conversationsAdapter.remove(swipedConversation.peek(), position);
137 activity.xmppConnectionService.markRead(swipedConversation.peek());
138
139 if (position == 0 && conversationsAdapter.getItemCount() == 0) {
140 final Conversation c = swipedConversation.pop();
141 activity.xmppConnectionService.archiveConversation(c);
142 return;
143 }
144 final boolean formerlySelected = ConversationFragment.getConversation(getActivity()) == swipedConversation.peek();
145 if (activity instanceof OnConversationArchived) {
146 ((OnConversationArchived) activity).onConversationArchived(swipedConversation.peek());
147 }
148 final Conversation c = swipedConversation.peek();
149 final int title;
150 if (c.getMode() == Conversational.MODE_MULTI) {
151 if (c.getMucOptions().isPrivateAndNonAnonymous()) {
152 title = R.string.title_undo_swipe_out_group_chat;
153 } else {
154 title = R.string.title_undo_swipe_out_channel;
155 }
156 } else {
157 title = R.string.title_undo_swipe_out_chat;
158 }
159
160 final Snackbar snackbar = Snackbar.make(binding.list, title, 5000)
161 .setAction(R.string.undo, v -> {
162 pendingActionHelper.undo();
163 Conversation conversation = swipedConversation.pop();
164 conversationsAdapter.insert(conversation, position);
165 if (formerlySelected) {
166 if (activity instanceof OnConversationSelected) {
167 ((OnConversationSelected) activity).onConversationSelected(c);
168 }
169 }
170 LinearLayoutManager layoutManager = (LinearLayoutManager) binding.list.getLayoutManager();
171 if (position > layoutManager.findLastVisibleItemPosition()) {
172 binding.list.smoothScrollToPosition(position);
173 }
174 })
175 .addCallback(new Snackbar.Callback() {
176 @Override
177 public void onDismissed(Snackbar transientBottomBar, int event) {
178 switch (event) {
179 case DISMISS_EVENT_SWIPE:
180 case DISMISS_EVENT_TIMEOUT:
181 pendingActionHelper.execute();
182 break;
183 }
184 }
185 });
186
187 pendingActionHelper.push(() -> {
188 if (snackbar.isShownOrQueued()) {
189 snackbar.dismiss();
190 }
191 final Conversation conversation = swipedConversation.pop();
192 if(conversation != null){
193 if (!conversation.isRead() && conversation.getMode() == Conversation.MODE_SINGLE) {
194 return;
195 }
196 activity.xmppConnectionService.archiveConversation(c);
197 }
198 });
199 snackbar.show();
200 }
201 };
202
203 private ItemTouchHelper touchHelper;
204
205 public static Conversation getSuggestion(Activity activity) {
206 final Conversation exception;
207 Fragment fragment = activity.getFragmentManager().findFragmentById(R.id.main_fragment);
208 if (fragment instanceof ConversationsOverviewFragment) {
209 exception = ((ConversationsOverviewFragment) fragment).swipedConversation.peek();
210 } else {
211 exception = null;
212 }
213 return getSuggestion(activity, exception);
214 }
215
216 public static Conversation getSuggestion(Activity activity, Conversation exception) {
217 Fragment fragment = activity.getFragmentManager().findFragmentById(R.id.main_fragment);
218 if (fragment instanceof ConversationsOverviewFragment) {
219 List<Conversation> conversations = ((ConversationsOverviewFragment) fragment).conversations;
220 if (conversations.size() > 0) {
221 Conversation suggestion = conversations.get(0);
222 if (suggestion == exception) {
223 if (conversations.size() > 1) {
224 return conversations.get(1);
225 }
226 } else {
227 return suggestion;
228 }
229 }
230 }
231 return null;
232
233 }
234
235 @Override
236 public void onActivityCreated(Bundle savedInstanceState) {
237 super.onActivityCreated(savedInstanceState);
238 if (savedInstanceState == null) {
239 return;
240 }
241 pendingScrollState.push(savedInstanceState.getParcelable(STATE_SCROLL_POSITION));
242 }
243
244 @Override
245 public void onAttach(Activity activity) {
246 super.onAttach(activity);
247 if (activity instanceof XmppActivity) {
248 this.activity = (XmppActivity) activity;
249 } else {
250 throw new IllegalStateException("Trying to attach fragment to activity that is not an XmppActivity");
251 }
252 }
253 @Override
254 public void onDestroyView() {
255 Log.d(Config.LOGTAG,"ConversationsOverviewFragment.onDestroyView()");
256 super.onDestroyView();
257 this.binding = null;
258 this.conversationsAdapter = null;
259 this.touchHelper = null;
260 }
261 @Override
262 public void onDestroy() {
263 Log.d(Config.LOGTAG,"ConversationsOverviewFragment.onDestroy()");
264 super.onDestroy();
265
266 }
267 @Override
268 public void onPause() {
269 Log.d(Config.LOGTAG,"ConversationsOverviewFragment.onPause()");
270 pendingActionHelper.execute();
271 super.onPause();
272 }
273
274 @Override
275 public void onDetach() {
276 super.onDetach();
277 this.activity = null;
278 }
279
280 @Override
281 public void onCreate(Bundle savedInstanceState) {
282 super.onCreate(savedInstanceState);
283 setHasOptionsMenu(true);
284 }
285
286 @Override
287 public View onCreateView(final LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
288 this.mSwipeEscapeVelocity = getResources().getDimension(R.dimen.swipe_escape_velocity);
289 this.binding = DataBindingUtil.inflate(inflater, R.layout.fragment_conversations_overview, container, false);
290 this.binding.fab.setOnClickListener((view) -> StartConversationActivity.launch(getActivity()));
291
292 this.conversationsAdapter = new ConversationAdapter(this.activity, this.conversations);
293 this.conversationsAdapter.setConversationClickListener((view, conversation) -> {
294 if (activity instanceof OnConversationSelected) {
295 ((OnConversationSelected) activity).onConversationSelected(conversation);
296 } else {
297 Log.w(ConversationsOverviewFragment.class.getCanonicalName(), "Activity does not implement OnConversationSelected");
298 }
299 });
300 this.binding.list.setAdapter(this.conversationsAdapter);
301 this.binding.list.setLayoutManager(new LinearLayoutManager(getActivity(),LinearLayoutManager.VERTICAL,false));
302 this.binding.list.addOnScrollListener(ExtendedFabSizeChanger.of(binding.fab));
303 this.touchHelper = new ItemTouchHelper(this.callback);
304 this.touchHelper.attachToRecyclerView(this.binding.list);
305 return binding.getRoot();
306 }
307
308 @Override
309 public void onCreateOptionsMenu(Menu menu, MenuInflater menuInflater) {
310 menuInflater.inflate(R.menu.fragment_conversations_overview, menu);
311 AccountUtils.showHideMenuItems(menu);
312 final MenuItem easyOnboardInvite = menu.findItem(R.id.action_easy_invite);
313 easyOnboardInvite.setVisible(
314 EasyOnboardingInvite.anyHasSupport(
315 activity == null ? null : activity.xmppConnectionService));
316 final MenuItem privacyPolicyMenuItem = menu.findItem(R.id.action_privacy_policy);
317 privacyPolicyMenuItem.setVisible(
318 BuildConfig.PRIVACY_POLICY != null
319 && QuickConversationsService.isPlayStoreFlavor());
320 }
321
322 @Override
323 public void onBackendConnected() {
324 refresh();
325 }
326
327 @Override
328 public void onSaveInstanceState(Bundle bundle) {
329 super.onSaveInstanceState(bundle);
330 ScrollState scrollState = getScrollState();
331 if (scrollState != null) {
332 bundle.putParcelable(STATE_SCROLL_POSITION, scrollState);
333 }
334 }
335
336 private ScrollState getScrollState() {
337 if (this.binding == null) {
338 return null;
339 }
340 LinearLayoutManager layoutManager = (LinearLayoutManager) this.binding.list.getLayoutManager();
341 int position = layoutManager.findFirstVisibleItemPosition();
342 final View view = this.binding.list.getChildAt(0);
343 if (view != null) {
344 return new ScrollState(position,view.getTop());
345 } else {
346 return new ScrollState(position, 0);
347 }
348 }
349
350 @Override
351 public void onStart() {
352 super.onStart();
353 Log.d(Config.LOGTAG, "ConversationsOverviewFragment.onStart()");
354 if (activity.xmppConnectionService != null) {
355 refresh();
356 }
357 }
358
359 @Override
360 public void onResume() {
361 super.onResume();
362 Log.d(Config.LOGTAG, "ConversationsOverviewFragment.onResume()");
363 }
364
365 @Override
366 public boolean onOptionsItemSelected(final MenuItem item) {
367 if (MenuDoubleTabUtil.shouldIgnoreTap()) {
368 return false;
369 }
370 switch (item.getItemId()) {
371 case R.id.action_search:
372 startActivity(new Intent(getActivity(), SearchActivity.class));
373 return true;
374 case R.id.action_easy_invite:
375 selectAccountToStartEasyInvite();
376 return true;
377 }
378 return super.onOptionsItemSelected(item);
379 }
380
381 private void selectAccountToStartEasyInvite() {
382 final List<Account> accounts = EasyOnboardingInvite.getSupportingAccounts(activity.xmppConnectionService);
383 if (accounts.isEmpty()) {
384 //This can technically happen if opening the menu item races with accounts reconnecting or something
385 Toast.makeText(getActivity(),R.string.no_active_accounts_support_this, Toast.LENGTH_LONG).show();
386 } else if (accounts.size() == 1) {
387 openEasyInviteScreen(accounts.get(0));
388 } else {
389 final AtomicReference<Account> selectedAccount = new AtomicReference<>(accounts.get(0));
390 final MaterialAlertDialogBuilder alertDialogBuilder = new MaterialAlertDialogBuilder(activity);
391 alertDialogBuilder.setTitle(R.string.choose_account);
392 final String[] asStrings = Collections2.transform(accounts, a -> a.getJid().asBareJid().toEscapedString()).toArray(new String[0]);
393 alertDialogBuilder.setSingleChoiceItems(asStrings, 0, (dialog, which) -> selectedAccount.set(accounts.get(which)));
394 alertDialogBuilder.setNegativeButton(R.string.cancel, null);
395 alertDialogBuilder.setPositiveButton(R.string.ok, (dialog, which) -> openEasyInviteScreen(selectedAccount.get()));
396 alertDialogBuilder.create().show();
397 }
398 }
399
400 private void openEasyInviteScreen(final Account account) {
401 EasyOnboardingInviteActivity.launch(account, activity);
402 }
403
404 @Override
405 void refresh() {
406 if (this.binding == null || this.activity == null) {
407 Log.d(Config.LOGTAG,"ConversationsOverviewFragment.refresh() skipped updated because view binding or activity was null");
408 return;
409 }
410 this.activity.xmppConnectionService.populateWithOrderedConversations(this.conversations);
411 Conversation removed = this.swipedConversation.peek();
412 if (removed != null) {
413 if (removed.isRead()) {
414 this.conversations.remove(removed);
415 } else {
416 pendingActionHelper.execute();
417 }
418 }
419 this.conversationsAdapter.notifyDataSetChanged();
420 ScrollState scrollState = pendingScrollState.pop();
421 if (scrollState != null) {
422 setScrollPosition(scrollState);
423 }
424 }
425
426 private void setScrollPosition(ScrollState scrollPosition) {
427 if (scrollPosition != null) {
428 LinearLayoutManager layoutManager = (LinearLayoutManager) binding.list.getLayoutManager();
429 layoutManager.scrollToPositionWithOffset(scrollPosition.position, scrollPosition.offset);
430 }
431 }
432}