ConversationsOverviewFragment.java

  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}