mirror of
https://github.com/libre-tube/LibreTube.git
synced 2025-01-09 19:10:30 +05:30
Merge pull request #6887 from Bnyro/master
refactor: don't use Cronet directly for downloading
This commit is contained in:
commit
9f5aab4771
@ -37,7 +37,6 @@ object DownloadHelper {
|
|||||||
const val THUMBNAIL_DIR = "thumbnail"
|
const val THUMBNAIL_DIR = "thumbnail"
|
||||||
const val DOWNLOAD_CHUNK_SIZE = 8L * 1024
|
const val DOWNLOAD_CHUNK_SIZE = 8L * 1024
|
||||||
const val DEFAULT_TIMEOUT = 15 * 1000
|
const val DEFAULT_TIMEOUT = 15 * 1000
|
||||||
const val DEFAULT_RETRY = 3
|
|
||||||
private const val VIDEO_MIMETYPE = "video/*"
|
private const val VIDEO_MIMETYPE = "video/*"
|
||||||
|
|
||||||
fun getDownloadDir(context: Context, path: String): Path {
|
fun getDownloadDir(context: Context, path: String): Path {
|
||||||
|
@ -6,8 +6,10 @@ import android.app.PendingIntent.FLAG_UPDATE_CURRENT
|
|||||||
import android.content.Intent
|
import android.content.Intent
|
||||||
import android.os.Binder
|
import android.os.Binder
|
||||||
import android.os.IBinder
|
import android.os.IBinder
|
||||||
|
import android.util.Log
|
||||||
import android.util.SparseBooleanArray
|
import android.util.SparseBooleanArray
|
||||||
import androidx.core.app.NotificationCompat
|
import androidx.core.app.NotificationCompat
|
||||||
|
import androidx.core.app.NotificationCompat.Builder
|
||||||
import androidx.core.app.PendingIntentCompat
|
import androidx.core.app.PendingIntentCompat
|
||||||
import androidx.core.app.ServiceCompat
|
import androidx.core.app.ServiceCompat
|
||||||
import androidx.core.content.getSystemService
|
import androidx.core.content.getSystemService
|
||||||
@ -19,8 +21,8 @@ import androidx.lifecycle.lifecycleScope
|
|||||||
import com.github.libretube.LibreTubeApp.Companion.DOWNLOAD_CHANNEL_NAME
|
import com.github.libretube.LibreTubeApp.Companion.DOWNLOAD_CHANNEL_NAME
|
||||||
import com.github.libretube.R
|
import com.github.libretube.R
|
||||||
import com.github.libretube.api.CronetHelper
|
import com.github.libretube.api.CronetHelper
|
||||||
import com.github.libretube.api.RetrofitInstance
|
|
||||||
import com.github.libretube.api.StreamsExtractor
|
import com.github.libretube.api.StreamsExtractor
|
||||||
|
import com.github.libretube.api.obj.Streams
|
||||||
import com.github.libretube.constants.IntentData
|
import com.github.libretube.constants.IntentData
|
||||||
import com.github.libretube.db.DatabaseHolder.Database
|
import com.github.libretube.db.DatabaseHolder.Database
|
||||||
import com.github.libretube.db.obj.Download
|
import com.github.libretube.db.obj.Download
|
||||||
@ -31,6 +33,7 @@ import com.github.libretube.enums.NotificationId
|
|||||||
import com.github.libretube.extensions.formatAsFileSize
|
import com.github.libretube.extensions.formatAsFileSize
|
||||||
import com.github.libretube.extensions.getContentLength
|
import com.github.libretube.extensions.getContentLength
|
||||||
import com.github.libretube.extensions.parcelableExtra
|
import com.github.libretube.extensions.parcelableExtra
|
||||||
|
import com.github.libretube.extensions.toastFromMainDispatcher
|
||||||
import com.github.libretube.extensions.toastFromMainThread
|
import com.github.libretube.extensions.toastFromMainThread
|
||||||
import com.github.libretube.helpers.DownloadHelper
|
import com.github.libretube.helpers.DownloadHelper
|
||||||
import com.github.libretube.helpers.DownloadHelper.getNotificationId
|
import com.github.libretube.helpers.DownloadHelper.getNotificationId
|
||||||
@ -43,7 +46,7 @@ import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWN
|
|||||||
import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWNLOAD_RESUME
|
import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWNLOAD_RESUME
|
||||||
import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWNLOAD_STOP
|
import com.github.libretube.receivers.NotificationReceiver.Companion.ACTION_DOWNLOAD_STOP
|
||||||
import com.github.libretube.ui.activities.MainActivity
|
import com.github.libretube.ui.activities.MainActivity
|
||||||
import kotlinx.coroutines.CancellationException
|
import com.google.net.cronet.okhttptransport.CronetInterceptor
|
||||||
import kotlinx.coroutines.Dispatchers
|
import kotlinx.coroutines.Dispatchers
|
||||||
import kotlinx.coroutines.Job
|
import kotlinx.coroutines.Job
|
||||||
import kotlinx.coroutines.SupervisorJob
|
import kotlinx.coroutines.SupervisorJob
|
||||||
@ -55,16 +58,22 @@ import kotlinx.coroutines.launch
|
|||||||
import kotlinx.coroutines.withContext
|
import kotlinx.coroutines.withContext
|
||||||
import kotlinx.datetime.TimeZone
|
import kotlinx.datetime.TimeZone
|
||||||
import kotlinx.datetime.toLocalDateTime
|
import kotlinx.datetime.toLocalDateTime
|
||||||
|
import okhttp3.OkHttpClient
|
||||||
|
import okhttp3.Request
|
||||||
|
import okhttp3.Response
|
||||||
|
import okhttp3.ResponseBody
|
||||||
import okio.buffer
|
import okio.buffer
|
||||||
import okio.sink
|
import okio.sink
|
||||||
import okio.source
|
import okio.source
|
||||||
import java.io.File
|
import java.io.File
|
||||||
|
import java.io.IOException
|
||||||
import java.net.HttpURLConnection
|
import java.net.HttpURLConnection
|
||||||
import java.net.SocketTimeoutException
|
|
||||||
import java.net.URL
|
import java.net.URL
|
||||||
import java.nio.file.Path
|
import java.nio.file.Path
|
||||||
import java.nio.file.StandardOpenOption
|
import java.nio.file.StandardOpenOption
|
||||||
|
import java.time.Duration
|
||||||
import java.util.concurrent.Executors
|
import java.util.concurrent.Executors
|
||||||
|
import kotlin.coroutines.cancellation.CancellationException
|
||||||
import kotlin.io.path.createFile
|
import kotlin.io.path.createFile
|
||||||
import kotlin.io.path.deleteIfExists
|
import kotlin.io.path.deleteIfExists
|
||||||
import kotlin.io.path.div
|
import kotlin.io.path.div
|
||||||
@ -80,12 +89,23 @@ class DownloadService : LifecycleService() {
|
|||||||
private val coroutineContext = dispatcher + SupervisorJob()
|
private val coroutineContext = dispatcher + SupervisorJob()
|
||||||
|
|
||||||
private lateinit var notificationManager: NotificationManager
|
private lateinit var notificationManager: NotificationManager
|
||||||
private lateinit var summaryNotificationBuilder: NotificationCompat.Builder
|
private lateinit var summaryNotificationBuilder: Builder
|
||||||
|
|
||||||
private val downloadQueue = SparseBooleanArray()
|
private val downloadQueue = SparseBooleanArray()
|
||||||
private val _downloadFlow = MutableSharedFlow<Pair<Int, DownloadStatus>>()
|
private val _downloadFlow = MutableSharedFlow<Pair<Int, DownloadStatus>>()
|
||||||
val downloadFlow: SharedFlow<Pair<Int, DownloadStatus>> = _downloadFlow
|
val downloadFlow: SharedFlow<Pair<Int, DownloadStatus>> = _downloadFlow
|
||||||
|
|
||||||
|
private val httpClient: OkHttpClient by lazy {
|
||||||
|
val cronetInterceptor = CronetInterceptor.newBuilder(CronetHelper.cronetEngine).build()
|
||||||
|
|
||||||
|
OkHttpClient.Builder()
|
||||||
|
.connectTimeout(Duration.ofMillis(DownloadHelper.DEFAULT_TIMEOUT.toLong()))
|
||||||
|
.readTimeout(Duration.ofMillis(DownloadHelper.DEFAULT_TIMEOUT.toLong()))
|
||||||
|
.addInterceptor(cronetInterceptor)
|
||||||
|
.retryOnConnectionFailure(true)
|
||||||
|
.build()
|
||||||
|
}
|
||||||
|
|
||||||
override fun onCreate() {
|
override fun onCreate() {
|
||||||
super.onCreate()
|
super.onCreate()
|
||||||
IS_DOWNLOAD_RUNNING = true
|
IS_DOWNLOAD_RUNNING = true
|
||||||
@ -108,11 +128,29 @@ class DownloadService : LifecycleService() {
|
|||||||
val fileName = name.ifEmpty { videoId }
|
val fileName = name.ifEmpty { videoId }
|
||||||
|
|
||||||
lifecycleScope.launch(coroutineContext) {
|
lifecycleScope.launch(coroutineContext) {
|
||||||
try {
|
val streams = try {
|
||||||
val streams = withContext(Dispatchers.IO) {
|
withContext(Dispatchers.IO) {
|
||||||
StreamsExtractor.extractStreams(videoId)
|
StreamsExtractor.extractStreams(videoId)
|
||||||
}
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
toastFromMainDispatcher(
|
||||||
|
StreamsExtractor.getExtractorErrorMessageString(this@DownloadService, e)
|
||||||
|
)
|
||||||
|
return@launch
|
||||||
|
}
|
||||||
|
|
||||||
|
storeVideoMetadata(videoId, streams, fileName)
|
||||||
|
|
||||||
|
val downloadItems = streams.toDownloadItems(downloadData.copy(fileName = fileName))
|
||||||
|
for (downloadItem in downloadItems) {
|
||||||
|
start(downloadItem)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return START_NOT_STICKY
|
||||||
|
}
|
||||||
|
|
||||||
|
private suspend fun storeVideoMetadata(videoId: String, streams: Streams, fileName: String) {
|
||||||
val thumbnailTargetPath = getDownloadPath(DownloadHelper.THUMBNAIL_DIR, fileName)
|
val thumbnailTargetPath = getDownloadPath(DownloadHelper.THUMBNAIL_DIR, fileName)
|
||||||
|
|
||||||
val download = Download(
|
val download = Download(
|
||||||
@ -125,6 +163,7 @@ class DownloadService : LifecycleService() {
|
|||||||
thumbnailTargetPath
|
thumbnailTargetPath
|
||||||
)
|
)
|
||||||
Database.downloadDao().insertDownload(download)
|
Database.downloadDao().insertDownload(download)
|
||||||
|
|
||||||
for (chapter in streams.chapters) {
|
for (chapter in streams.chapters) {
|
||||||
val downloadChapter = DownloadChapter(
|
val downloadChapter = DownloadChapter(
|
||||||
videoId = videoId,
|
videoId = videoId,
|
||||||
@ -134,22 +173,21 @@ class DownloadService : LifecycleService() {
|
|||||||
)
|
)
|
||||||
Database.downloadDao().insertDownloadChapter(downloadChapter)
|
Database.downloadDao().insertDownloadChapter(downloadChapter)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
ImageHelper.downloadImage(
|
ImageHelper.downloadImage(
|
||||||
this@DownloadService,
|
this@DownloadService,
|
||||||
streams.thumbnailUrl,
|
streams.thumbnailUrl,
|
||||||
thumbnailTargetPath
|
thumbnailTargetPath
|
||||||
)
|
)
|
||||||
|
|
||||||
val downloadItems = streams.toDownloadItems(downloadData.copy(fileName = fileName))
|
|
||||||
downloadItems.forEach { start(it) }
|
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
return@launch
|
Log.e(
|
||||||
|
this@DownloadService::class.java.name,
|
||||||
|
"failed to download image ${streams.thumbnailUrl}"
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return START_NOT_STICKY
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initiate download [Job] using [DownloadItem] by creating file according to [FileType]
|
* Initiate download [Job] using [DownloadItem] by creating file according to [FileType]
|
||||||
* for the requested file.
|
* for the requested file.
|
||||||
@ -189,67 +227,15 @@ class DownloadService : LifecycleService() {
|
|||||||
|
|
||||||
while (totalRead < item.downloadSize) {
|
while (totalRead < item.downloadSize) {
|
||||||
try {
|
try {
|
||||||
val con = startConnection(item, url, totalRead, item.downloadSize) ?: return
|
totalRead = progressDownload(item, url, totalRead, notificationBuilder)
|
||||||
|
|
||||||
@Suppress("NewApi") // The StandardOpenOption enum is desugared.
|
|
||||||
val sink = item.path.sink(StandardOpenOption.APPEND).buffer()
|
|
||||||
val sourceByte = con.inputStream.source()
|
|
||||||
|
|
||||||
var lastTime = System.currentTimeMillis() / 1000
|
|
||||||
var lastRead = 0L
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Check if downloading is still active and read next bytes.
|
|
||||||
while (downloadQueue[item.id] && sourceByte
|
|
||||||
.read(sink.buffer, DownloadHelper.DOWNLOAD_CHUNK_SIZE)
|
|
||||||
.also { lastRead = it } != -1L
|
|
||||||
) {
|
|
||||||
sink.emit()
|
|
||||||
totalRead += lastRead
|
|
||||||
_downloadFlow.emit(
|
|
||||||
item.id to DownloadStatus.Progress(
|
|
||||||
lastRead,
|
|
||||||
totalRead,
|
|
||||||
item.downloadSize
|
|
||||||
)
|
|
||||||
)
|
|
||||||
if (item.downloadSize != -1L &&
|
|
||||||
System.currentTimeMillis() / 1000 > lastTime
|
|
||||||
) {
|
|
||||||
notificationBuilder
|
|
||||||
.setContentText(
|
|
||||||
totalRead.formatAsFileSize() + " / " +
|
|
||||||
item.downloadSize.formatAsFileSize()
|
|
||||||
)
|
|
||||||
.setProgress(
|
|
||||||
item.downloadSize.toInt(),
|
|
||||||
totalRead.toInt(),
|
|
||||||
false
|
|
||||||
)
|
|
||||||
notificationManager.notify(
|
|
||||||
item.getNotificationId(),
|
|
||||||
notificationBuilder.build()
|
|
||||||
)
|
|
||||||
lastTime = System.currentTimeMillis() / 1000
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (_: CancellationException) {
|
} catch (_: CancellationException) {
|
||||||
break
|
break
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
toastFromMainThread("${getString(R.string.download)}: ${e.message}")
|
toastFromMainThread("${getString(R.string.download)}: ${e.message}")
|
||||||
|
Log.e(this@DownloadService::class.java.name, e.stackTraceToString())
|
||||||
_downloadFlow.emit(item.id to DownloadStatus.Error(e.message.toString(), e))
|
_downloadFlow.emit(item.id to DownloadStatus.Error(e.message.toString(), e))
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
withContext(Dispatchers.IO) {
|
|
||||||
sink.flush()
|
|
||||||
sink.close()
|
|
||||||
sourceByte.close()
|
|
||||||
con.disconnect()
|
|
||||||
}
|
|
||||||
} catch (_: Exception) {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
val completed = totalRead >= item.downloadSize
|
val completed = totalRead >= item.downloadSize
|
||||||
@ -270,56 +256,131 @@ class DownloadService : LifecycleService() {
|
|||||||
stopServiceIfDone()
|
stopServiceIfDone()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private suspend fun progressDownload(
|
||||||
|
item: DownloadItem,
|
||||||
|
url: URL,
|
||||||
|
totalReadBefore: Long,
|
||||||
|
notificationBuilder: Builder
|
||||||
|
): Long {
|
||||||
|
val source =
|
||||||
|
startConnection(item, url, totalReadBefore, item.downloadSize) ?: return totalReadBefore
|
||||||
|
|
||||||
|
var totalRead = totalReadBefore
|
||||||
|
|
||||||
|
val sink = item.path.sink(StandardOpenOption.APPEND).buffer()
|
||||||
|
val sourceByte = source.byteStream().source()
|
||||||
|
|
||||||
|
var lastTime = System.currentTimeMillis() / 1000
|
||||||
|
var lastRead = 0L
|
||||||
|
|
||||||
|
// Check if downloading is still active and read next bytes.
|
||||||
|
while (downloadQueue[item.id] && sourceByte
|
||||||
|
.read(sink.buffer, DownloadHelper.DOWNLOAD_CHUNK_SIZE)
|
||||||
|
.also { lastRead = it } != -1L
|
||||||
|
) {
|
||||||
|
sink.emit()
|
||||||
|
totalRead += lastRead
|
||||||
|
_downloadFlow.emit(
|
||||||
|
item.id to DownloadStatus.Progress(
|
||||||
|
lastRead,
|
||||||
|
totalRead,
|
||||||
|
item.downloadSize
|
||||||
|
)
|
||||||
|
)
|
||||||
|
if (item.downloadSize != -1L &&
|
||||||
|
System.currentTimeMillis() / 1000 > lastTime
|
||||||
|
) {
|
||||||
|
updateNotification(notificationBuilder, item, totalRead.toInt())
|
||||||
|
|
||||||
|
lastTime = System.currentTimeMillis() / 1000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
withContext(Dispatchers.IO) {
|
||||||
|
sink.flush()
|
||||||
|
sink.close()
|
||||||
|
sourceByte.close()
|
||||||
|
source.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
return totalRead
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun updateNotification(
|
||||||
|
notificationBuilder: Builder,
|
||||||
|
item: DownloadItem,
|
||||||
|
totalRead: Int
|
||||||
|
) {
|
||||||
|
notificationBuilder
|
||||||
|
.setContentText(
|
||||||
|
totalRead.formatAsFileSize() + " / " +
|
||||||
|
item.downloadSize.formatAsFileSize()
|
||||||
|
)
|
||||||
|
.setProgress(
|
||||||
|
item.downloadSize.toInt(),
|
||||||
|
totalRead,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
notificationManager.notify(
|
||||||
|
item.getNotificationId(),
|
||||||
|
notificationBuilder.build()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
private suspend fun startConnection(
|
private suspend fun startConnection(
|
||||||
item: DownloadItem,
|
item: DownloadItem,
|
||||||
url: URL,
|
url: URL,
|
||||||
alreadyRead: Long,
|
alreadyRead: Long,
|
||||||
readLimit: Long?
|
readLimit: Long?
|
||||||
): HttpURLConnection? {
|
): ResponseBody? {
|
||||||
// Set start range where last downloading was held.
|
val limit = readLimit?.let {
|
||||||
val con = CronetHelper.cronetEngine.openConnection(url) as HttpURLConnection
|
|
||||||
con.requestMethod = "GET"
|
|
||||||
val limit = if (readLimit == null) {
|
|
||||||
""
|
|
||||||
} else {
|
|
||||||
// generate a random byte distance to make it more difficult to fingerprint
|
// generate a random byte distance to make it more difficult to fingerprint
|
||||||
val nextBytesToReadSize = (BYTES_PER_REQUEST_MIN..BYTES_PER_REQUEST_MAX).random()
|
val nextBytesToReadSize = (BYTES_PER_REQUEST_MIN..BYTES_PER_REQUEST_MAX).random()
|
||||||
min(readLimit, alreadyRead + nextBytesToReadSize)
|
min(readLimit, alreadyRead + nextBytesToReadSize)
|
||||||
}
|
}?.toString().orEmpty()
|
||||||
con.setRequestProperty("Range", "bytes=$alreadyRead-$limit")
|
|
||||||
con.connectTimeout = DownloadHelper.DEFAULT_TIMEOUT
|
|
||||||
con.readTimeout = DownloadHelper.DEFAULT_TIMEOUT
|
|
||||||
|
|
||||||
withContext(Dispatchers.IO) {
|
val request = Request.Builder()
|
||||||
|
.url(url)
|
||||||
|
.method("GET", null)
|
||||||
|
.header("Range", "bytes=$alreadyRead-$limit")
|
||||||
|
.build()
|
||||||
|
|
||||||
|
return withContext(Dispatchers.IO) {
|
||||||
// Retry connecting to server for n times.
|
// Retry connecting to server for n times.
|
||||||
for (i in 1..DownloadHelper.DEFAULT_RETRY) {
|
|
||||||
try {
|
try {
|
||||||
con.connect()
|
val call = httpClient.newCall(request)
|
||||||
break
|
val response = call.execute()
|
||||||
} catch (_: SocketTimeoutException) {
|
|
||||||
val message = getString(R.string.downloadfailed) + " " + i
|
return@withContext handleResponse(item, response)
|
||||||
|
} catch (e: IOException) {
|
||||||
|
Log.e(this::javaClass.name, e.printStackTrace().toString())
|
||||||
|
|
||||||
|
val message = getString(R.string.downloadfailed)
|
||||||
_downloadFlow.emit(item.id to DownloadStatus.Error(message))
|
_downloadFlow.emit(item.id to DownloadStatus.Error(message))
|
||||||
toastFromMainThread(message)
|
toastFromMainThread(message)
|
||||||
|
|
||||||
|
return@withContext null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private suspend fun handleResponse(item: DownloadItem, response: Response): ResponseBody? {
|
||||||
// If link is expired try to regenerate using available info.
|
// If link is expired try to regenerate using available info.
|
||||||
if (con.responseCode == 403) {
|
if (response.code == 403) {
|
||||||
regenerateLink(item)
|
regenerateLink(item)
|
||||||
con.disconnect()
|
response.close()
|
||||||
downloadFile(item)
|
downloadFile(item)
|
||||||
return null
|
return null
|
||||||
} else if (con.responseCode !in 200..299) {
|
} else if (response.code !in 200..299) {
|
||||||
val message = getString(R.string.downloadfailed) + ": " + con.responseMessage
|
val message = getString(R.string.downloadfailed) + ": " + response.message
|
||||||
_downloadFlow.emit(item.id to DownloadStatus.Error(message))
|
_downloadFlow.emit(item.id to DownloadStatus.Error(message))
|
||||||
toastFromMainThread(message)
|
toastFromMainThread(message)
|
||||||
con.disconnect()
|
response.close()
|
||||||
pause(item.id)
|
pause(item.id)
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|
||||||
return con
|
return response.body
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -413,8 +474,7 @@ class DownloadService : LifecycleService() {
|
|||||||
private fun notifyForeground() {
|
private fun notifyForeground() {
|
||||||
notificationManager = getSystemService()!!
|
notificationManager = getSystemService()!!
|
||||||
|
|
||||||
summaryNotificationBuilder = NotificationCompat
|
summaryNotificationBuilder = Builder(this, DOWNLOAD_CHANNEL_NAME)
|
||||||
.Builder(this, DOWNLOAD_CHANNEL_NAME)
|
|
||||||
.setSmallIcon(R.drawable.ic_launcher_lockscreen)
|
.setSmallIcon(R.drawable.ic_launcher_lockscreen)
|
||||||
.setContentTitle(getString(R.string.downloading))
|
.setContentTitle(getString(R.string.downloading))
|
||||||
.setForegroundServiceBehavior(NotificationCompat.FOREGROUND_SERVICE_IMMEDIATE)
|
.setForegroundServiceBehavior(NotificationCompat.FOREGROUND_SERVICE_IMMEDIATE)
|
||||||
@ -426,14 +486,13 @@ class DownloadService : LifecycleService() {
|
|||||||
startForeground(NotificationId.DOWNLOAD_IN_PROGRESS.id, summaryNotificationBuilder.build())
|
startForeground(NotificationId.DOWNLOAD_IN_PROGRESS.id, summaryNotificationBuilder.build())
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun getNotificationBuilder(item: DownloadItem): NotificationCompat.Builder {
|
private fun getNotificationBuilder(item: DownloadItem): Builder {
|
||||||
val intent = Intent(this@DownloadService, MainActivity::class.java)
|
val intent = Intent(this@DownloadService, MainActivity::class.java)
|
||||||
.putExtra("fragmentToOpen", "downloads")
|
.putExtra("fragmentToOpen", "downloads")
|
||||||
val activityIntent = PendingIntentCompat
|
val activityIntent = PendingIntentCompat
|
||||||
.getActivity(this@DownloadService, 0, intent, FLAG_CANCEL_CURRENT, false)
|
.getActivity(this@DownloadService, 0, intent, FLAG_CANCEL_CURRENT, false)
|
||||||
|
|
||||||
return NotificationCompat
|
return Builder(this, DOWNLOAD_CHANNEL_NAME)
|
||||||
.Builder(this, DOWNLOAD_CHANNEL_NAME)
|
|
||||||
.setContentTitle("[${item.type}] ${item.fileName}")
|
.setContentTitle("[${item.type}] ${item.fileName}")
|
||||||
.setProgress(0, 0, true)
|
.setProgress(0, 0, true)
|
||||||
.setOngoing(true)
|
.setOngoing(true)
|
||||||
@ -444,7 +503,7 @@ class DownloadService : LifecycleService() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun setResumeNotification(
|
private fun setResumeNotification(
|
||||||
notificationBuilder: NotificationCompat.Builder,
|
notificationBuilder: Builder,
|
||||||
item: DownloadItem
|
item: DownloadItem
|
||||||
) {
|
) {
|
||||||
notificationBuilder
|
notificationBuilder
|
||||||
@ -459,7 +518,7 @@ class DownloadService : LifecycleService() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun setPauseNotification(
|
private fun setPauseNotification(
|
||||||
notificationBuilder: NotificationCompat.Builder,
|
notificationBuilder: Builder,
|
||||||
item: DownloadItem,
|
item: DownloadItem,
|
||||||
isCompleted: Boolean = false
|
isCompleted: Boolean = false
|
||||||
) {
|
) {
|
||||||
|
Loading…
Reference in New Issue
Block a user