fedilab-Android-App/app/src/main/java/app/fedilab/android/jobs/NotificationsSyncJob.java

325 lines
17 KiB
Java
Raw Normal View History

2019-05-18 11:10:30 +02:00
package app.fedilab.android.jobs;
2017-08-29 16:22:57 +02:00
/* Copyright 2017 Thomas Schneider
*
2019-05-18 11:10:30 +02:00
* This file is a part of Fedilab
2017-08-29 16:22:57 +02: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.
*
2019-05-18 11:10:30 +02:00
* Fedilab is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
2017-08-29 16:22:57 +02:00
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
* Public License for more details.
*
2019-05-18 11:10:30 +02:00
* You should have received a copy of the GNU General Public License along with Fedilab; if not,
2017-08-29 16:22:57 +02:00
* see <http://www.gnu.org/licenses>. */
import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
import android.database.sqlite.SQLiteDatabase;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
2018-01-18 19:51:33 +01:00
import android.os.Handler;
import android.os.Looper;
2017-08-29 16:22:57 +02:00
import android.support.annotation.NonNull;
2017-12-02 11:02:25 +01:00
import android.support.annotation.Nullable;
2017-08-29 16:22:57 +02:00
2017-12-02 11:02:25 +01:00
import com.bumptech.glide.Glide;
import com.bumptech.glide.load.DataSource;
import com.bumptech.glide.load.engine.GlideException;
import com.bumptech.glide.request.RequestListener;
import com.bumptech.glide.request.target.SimpleTarget;
import com.bumptech.glide.request.target.Target;
import com.bumptech.glide.request.transition.Transition;
2017-08-29 16:22:57 +02:00
import com.evernote.android.job.Job;
import com.evernote.android.job.JobManager;
import com.evernote.android.job.JobRequest;
2017-11-06 07:20:52 +01:00
import java.util.ArrayList;
2017-08-29 16:22:57 +02:00
import java.util.List;
import java.util.Set;
import java.util.concurrent.TimeUnit;
2019-01-13 14:31:32 +01:00
2019-05-18 11:10:30 +02:00
import app.fedilab.android.R;
import app.fedilab.android.activities.MainActivity;
import app.fedilab.android.client.API;
import app.fedilab.android.client.APIResponse;
import app.fedilab.android.client.Entities.Account;
import app.fedilab.android.client.Entities.Notification;
import app.fedilab.android.fragments.DisplayNotificationsFragment;
import app.fedilab.android.helper.Helper;
import app.fedilab.android.sqlite.AccountDAO;
import app.fedilab.android.sqlite.Sqlite;
2017-08-29 16:22:57 +02:00
2019-05-18 11:10:30 +02:00
import static app.fedilab.android.helper.Helper.INTENT_ACTION;
import static app.fedilab.android.helper.Helper.INTENT_TARGETED_ACCOUNT;
import static app.fedilab.android.helper.Helper.NOTIFICATION_INTENT;
import static app.fedilab.android.helper.Helper.PREF_INSTANCE;
import static app.fedilab.android.helper.Helper.PREF_KEY_ID;
import static app.fedilab.android.helper.Helper.canNotify;
import static app.fedilab.android.helper.Helper.notify_user;
2017-08-29 16:22:57 +02:00
/**
* Created by Thomas on 29/04/2017.
* Notifications refresh job
*/
2018-01-16 18:26:10 +01:00
public class NotificationsSyncJob extends Job {
2017-08-29 16:22:57 +02:00
static final String NOTIFICATION_REFRESH = "job_notification";
2017-12-12 19:27:33 +01:00
static {
2017-12-14 07:26:37 +01:00
Helper.installProvider();
2017-12-12 19:27:33 +01:00
}
2017-08-29 16:22:57 +02:00
@NonNull
@Override
2017-12-02 11:29:29 +01:00
protected Result onRunJob(@NonNull Params params) {
2017-08-29 16:22:57 +02:00
//Code refresh here
callAsynchronousTask();
return Result.SUCCESS;
}
2018-08-16 15:26:39 +02:00
public static int schedule(boolean updateCurrent) {
2017-08-29 16:22:57 +02:00
Set<JobRequest> jobRequests = JobManager.instance().getAllJobRequestsForTag(NOTIFICATION_REFRESH);
if (!jobRequests.isEmpty() && !updateCurrent) {
return jobRequests.iterator().next().getJobId();
}
2018-08-16 15:26:39 +02:00
int jobRequestschedule = -1;
try {
jobRequestschedule = new JobRequest.Builder(NotificationsSyncJob.NOTIFICATION_REFRESH)
.setPeriodic(TimeUnit.MINUTES.toMillis(Helper.MINUTES_BETWEEN_NOTIFICATIONS_REFRESH), TimeUnit.MINUTES.toMillis(5))
.setUpdateCurrent(updateCurrent)
.setRequiredNetworkType(JobRequest.NetworkType.METERED)
.setRequirementsEnforced(false)
.build()
.schedule();
}catch (Exception ignored){}
return jobRequestschedule;
2017-08-29 16:22:57 +02:00
}
/**
* Task in background starts here.
*/
private void callAsynchronousTask() {
if( !canNotify(getContext()))
return;
SQLiteDatabase db = Sqlite.getInstance(getContext(), Sqlite.DB_NAME, null, Sqlite.DB_VERSION).open();
//If an Internet connection and user agrees with notification refresh
final SharedPreferences sharedpreferences = getContext().getSharedPreferences(Helper.APP_PREFS, Context.MODE_PRIVATE);
//Check which notifications the user wants to see
boolean notif_follow = sharedpreferences.getBoolean(Helper.SET_NOTIF_FOLLOW, true);
boolean notif_add = sharedpreferences.getBoolean(Helper.SET_NOTIF_ADD, true);
boolean notif_mention = sharedpreferences.getBoolean(Helper.SET_NOTIF_MENTION, true);
boolean notif_share = sharedpreferences.getBoolean(Helper.SET_NOTIF_SHARE, true);
boolean notif_poll = sharedpreferences.getBoolean(Helper.SET_NOTIF_POLL, true);
2017-08-29 16:22:57 +02:00
//User disagree with all notifications
if( !notif_follow && !notif_add && !notif_mention && !notif_share && !notif_poll)
2017-08-29 16:22:57 +02:00
return; //Nothing is done
//No account connected, the service is stopped
if(!Helper.isLoggedIn(getContext()))
return;
//If WIFI only and on WIFI OR user defined any connections to use the service.
if(!sharedpreferences.getBoolean(Helper.SET_WIFI_ONLY, false) || Helper.isOnWIFI(getContext())) {
2019-01-29 14:08:02 +01:00
List<Account> accounts = new AccountDAO(getContext(),db).getAllAccountCrossAction();
2017-08-29 16:22:57 +02:00
//It means there is no user in DB.
if( accounts == null )
return;
//Retrieve users in db that owner has.
for (Account account: accounts) {
2019-01-27 10:40:09 +01:00
if( account.getSocial() == null || account.getSocial().equals("MASTODON")|| account.getSocial().equals("PLEROMA")) {
2019-01-13 14:31:32 +01:00
API api = new API(getContext(), account.getInstance(), account.getToken());
2019-02-03 12:07:43 +01:00
APIResponse apiResponse = api.getNotificationsSince(DisplayNotificationsFragment.Type.ALL, null, false);
2019-01-13 14:31:32 +01:00
onRetrieveNotifications(apiResponse, account);
}
2017-08-29 16:22:57 +02:00
}
}
}
2018-01-16 18:26:10 +01:00
private void onRetrieveNotifications(APIResponse apiResponse, final Account account) {
2017-11-06 07:20:52 +01:00
List<Notification> notificationsReceived = apiResponse.getNotifications();
if( apiResponse.getError() != null || notificationsReceived == null || notificationsReceived.size() == 0 || account == null)
2017-08-29 16:22:57 +02:00
return;
final SharedPreferences sharedpreferences = getContext().getSharedPreferences(Helper.APP_PREFS, Context.MODE_PRIVATE);
boolean notif_follow = sharedpreferences.getBoolean(Helper.SET_NOTIF_FOLLOW, true);
boolean notif_add = sharedpreferences.getBoolean(Helper.SET_NOTIF_ADD, true);
boolean notif_mention = sharedpreferences.getBoolean(Helper.SET_NOTIF_MENTION, true);
boolean notif_share = sharedpreferences.getBoolean(Helper.SET_NOTIF_SHARE, true);
boolean notif_poll = sharedpreferences.getBoolean(Helper.SET_NOTIF_POLL, true);
final String max_id = sharedpreferences.getString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), null);
2017-11-06 07:20:52 +01:00
final List<Notification> notifications = new ArrayList<>();
int pos = 0;
for(Notification notif: notificationsReceived){
2017-11-06 12:59:42 +01:00
if( max_id == null || Long.parseLong(notif.getId()) > Long.parseLong(max_id) ) {
2017-11-06 07:20:52 +01:00
notifications.add(pos, notif);
pos++;
}
}
2017-11-06 18:30:52 +01:00
if( notifications.size() == 0 )
return;
2017-08-29 16:22:57 +02:00
//No previous notifications in cache, so no notification will be sent
int newFollows = 0;
int newAdds = 0;
int newMentions = 0;
int newShare = 0;
int newPolls = 0;
2017-08-29 16:22:57 +02:00
String notificationUrl = null;
String title = null;
final String message;
String targeted_account = null;
2018-09-16 18:03:24 +02:00
Helper.NotifType notifType = Helper.NotifType.MENTION;
2017-08-29 16:22:57 +02:00
for(Notification notification: notifications){
switch (notification.getType()){
case "mention":
2018-09-16 18:03:24 +02:00
notifType = Helper.NotifType.MENTION;
2017-08-29 16:22:57 +02:00
if(notif_mention){
newMentions++;
if( notificationUrl == null){
notificationUrl = notification.getAccount().getAvatar();
if( notification.getAccount().getDisplay_name() != null && notification.getAccount().getDisplay_name().length() > 0 )
2017-11-12 09:34:57 +01:00
title = String.format("%s %s", Helper.shortnameToUnicode(notification.getAccount().getDisplay_name(), true),getContext().getString(R.string.notif_mention));
2017-08-29 16:22:57 +02:00
else
2017-11-12 09:34:57 +01:00
title = String.format("@%s %s", notification.getAccount().getAcct(),getContext().getString(R.string.notif_mention));
2017-08-29 16:22:57 +02:00
}
}
break;
case "reblog":
2018-09-16 18:03:24 +02:00
notifType = Helper.NotifType.BOOST;
2017-08-29 16:22:57 +02:00
if(notif_share){
newShare++;
if( notificationUrl == null){
notificationUrl = notification.getAccount().getAvatar();
if( notification.getAccount().getDisplay_name() != null && notification.getAccount().getDisplay_name().length() > 0 )
2017-11-12 09:34:57 +01:00
title = String.format("%s %s", Helper.shortnameToUnicode(notification.getAccount().getDisplay_name(), true),getContext().getString(R.string.notif_reblog));
2017-08-29 16:22:57 +02:00
else
2017-11-12 09:34:57 +01:00
title = String.format("@%s %s", notification.getAccount().getAcct(),getContext().getString(R.string.notif_reblog));
2017-08-29 16:22:57 +02:00
}
}
break;
case "favourite":
2018-09-16 18:03:24 +02:00
notifType = Helper.NotifType.FAV;
2017-08-29 16:22:57 +02:00
if(notif_add){
newAdds++;
if( notificationUrl == null){
notificationUrl = notification.getAccount().getAvatar();
if( notification.getAccount().getDisplay_name() != null && notification.getAccount().getDisplay_name().length() > 0 )
2017-11-12 09:34:57 +01:00
title = String.format("%s %s", Helper.shortnameToUnicode(notification.getAccount().getDisplay_name(), true),getContext().getString(R.string.notif_favourite));
2017-08-29 16:22:57 +02:00
else
2017-11-12 09:34:57 +01:00
title = String.format("@%s %s", notification.getAccount().getAcct(),getContext().getString(R.string.notif_favourite));
2017-08-29 16:22:57 +02:00
}
}
break;
case "follow":
2018-09-16 18:03:24 +02:00
notifType = Helper.NotifType.FOLLLOW;
2017-08-29 16:22:57 +02:00
if(notif_follow){
newFollows++;
if( notificationUrl == null){
notificationUrl = notification.getAccount().getAvatar();
if( notification.getAccount().getDisplay_name() != null && notification.getAccount().getDisplay_name().length() > 0 )
2017-11-12 09:34:57 +01:00
title = String.format("%s %s", Helper.shortnameToUnicode(notification.getAccount().getDisplay_name(), true),getContext().getString(R.string.notif_follow));
2017-08-29 16:22:57 +02:00
else
2017-11-12 09:34:57 +01:00
title = String.format("@%s %s", notification.getAccount().getAcct(),getContext().getString(R.string.notif_follow));
targeted_account = notification.getAccount().getId();
2017-08-29 16:22:57 +02:00
}
}
break;
case "poll":
notifType = Helper.NotifType.POLL;
if(notif_poll){
newPolls++;
if( notificationUrl == null){
notificationUrl = notification.getAccount().getAvatar();
String userId = sharedpreferences.getString(Helper.PREF_KEY_ID, null);
if (notification.getAccount().getId() != null && notification.getAccount().getId().equals(userId))
title = getContext().getString(R.string.notif_poll_self);
else
title = getContext().getString(R.string.notif_poll);
}
}
break;
2017-08-29 16:22:57 +02:00
default:
}
}
2017-09-03 15:06:07 +02:00
int allNotifCount = newFollows + newAdds + newMentions + newShare + newPolls;
2017-08-29 16:22:57 +02:00
if( allNotifCount > 0){
//Some others notification
int other = allNotifCount -1;
if(other > 0 )
message = getContext().getResources().getQuantityString(R.plurals.other_notifications, other, other);
else
message = "";
final Intent intent = new Intent(getContext(), MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP | Intent.FLAG_ACTIVITY_NEW_TASK );
intent.putExtra(INTENT_ACTION, NOTIFICATION_INTENT);
intent.putExtra(PREF_KEY_ID, account.getId());
2018-10-24 15:19:33 +02:00
if( targeted_account != null && notifType == Helper.NotifType.FOLLLOW)
intent.putExtra(INTENT_TARGETED_ACCOUNT, targeted_account);
intent.putExtra(PREF_INSTANCE, account.getInstance());
long notif_id = Long.parseLong(account.getId());
2017-08-29 16:22:57 +02:00
final int notificationId = ((notif_id + 1) > 2147483647) ? (int) (2147483647 - notif_id - 1) : (int) (notif_id + 1);
2017-11-06 07:20:52 +01:00
if( notificationUrl != null ){
2017-12-02 11:02:25 +01:00
2017-08-29 16:22:57 +02:00
final String finalTitle = title;
2018-01-18 19:51:33 +01:00
Handler mainHandler = new Handler(Looper.getMainLooper());
2017-12-02 11:29:29 +01:00
2018-01-18 19:51:33 +01:00
final String finalNotificationUrl = notificationUrl;
2018-09-16 18:03:24 +02:00
Helper.NotifType finalNotifType = notifType;
2018-01-18 19:51:33 +01:00
Runnable myRunnable = new Runnable() {
@Override
public void run() {
Glide.with(getContext())
.asBitmap()
.load(finalNotificationUrl)
.listener(new RequestListener<Bitmap>() {
@Override
public boolean onResourceReady(Bitmap resource, Object model, Target<Bitmap> target, DataSource dataSource, boolean isFirstResource) {
return false;
}
@Override
public boolean onLoadFailed(@Nullable GlideException e, Object model, Target target, boolean isFirstResource) {
notify_user(getContext(), intent, notificationId, BitmapFactory.decodeResource(getContext().getResources(),
2018-09-16 18:03:24 +02:00
R.drawable.mastodonlogo), finalNotifType, finalTitle, message);
2018-01-18 19:51:33 +01:00
String lastNotif = sharedpreferences.getString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), null);
if( lastNotif == null || Long.parseLong(notifications.get(0).getId()) > Long.parseLong(lastNotif)){
SharedPreferences.Editor editor = sharedpreferences.edit();
editor.putString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), notifications.get(0).getId());
editor.apply();
}
return false;
}
})
.into(new SimpleTarget<Bitmap>() {
@Override
public void onResourceReady(@NonNull Bitmap resource, Transition<? super Bitmap> transition) {
2018-09-16 18:03:24 +02:00
notify_user(getContext(), intent, notificationId, resource, finalNotifType, finalTitle, message);
2018-01-18 19:51:33 +01:00
String lastNotif = sharedpreferences.getString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), null);
if( lastNotif == null || Long.parseLong(notifications.get(0).getId()) > Long.parseLong(lastNotif)){
SharedPreferences.Editor editor = sharedpreferences.edit();
editor.putString(Helper.LAST_NOTIFICATION_MAX_ID + account.getId() + account.getInstance(), notifications.get(0).getId());
editor.apply();
}
}
});
}
};
mainHandler.post(myRunnable);
2017-12-02 11:02:25 +01:00
2017-08-29 16:22:57 +02:00
}
}
}
}