diff --git a/app/src/main/java/eu/kanade/tachiyomi/data/download/Downloader.kt b/app/src/main/java/eu/kanade/tachiyomi/data/download/Downloader.kt index 2822e3c93..422aa02df 100644 --- a/app/src/main/java/eu/kanade/tachiyomi/data/download/Downloader.kt +++ b/app/src/main/java/eu/kanade/tachiyomi/data/download/Downloader.kt @@ -1,7 +1,10 @@ package eu.kanade.tachiyomi.data.download import android.content.Context +import android.graphics.Bitmap +import android.graphics.BitmapFactory import android.webkit.MimeTypeMap +import androidx.core.graphics.BitmapCompat import com.hippo.unifile.UniFile import com.jakewharton.rxrelay.BehaviorRelay import com.jakewharton.rxrelay.PublishRelay @@ -27,6 +30,8 @@ import eu.kanade.tachiyomi.util.lang.withUIContext import eu.kanade.tachiyomi.util.storage.DiskUtil import eu.kanade.tachiyomi.util.storage.saveTo import eu.kanade.tachiyomi.util.system.ImageUtil +import eu.kanade.tachiyomi.util.system.ImageUtil.isAnimatedAndSupported +import eu.kanade.tachiyomi.util.system.ImageUtil.isTallImage import eu.kanade.tachiyomi.util.system.logcat import kotlinx.coroutines.async import logcat.LogPriority @@ -38,6 +43,8 @@ import rx.subscriptions.CompositeSubscription import uy.kohesive.injekt.injectLazy import java.io.BufferedOutputStream import java.io.File +import java.io.FileOutputStream +import java.io.OutputStream import java.util.zip.CRC32 import java.util.zip.ZipEntry import java.util.zip.ZipOutputStream @@ -345,7 +352,12 @@ class Downloader( .flatMap({ page -> getOrDownloadImage(page, download, tmpDir) }, 5) .onBackpressureLatest() // Do when page is downloaded. - .doOnNext { notifier.onProgressChange(download) } + .doOnNext { page -> + if (preferences.splitTallImages().get()) { + splitTallImage(page, download, tmpDir) + } + notifier.onProgressChange(download) + } .toList() .map { download } // Do after download completes @@ -379,7 +391,7 @@ class Downloader( tmpFile?.delete() // Try to find the image file. - val imageFile = tmpDir.listFiles()!!.find { it.name!!.startsWith("$filename.") } + val imageFile = tmpDir.listFiles()!!.find { it.name!!.startsWith("$filename.") || it.name!!.contains("${filename}__001") } // If the image is already downloaded, do nothing. Otherwise download from network val pageObservable = when { @@ -490,7 +502,7 @@ class Downloader( dirname: String, ) { // Ensure that the chapter folder has all the images. - val downloadedImages = tmpDir.listFiles().orEmpty().filterNot { it.name!!.endsWith(".tmp") } + val downloadedImages = tmpDir.listFiles().orEmpty().filterNot { it.name!!.endsWith(".tmp") || (it.name!!.contains("__") && !it.name!!.contains("__001.jpg")) } download.status = if (downloadedImages.size == download.pages!!.size) { Download.State.DOWNLOADED @@ -545,6 +557,57 @@ class Downloader( tmpDir.delete() } + /** + * Splits tall images to improve performance of reader + */ + private fun splitTallImage(page: Page, download: Download, tmpDir: UniFile) { + val filename = String.format("%03d", page.number) + val imageFile = tmpDir.listFiles()!!.find { it.name!!.startsWith("$filename.") } + if (imageFile == null) { + notifier.onError("Error: imageFile was not found", download.chapter.name, download.manga.title) + return + } + + if (!isAnimatedAndSupported(imageFile.openInputStream()) && isTallImage(imageFile.openInputStream())) { + // Getting the scaled bitmap of the source image + val bitmap = BitmapFactory.decodeFile(imageFile.filePath) + val scaledBitmap: Bitmap = + BitmapCompat.createScaledBitmap(bitmap, bitmap.width, bitmap.height, null, true) + + val splitsCount: Int = bitmap.height / context.resources.displayMetrics.heightPixels + 1 + val splitHeight = bitmap.height / splitsCount + + // xCoord and yCoord are the pixel positions of the image splits + val xCoord = 0 + var yCoord = 0 + try { + for (i in 0 until splitsCount) { + val splitPath = imageFile.filePath!!.substringBeforeLast(".") + "__${"%03d".format(i + 1)}.jpg" + // Compress the bitmap and save in jpg format + val stream: OutputStream = FileOutputStream(splitPath) + stream.use { + Bitmap.createBitmap( + scaledBitmap, + xCoord, + yCoord, + bitmap.width, + splitHeight, + ).compress(Bitmap.CompressFormat.JPEG, 100, stream) + } + yCoord += splitHeight + } + imageFile.delete() + } catch (e: Exception) { + // Image splits were not successfully saved so delete them and keep the original image + for (i in 0 until splitsCount) { + val splitPath = imageFile.filePath!!.substringBeforeLast(".") + "__${"%03d".format(i + 1)}.jpg" + File(splitPath).delete() + } + throw e + } + } + } + /** * Completes a download. This method is called in the main thread. */ diff --git a/app/src/main/java/eu/kanade/tachiyomi/data/preference/PreferencesHelper.kt b/app/src/main/java/eu/kanade/tachiyomi/data/preference/PreferencesHelper.kt index 082c7128a..3aa1b028a 100644 --- a/app/src/main/java/eu/kanade/tachiyomi/data/preference/PreferencesHelper.kt +++ b/app/src/main/java/eu/kanade/tachiyomi/data/preference/PreferencesHelper.kt @@ -200,6 +200,8 @@ class PreferencesHelper(val context: Context) { fun saveChaptersAsCBZ() = flowPrefs.getBoolean("save_chapter_as_cbz", true) + fun splitTallImages() = flowPrefs.getBoolean("split_tall_images", false) + fun folderPerManga() = prefs.getBoolean(Keys.folderPerManga, false) fun numberOfBackups() = flowPrefs.getInt("backup_slots", 2) diff --git a/app/src/main/java/eu/kanade/tachiyomi/ui/setting/SettingsDownloadController.kt b/app/src/main/java/eu/kanade/tachiyomi/ui/setting/SettingsDownloadController.kt index 74fce340b..eaf43573e 100644 --- a/app/src/main/java/eu/kanade/tachiyomi/ui/setting/SettingsDownloadController.kt +++ b/app/src/main/java/eu/kanade/tachiyomi/ui/setting/SettingsDownloadController.kt @@ -24,6 +24,7 @@ import eu.kanade.tachiyomi.util.preference.multiSelectListPreference import eu.kanade.tachiyomi.util.preference.onClick import eu.kanade.tachiyomi.util.preference.preference import eu.kanade.tachiyomi.util.preference.preferenceCategory +import eu.kanade.tachiyomi.util.preference.summaryRes import eu.kanade.tachiyomi.util.preference.switchPreference import eu.kanade.tachiyomi.util.preference.titleRes import eu.kanade.tachiyomi.util.system.toast @@ -72,6 +73,12 @@ class SettingsDownloadController : SettingsController() { bindTo(preferences.saveChaptersAsCBZ()) titleRes = R.string.save_chapter_as_cbz } + switchPreference { + bindTo(preferences.splitTallImages()) + titleRes = R.string.split_tall_images + summaryRes = R.string.split_tall_images_summary + } + preferenceCategory { titleRes = R.string.pref_category_delete_chapters diff --git a/app/src/main/java/eu/kanade/tachiyomi/util/system/ImageUtil.kt b/app/src/main/java/eu/kanade/tachiyomi/util/system/ImageUtil.kt index 22c9fc440..8e9eeda55 100644 --- a/app/src/main/java/eu/kanade/tachiyomi/util/system/ImageUtil.kt +++ b/app/src/main/java/eu/kanade/tachiyomi/util/system/ImageUtil.kt @@ -115,6 +115,24 @@ object ImageUtil { return options.outWidth > options.outHeight } + /** + * Check whether the image is considered a tall image + * @return true if the height:width ratio is greater than the 3:! + */ + fun isTallImage(imageStream: InputStream): Boolean { + imageStream.mark(imageStream.available() + 1) + + val imageBytes = imageStream.readBytes() + // Checking the image dimensions without loading it in the memory. + val options = BitmapFactory.Options().apply { inJustDecodeBounds = true } + BitmapFactory.decodeByteArray(imageBytes, 0, imageBytes.size, options) + val width = options.outWidth + val height = options.outHeight + val ratio = height / width + + return ratio > 3 + } + /** * Extract the 'side' part from imageStream and return it as InputStream. */ diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index caf74d51d..c5e281309 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -407,6 +407,8 @@ Download new chapters Manga in excluded categories will not be downloaded even if they are also in included categories. Save as CBZ archive + Auto split tall images + Improves reader performance by splitting tall downloaded images. Tracking guide