LibraryUpdateNotifier.kt 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. package eu.kanade.tachiyomi.data.library
  2. import android.app.Notification
  3. import android.app.PendingIntent
  4. import android.content.Context
  5. import android.content.Intent
  6. import android.graphics.Bitmap
  7. import android.graphics.BitmapFactory
  8. import android.net.Uri
  9. import androidx.core.app.NotificationCompat
  10. import androidx.core.app.NotificationManagerCompat
  11. import com.bumptech.glide.Glide
  12. import eu.kanade.tachiyomi.R
  13. import eu.kanade.tachiyomi.data.database.models.Chapter
  14. import eu.kanade.tachiyomi.data.database.models.Manga
  15. import eu.kanade.tachiyomi.data.glide.toMangaThumbnail
  16. import eu.kanade.tachiyomi.data.notification.NotificationReceiver
  17. import eu.kanade.tachiyomi.data.notification.Notifications
  18. import eu.kanade.tachiyomi.data.preference.PreferencesHelper
  19. import eu.kanade.tachiyomi.ui.main.MainActivity
  20. import eu.kanade.tachiyomi.util.lang.chop
  21. import eu.kanade.tachiyomi.util.system.notification
  22. import eu.kanade.tachiyomi.util.system.notificationBuilder
  23. import eu.kanade.tachiyomi.util.system.notificationManager
  24. import uy.kohesive.injekt.injectLazy
  25. import java.text.DecimalFormat
  26. import java.text.DecimalFormatSymbols
  27. class LibraryUpdateNotifier(private val context: Context) {
  28. private val preferences: PreferencesHelper by injectLazy()
  29. /**
  30. * Pending intent of action that cancels the library update
  31. */
  32. private val cancelIntent by lazy {
  33. NotificationReceiver.cancelLibraryUpdatePendingBroadcast(context)
  34. }
  35. /**
  36. * Bitmap of the app for notifications.
  37. */
  38. private val notificationBitmap by lazy {
  39. BitmapFactory.decodeResource(context.resources, R.mipmap.ic_launcher)
  40. }
  41. /**
  42. * Cached progress notification to avoid creating a lot.
  43. */
  44. val progressNotificationBuilder by lazy {
  45. context.notificationBuilder(Notifications.CHANNEL_LIBRARY) {
  46. setContentTitle(context.getString(R.string.app_name))
  47. setSmallIcon(R.drawable.ic_refresh_24dp)
  48. setLargeIcon(notificationBitmap)
  49. setOngoing(true)
  50. setOnlyAlertOnce(true)
  51. addAction(R.drawable.ic_close_24dp, context.getString(android.R.string.cancel), cancelIntent)
  52. }
  53. }
  54. /**
  55. * Shows the notification containing the currently updating manga and the progress.
  56. *
  57. * @param manga the manga that's being updated.
  58. * @param current the current progress.
  59. * @param total the total progress.
  60. */
  61. fun showProgressNotification(manga: Manga, current: Int, total: Int) {
  62. val title = if (preferences.hideNotificationContent()) {
  63. context.getString(R.string.notification_check_updates)
  64. } else {
  65. manga.title
  66. }
  67. context.notificationManager.notify(
  68. Notifications.ID_LIBRARY_PROGRESS,
  69. progressNotificationBuilder
  70. .setContentTitle(title)
  71. .setProgress(total, current, false)
  72. .build()
  73. )
  74. }
  75. /**
  76. * Shows notification containing update entries that failed with action to open full log.
  77. *
  78. * @param errors List of entry titles that failed to update.
  79. * @param uri Uri for error log file containing all titles that failed.
  80. */
  81. fun showUpdateErrorNotification(errors: List<String>, uri: Uri) {
  82. if (errors.isEmpty()) {
  83. return
  84. }
  85. context.notificationManager.notify(
  86. Notifications.ID_LIBRARY_ERROR,
  87. context.notificationBuilder(Notifications.CHANNEL_LIBRARY) {
  88. setContentTitle(context.resources.getQuantityString(R.plurals.notification_update_error, errors.size, errors.size))
  89. setStyle(
  90. NotificationCompat.BigTextStyle().bigText(
  91. errors.joinToString("\n") {
  92. it.chop(NOTIF_TITLE_MAX_LEN)
  93. }
  94. )
  95. )
  96. setSmallIcon(R.drawable.ic_tachi)
  97. val errorLogIntent = NotificationReceiver.openErrorLogPendingActivity(context, uri)
  98. setContentIntent(errorLogIntent)
  99. addAction(
  100. R.drawable.nnf_ic_file_folder,
  101. context.getString(R.string.action_open_log),
  102. errorLogIntent
  103. )
  104. }
  105. .build()
  106. )
  107. }
  108. /**
  109. * Shows the notification containing the result of the update done by the service.
  110. *
  111. * @param updates a list of manga with new updates.
  112. */
  113. fun showUpdateNotifications(updates: List<Pair<Manga, Array<Chapter>>>) {
  114. if (updates.isEmpty()) {
  115. return
  116. }
  117. NotificationManagerCompat.from(context).apply {
  118. // Parent group notification
  119. notify(
  120. Notifications.ID_NEW_CHAPTERS,
  121. context.notification(Notifications.CHANNEL_NEW_CHAPTERS) {
  122. setContentTitle(context.getString(R.string.notification_new_chapters))
  123. if (updates.size == 1 && !preferences.hideNotificationContent()) {
  124. setContentText(updates.first().first.title.chop(NOTIF_TITLE_MAX_LEN))
  125. } else {
  126. setContentText(context.resources.getQuantityString(R.plurals.notification_new_chapters_summary, updates.size, updates.size))
  127. if (!preferences.hideNotificationContent()) {
  128. setStyle(
  129. NotificationCompat.BigTextStyle().bigText(
  130. updates.joinToString("\n") {
  131. it.first.title.chop(NOTIF_TITLE_MAX_LEN)
  132. }
  133. )
  134. )
  135. }
  136. }
  137. setSmallIcon(R.drawable.ic_tachi)
  138. setLargeIcon(notificationBitmap)
  139. setGroup(Notifications.GROUP_NEW_CHAPTERS)
  140. setGroupAlertBehavior(NotificationCompat.GROUP_ALERT_SUMMARY)
  141. setGroupSummary(true)
  142. priority = NotificationCompat.PRIORITY_HIGH
  143. setContentIntent(getNotificationIntent())
  144. setAutoCancel(true)
  145. }
  146. )
  147. // Per-manga notification
  148. if (!preferences.hideNotificationContent()) {
  149. updates.forEach {
  150. val (manga, chapters) = it
  151. notify(manga.id.hashCode(), createNewChaptersNotification(manga, chapters))
  152. }
  153. }
  154. }
  155. }
  156. private fun createNewChaptersNotification(manga: Manga, chapters: Array<Chapter>): Notification {
  157. return context.notification(Notifications.CHANNEL_NEW_CHAPTERS) {
  158. setContentTitle(manga.title)
  159. val description = getNewChaptersDescription(chapters)
  160. setContentText(description)
  161. setStyle(NotificationCompat.BigTextStyle().bigText(description))
  162. setSmallIcon(R.drawable.ic_tachi)
  163. val icon = getMangaIcon(manga)
  164. if (icon != null) {
  165. setLargeIcon(icon)
  166. }
  167. setGroup(Notifications.GROUP_NEW_CHAPTERS)
  168. setGroupAlertBehavior(NotificationCompat.GROUP_ALERT_SUMMARY)
  169. priority = NotificationCompat.PRIORITY_HIGH
  170. // Open first chapter on tap
  171. setContentIntent(NotificationReceiver.openChapterPendingActivity(context, manga, chapters.first()))
  172. setAutoCancel(true)
  173. // Mark chapters as read action
  174. addAction(
  175. R.drawable.ic_glasses_black_24dp,
  176. context.getString(R.string.action_mark_as_read),
  177. NotificationReceiver.markAsReadPendingBroadcast(
  178. context,
  179. manga,
  180. chapters,
  181. Notifications.ID_NEW_CHAPTERS
  182. )
  183. )
  184. // View chapters action
  185. addAction(
  186. R.drawable.ic_book_24dp,
  187. context.getString(R.string.action_view_chapters),
  188. NotificationReceiver.openChapterPendingActivity(
  189. context,
  190. manga,
  191. Notifications.ID_NEW_CHAPTERS
  192. )
  193. )
  194. }
  195. }
  196. /**
  197. * Cancels the progress notification.
  198. */
  199. fun cancelProgressNotification() {
  200. context.notificationManager.cancel(Notifications.ID_LIBRARY_PROGRESS)
  201. }
  202. private fun getMangaIcon(manga: Manga): Bitmap? {
  203. return try {
  204. Glide.with(context)
  205. .asBitmap()
  206. .load(manga.toMangaThumbnail())
  207. .dontTransform()
  208. .centerCrop()
  209. .circleCrop()
  210. .override(
  211. NOTIF_ICON_SIZE,
  212. NOTIF_ICON_SIZE
  213. )
  214. .submit()
  215. .get()
  216. } catch (e: Exception) {
  217. null
  218. }
  219. }
  220. private fun getNewChaptersDescription(chapters: Array<Chapter>): String {
  221. val formatter = DecimalFormat(
  222. "#.###",
  223. DecimalFormatSymbols()
  224. .apply { decimalSeparator = '.' }
  225. )
  226. val displayableChapterNumbers = chapters
  227. .filter { it.isRecognizedNumber }
  228. .sortedBy { it.chapter_number }
  229. .map { formatter.format(it.chapter_number) }
  230. .toSet()
  231. return when (displayableChapterNumbers.size) {
  232. // No sensible chapter numbers to show (i.e. no chapters have parsed chapter number)
  233. 0 -> {
  234. // "1 new chapter" or "5 new chapters"
  235. context.resources.getQuantityString(R.plurals.notification_chapters_generic, chapters.size, chapters.size)
  236. }
  237. // Only 1 chapter has a parsed chapter number
  238. 1 -> {
  239. val remaining = chapters.size - displayableChapterNumbers.size
  240. if (remaining == 0) {
  241. // "Chapter 2.5"
  242. context.resources.getString(R.string.notification_chapters_single, displayableChapterNumbers.first())
  243. } else {
  244. // "Chapter 2.5 and 10 more"
  245. context.resources.getString(R.string.notification_chapters_single_and_more, displayableChapterNumbers.first(), remaining)
  246. }
  247. }
  248. // Everything else (i.e. multiple parsed chapter numbers)
  249. else -> {
  250. val shouldTruncate = displayableChapterNumbers.size > NOTIF_MAX_CHAPTERS
  251. if (shouldTruncate) {
  252. // "Chapters 1, 2.5, 3, 4, 5 and 10 more"
  253. val remaining = displayableChapterNumbers.size - NOTIF_MAX_CHAPTERS
  254. val joinedChapterNumbers = displayableChapterNumbers.take(NOTIF_MAX_CHAPTERS).joinToString(", ")
  255. context.resources.getQuantityString(R.plurals.notification_chapters_multiple_and_more, remaining, joinedChapterNumbers, remaining)
  256. } else {
  257. // "Chapters 1, 2.5, 3"
  258. context.resources.getString(R.string.notification_chapters_multiple, displayableChapterNumbers.joinToString(", "))
  259. }
  260. }
  261. }
  262. }
  263. /**
  264. * Returns an intent to open the main activity.
  265. */
  266. private fun getNotificationIntent(): PendingIntent {
  267. val intent = Intent(context, MainActivity::class.java).apply {
  268. flags = Intent.FLAG_ACTIVITY_CLEAR_TOP or Intent.FLAG_ACTIVITY_SINGLE_TOP
  269. action = MainActivity.SHORTCUT_RECENTLY_UPDATED
  270. }
  271. return PendingIntent.getActivity(context, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT)
  272. }
  273. companion object {
  274. private const val NOTIF_MAX_CHAPTERS = 5
  275. private const val NOTIF_TITLE_MAX_LEN = 45
  276. private const val NOTIF_ICON_SIZE = 192
  277. }
  278. }