2017-01-20 19:09:10 +11:00
|
|
|
/* Copyright 2017 Andrew Dawson
|
|
|
|
*
|
2017-04-10 10:12:31 +10:00
|
|
|
* This file is a part of Tusky.
|
2017-01-20 19:09:10 +11:00
|
|
|
*
|
2017-04-10 10:12:31 +10:00
|
|
|
* This program is free software; you can redistribute it and/or modify it under the terms of the
|
|
|
|
* GNU General Public License as published by the Free Software Foundation; either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
2017-01-20 19:09:10 +11:00
|
|
|
*
|
|
|
|
* Tusky is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
|
2017-04-10 10:12:31 +10:00
|
|
|
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
|
|
|
|
* Public License for more details.
|
2017-01-20 19:09:10 +11:00
|
|
|
*
|
2017-04-10 10:12:31 +10:00
|
|
|
* You should have received a copy of the GNU General Public License along with Tusky; if not,
|
|
|
|
* see <http://www.gnu.org/licenses>. */
|
2017-01-20 19:09:10 +11:00
|
|
|
|
2017-05-05 08:55:34 +10:00
|
|
|
package com.keylesspalace.tusky.fragment;
|
2017-01-08 09:24:02 +11:00
|
|
|
|
|
|
|
import android.content.Context;
|
2017-03-12 05:20:10 +11:00
|
|
|
import android.content.SharedPreferences;
|
2017-01-08 09:24:02 +11:00
|
|
|
import android.graphics.drawable.Drawable;
|
|
|
|
import android.os.Bundle;
|
2017-04-22 18:41:49 +10:00
|
|
|
import android.preference.PreferenceManager;
|
2017-01-08 09:24:02 +11:00
|
|
|
import android.support.annotation.Nullable;
|
2017-04-22 18:41:49 +10:00
|
|
|
import android.support.design.widget.FloatingActionButton;
|
2017-02-13 16:18:17 +11:00
|
|
|
import android.support.design.widget.TabLayout;
|
2017-06-07 07:15:29 +10:00
|
|
|
import android.support.v4.content.LocalBroadcastManager;
|
2017-01-08 09:24:02 +11:00
|
|
|
import android.support.v4.widget.SwipeRefreshLayout;
|
|
|
|
import android.support.v7.widget.DividerItemDecoration;
|
|
|
|
import android.support.v7.widget.LinearLayoutManager;
|
|
|
|
import android.support.v7.widget.RecyclerView;
|
2017-05-24 05:34:31 +10:00
|
|
|
import android.util.Log;
|
2017-01-08 09:24:02 +11:00
|
|
|
import android.view.LayoutInflater;
|
|
|
|
import android.view.View;
|
|
|
|
import android.view.ViewGroup;
|
|
|
|
|
2017-05-05 08:55:34 +10:00
|
|
|
import com.keylesspalace.tusky.MainActivity;
|
|
|
|
import com.keylesspalace.tusky.adapter.NotificationsAdapter;
|
|
|
|
import com.keylesspalace.tusky.R;
|
2017-03-09 10:27:37 +11:00
|
|
|
import com.keylesspalace.tusky.entity.Notification;
|
|
|
|
import com.keylesspalace.tusky.entity.Status;
|
2017-05-05 08:55:34 +10:00
|
|
|
import com.keylesspalace.tusky.interfaces.StatusActionListener;
|
2017-06-07 07:15:29 +10:00
|
|
|
import com.keylesspalace.tusky.receiver.TimelineReceiver;
|
2017-05-05 08:55:34 +10:00
|
|
|
import com.keylesspalace.tusky.util.ThemeUtils;
|
2017-05-29 20:14:09 +10:00
|
|
|
import com.keylesspalace.tusky.view.EndlessOnScrollListener;
|
2017-01-08 09:24:02 +11:00
|
|
|
|
|
|
|
import java.util.List;
|
|
|
|
|
2017-03-09 10:27:37 +11:00
|
|
|
import retrofit2.Call;
|
|
|
|
import retrofit2.Callback;
|
2017-03-20 18:03:03 +11:00
|
|
|
import retrofit2.Response;
|
2017-03-09 10:27:37 +11:00
|
|
|
|
2017-01-23 10:42:05 +11:00
|
|
|
public class NotificationsFragment extends SFragment implements
|
2017-06-07 07:15:29 +10:00
|
|
|
SwipeRefreshLayout.OnRefreshListener, StatusActionListener,
|
|
|
|
NotificationsAdapter.NotificationActionListener,
|
|
|
|
SharedPreferences.OnSharedPreferenceChangeListener {
|
2017-03-10 10:20:08 +11:00
|
|
|
private static final String TAG = "Notifications"; // logging tag
|
2017-02-07 18:05:50 +11:00
|
|
|
|
2017-01-08 09:24:02 +11:00
|
|
|
private SwipeRefreshLayout swipeRefreshLayout;
|
2017-02-13 16:18:17 +11:00
|
|
|
private LinearLayoutManager layoutManager;
|
2017-04-22 18:41:49 +10:00
|
|
|
private RecyclerView recyclerView;
|
2017-02-13 16:18:17 +11:00
|
|
|
private EndlessOnScrollListener scrollListener;
|
2017-01-08 09:24:02 +11:00
|
|
|
private NotificationsAdapter adapter;
|
2017-02-13 16:18:17 +11:00
|
|
|
private TabLayout.OnTabSelectedListener onTabSelectedListener;
|
2017-03-14 22:45:19 +11:00
|
|
|
private Call<List<Notification>> listCall;
|
2017-04-22 18:41:49 +10:00
|
|
|
private boolean hideFab;
|
2017-06-07 07:15:29 +10:00
|
|
|
private TimelineReceiver timelineReceiver;
|
2017-01-08 09:24:02 +11:00
|
|
|
|
|
|
|
public static NotificationsFragment newInstance() {
|
|
|
|
NotificationsFragment fragment = new NotificationsFragment();
|
|
|
|
Bundle arguments = new Bundle();
|
|
|
|
fragment.setArguments(arguments);
|
|
|
|
return fragment;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Nullable
|
|
|
|
@Override
|
|
|
|
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
|
2017-04-22 18:41:49 +10:00
|
|
|
@Nullable Bundle savedInstanceState) {
|
2017-01-08 09:24:02 +11:00
|
|
|
View rootView = inflater.inflate(R.layout.fragment_timeline, container, false);
|
|
|
|
|
|
|
|
// Setup the SwipeRefreshLayout.
|
2017-01-23 10:42:05 +11:00
|
|
|
Context context = getContext();
|
2017-01-08 09:24:02 +11:00
|
|
|
swipeRefreshLayout = (SwipeRefreshLayout) rootView.findViewById(R.id.swipe_refresh_layout);
|
|
|
|
swipeRefreshLayout.setOnRefreshListener(this);
|
|
|
|
// Setup the RecyclerView.
|
2017-04-22 18:41:49 +10:00
|
|
|
recyclerView = (RecyclerView) rootView.findViewById(R.id.recycler_view);
|
2017-01-08 09:24:02 +11:00
|
|
|
recyclerView.setHasFixedSize(true);
|
2017-02-13 16:18:17 +11:00
|
|
|
layoutManager = new LinearLayoutManager(context);
|
2017-01-08 09:24:02 +11:00
|
|
|
recyclerView.setLayoutManager(layoutManager);
|
|
|
|
DividerItemDecoration divider = new DividerItemDecoration(
|
|
|
|
context, layoutManager.getOrientation());
|
2017-02-22 09:55:37 +11:00
|
|
|
Drawable drawable = ThemeUtils.getDrawable(context, R.attr.status_divider_drawable,
|
|
|
|
R.drawable.status_divider_dark);
|
2017-01-08 09:24:02 +11:00
|
|
|
divider.setDrawable(drawable);
|
|
|
|
recyclerView.addItemDecoration(divider);
|
2017-04-22 18:41:49 +10:00
|
|
|
|
2017-03-11 07:12:40 +11:00
|
|
|
adapter = new NotificationsAdapter(this, this);
|
2017-01-08 09:24:02 +11:00
|
|
|
recyclerView.setAdapter(adapter);
|
|
|
|
|
2017-02-13 16:18:17 +11:00
|
|
|
TabLayout layout = (TabLayout) getActivity().findViewById(R.id.tab_layout);
|
|
|
|
onTabSelectedListener = new TabLayout.OnTabSelectedListener() {
|
|
|
|
@Override
|
|
|
|
public void onTabSelected(TabLayout.Tab tab) {}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onTabUnselected(TabLayout.Tab tab) {}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onTabReselected(TabLayout.Tab tab) {
|
|
|
|
jumpToTop();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
layout.addOnTabSelectedListener(onTabSelectedListener);
|
|
|
|
|
2017-06-07 07:15:29 +10:00
|
|
|
timelineReceiver = new TimelineReceiver(adapter);
|
|
|
|
LocalBroadcastManager.getInstance(context.getApplicationContext())
|
|
|
|
.registerReceiver(timelineReceiver, TimelineReceiver.getFilter(null));
|
|
|
|
|
2017-01-08 09:24:02 +11:00
|
|
|
return rootView;
|
|
|
|
}
|
|
|
|
|
2017-04-22 18:41:49 +10:00
|
|
|
@Override
|
|
|
|
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
|
|
|
|
super.onActivityCreated(savedInstanceState);
|
|
|
|
|
|
|
|
/* This is delayed until onActivityCreated solely because MainActivity.composeButton isn't
|
|
|
|
* guaranteed to be set until then.
|
|
|
|
* Use a modified scroll listener that both loads more notifications as it goes, and hides
|
|
|
|
* the compose button on down-scroll. */
|
|
|
|
MainActivity activity = (MainActivity) getActivity();
|
|
|
|
final FloatingActionButton composeButton = activity.composeButton;
|
|
|
|
final SharedPreferences preferences = PreferenceManager.getDefaultSharedPreferences(
|
|
|
|
activity);
|
|
|
|
preferences.registerOnSharedPreferenceChangeListener(this);
|
|
|
|
hideFab = preferences.getBoolean("fabHide", false);
|
|
|
|
scrollListener = new EndlessOnScrollListener(layoutManager) {
|
|
|
|
@Override
|
|
|
|
public void onScrolled(RecyclerView view, int dx, int dy) {
|
|
|
|
super.onScrolled(view, dx, dy);
|
|
|
|
|
|
|
|
if (hideFab) {
|
|
|
|
if (dy > 0 && composeButton.isShown()) {
|
|
|
|
composeButton.hide(); // hides the button if we're scrolling down
|
|
|
|
} else if (dy < 0 && !composeButton.isShown()) {
|
|
|
|
composeButton.show(); // shows it if we are scrolling up
|
|
|
|
}
|
|
|
|
} else if (!composeButton.isShown()) {
|
|
|
|
composeButton.show();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onLoadMore(int page, int totalItemsCount, RecyclerView view) {
|
|
|
|
NotificationsAdapter adapter = (NotificationsAdapter) view.getAdapter();
|
|
|
|
Notification notification = adapter.getItem(adapter.getItemCount() - 2);
|
|
|
|
if (notification != null) {
|
|
|
|
sendFetchNotificationsRequest(notification.id, null);
|
|
|
|
} else {
|
|
|
|
sendFetchNotificationsRequest();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
recyclerView.addOnScrollListener(scrollListener);
|
|
|
|
}
|
|
|
|
|
2017-03-14 22:45:19 +11:00
|
|
|
@Override
|
|
|
|
public void onDestroy() {
|
|
|
|
super.onDestroy();
|
|
|
|
if (listCall != null) listCall.cancel();
|
|
|
|
}
|
|
|
|
|
2017-02-13 16:18:17 +11:00
|
|
|
@Override
|
|
|
|
public void onDestroyView() {
|
|
|
|
TabLayout tabLayout = (TabLayout) getActivity().findViewById(R.id.tab_layout);
|
|
|
|
tabLayout.removeOnTabSelectedListener(onTabSelectedListener);
|
2017-06-07 07:15:29 +10:00
|
|
|
|
|
|
|
LocalBroadcastManager.getInstance(getContext())
|
|
|
|
.unregisterReceiver(timelineReceiver);
|
|
|
|
|
2017-02-13 16:18:17 +11:00
|
|
|
super.onDestroyView();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void jumpToTop() {
|
|
|
|
layoutManager.scrollToPosition(0);
|
|
|
|
scrollListener.reset();
|
|
|
|
}
|
|
|
|
|
2017-03-11 07:12:40 +11:00
|
|
|
private void sendFetchNotificationsRequest(final String fromId, String uptoId) {
|
2017-04-16 03:25:39 +10:00
|
|
|
if (fromId != null || adapter.getItemCount() <= 1) {
|
|
|
|
adapter.setFooterState(NotificationsAdapter.FooterState.LOADING);
|
|
|
|
}
|
|
|
|
|
2017-04-22 13:16:05 +10:00
|
|
|
listCall = mastodonAPI.notifications(fromId, uptoId, null);
|
2017-03-14 22:45:19 +11:00
|
|
|
|
|
|
|
listCall.enqueue(new Callback<List<Notification>>() {
|
2017-01-08 09:24:02 +11:00
|
|
|
@Override
|
2017-03-20 18:03:03 +11:00
|
|
|
public void onResponse(Call<List<Notification>> call, Response<List<Notification>> response) {
|
2017-03-14 11:49:12 +11:00
|
|
|
if (response.isSuccessful()) {
|
|
|
|
onFetchNotificationsSuccess(response.body(), fromId);
|
|
|
|
} else {
|
|
|
|
onFetchNotificationsFailure(new Exception(response.message()));
|
|
|
|
}
|
2017-01-08 09:24:02 +11:00
|
|
|
}
|
2017-03-09 10:27:37 +11:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onFailure(Call<List<Notification>> call, Throwable t) {
|
|
|
|
onFetchNotificationsFailure((Exception) t);
|
|
|
|
}
|
|
|
|
});
|
2017-03-15 11:34:27 +11:00
|
|
|
callList.add(listCall);
|
2017-01-08 09:24:02 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
private void sendFetchNotificationsRequest() {
|
2017-03-11 07:12:40 +11:00
|
|
|
sendFetchNotificationsRequest(null, null);
|
2017-01-08 09:24:02 +11:00
|
|
|
}
|
|
|
|
|
2017-02-22 09:55:37 +11:00
|
|
|
private static boolean findNotification(List<Notification> notifications, String id) {
|
|
|
|
for (Notification notification : notifications) {
|
2017-03-09 10:27:37 +11:00
|
|
|
if (notification.id.equals(id)) {
|
2017-02-22 09:55:37 +11:00
|
|
|
return true;
|
|
|
|
}
|
2017-01-08 09:24:02 +11:00
|
|
|
}
|
2017-02-22 09:55:37 +11:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void onFetchNotificationsSuccess(List<Notification> notifications, String fromId) {
|
|
|
|
if (fromId != null) {
|
|
|
|
if (notifications.size() > 0 && !findNotification(notifications, fromId)) {
|
|
|
|
adapter.addItems(notifications);
|
2017-03-12 05:20:10 +11:00
|
|
|
|
|
|
|
// Set last update id for pull notifications so that we don't get notified
|
|
|
|
// about things we already loaded here
|
|
|
|
SharedPreferences preferences = getActivity().getSharedPreferences(getString(R.string.preferences_file_key), Context.MODE_PRIVATE);
|
|
|
|
SharedPreferences.Editor editor = preferences.edit();
|
|
|
|
editor.putString("lastUpdateId", notifications.get(0).id);
|
|
|
|
editor.apply();
|
2017-02-22 09:55:37 +11:00
|
|
|
}
|
2017-02-08 08:47:05 +11:00
|
|
|
} else {
|
2017-02-22 09:55:37 +11:00
|
|
|
adapter.update(notifications);
|
2017-02-08 08:47:05 +11:00
|
|
|
}
|
2017-04-16 03:25:39 +10:00
|
|
|
if (notifications.size() == 0 && adapter.getItemCount() == 1) {
|
|
|
|
adapter.setFooterState(NotificationsAdapter.FooterState.EMPTY);
|
|
|
|
} else if (fromId != null) {
|
|
|
|
adapter.setFooterState(NotificationsAdapter.FooterState.END);
|
|
|
|
}
|
2017-01-08 09:24:02 +11:00
|
|
|
swipeRefreshLayout.setRefreshing(false);
|
|
|
|
}
|
|
|
|
|
2017-02-07 18:05:50 +11:00
|
|
|
private void onFetchNotificationsFailure(Exception exception) {
|
2017-01-08 09:24:02 +11:00
|
|
|
swipeRefreshLayout.setRefreshing(false);
|
2017-02-07 18:05:50 +11:00
|
|
|
Log.e(TAG, "Fetch failure: " + exception.getMessage());
|
2017-01-08 09:24:02 +11:00
|
|
|
}
|
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
2017-01-08 09:24:02 +11:00
|
|
|
public void onRefresh() {
|
2017-03-11 07:12:40 +11:00
|
|
|
Notification notification = adapter.getItem(0);
|
2017-01-23 10:42:05 +11:00
|
|
|
if (notification != null) {
|
2017-03-11 07:12:40 +11:00
|
|
|
sendFetchNotificationsRequest(null, notification.id);
|
2017-01-23 10:42:05 +11:00
|
|
|
} else {
|
|
|
|
sendFetchNotificationsRequest();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
2017-01-23 10:42:05 +11:00
|
|
|
public void onReply(int position) {
|
|
|
|
Notification notification = adapter.getItem(position);
|
2017-03-09 10:27:37 +11:00
|
|
|
super.reply(notification.status);
|
2017-01-23 10:42:05 +11:00
|
|
|
}
|
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
2017-01-23 10:42:05 +11:00
|
|
|
public void onReblog(boolean reblog, int position) {
|
|
|
|
Notification notification = adapter.getItem(position);
|
2017-03-09 10:27:37 +11:00
|
|
|
super.reblog(notification.status, reblog, adapter, position);
|
2017-01-23 10:42:05 +11:00
|
|
|
}
|
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
2017-01-23 10:42:05 +11:00
|
|
|
public void onFavourite(boolean favourite, int position) {
|
|
|
|
Notification notification = adapter.getItem(position);
|
2017-03-09 10:27:37 +11:00
|
|
|
super.favourite(notification.status, favourite, adapter, position);
|
2017-01-23 10:42:05 +11:00
|
|
|
}
|
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
2017-01-23 10:42:05 +11:00
|
|
|
public void onMore(View view, int position) {
|
|
|
|
Notification notification = adapter.getItem(position);
|
2017-03-09 10:27:37 +11:00
|
|
|
super.more(notification.status, view, adapter, position);
|
2017-01-23 10:42:05 +11:00
|
|
|
}
|
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
|
|
|
public void onViewMedia(String[] urls, int urlIndex, Status.MediaAttachment.Type type) {
|
|
|
|
super.viewMedia(urls, urlIndex, type);
|
2017-01-23 10:42:05 +11:00
|
|
|
}
|
2017-01-23 16:19:30 +11:00
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
2017-01-23 16:19:30 +11:00
|
|
|
public void onViewThread(int position) {
|
|
|
|
Notification notification = adapter.getItem(position);
|
2017-03-09 10:27:37 +11:00
|
|
|
super.viewThread(notification.status);
|
2017-01-23 16:19:30 +11:00
|
|
|
}
|
2017-01-27 11:34:32 +11:00
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
2017-01-27 11:34:32 +11:00
|
|
|
public void onViewTag(String tag) {
|
|
|
|
super.viewTag(tag);
|
|
|
|
}
|
2017-01-28 14:33:43 +11:00
|
|
|
|
2017-06-25 15:07:41 +10:00
|
|
|
@Override
|
2017-03-03 11:25:35 +11:00
|
|
|
public void onViewAccount(String id) {
|
|
|
|
super.viewAccount(id);
|
2017-01-28 14:33:43 +11:00
|
|
|
}
|
2017-04-22 18:41:49 +10:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) {
|
|
|
|
if(key.equals("fabHide")) {
|
|
|
|
hideFab = sharedPreferences.getBoolean("fabHide", false);
|
|
|
|
}
|
|
|
|
}
|
2017-01-08 09:24:02 +11:00
|
|
|
}
|