Compare commits
1 Commits
5.2.8-BETA
...
5.2.8
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
502b4890e3 |
@@ -2,7 +2,7 @@
|
|||||||
*Pupil, Hitomi.la viewer for Android*
|
*Pupil, Hitomi.la viewer for Android*
|
||||||
|
|
||||||

|

|
||||||
[](https://github.com/tom5079/Pupil/releases/download/5.2.7/Pupil-v5.2.7.apk)
|
[](https://github.com/tom5079/Pupil/releases/download/5.2.8/Pupil-v5.2.8.apk)
|
||||||
[](https://discord.gg/Stj4b5v)
|
[](https://discord.gg/Stj4b5v)
|
||||||
|
|
||||||
# Features
|
# Features
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ android {
|
|||||||
minSdkVersion 16
|
minSdkVersion 16
|
||||||
targetSdkVersion 31
|
targetSdkVersion 31
|
||||||
versionCode 69
|
versionCode 69
|
||||||
versionName "5.2.8-BETA01"
|
versionName "5.2.8"
|
||||||
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
|
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
|
||||||
vectorDrawables.useSupportLibrary = true
|
vectorDrawables.useSupportLibrary = true
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
"filters": [],
|
"filters": [],
|
||||||
"attributes": [],
|
"attributes": [],
|
||||||
"versionCode": 69,
|
"versionCode": 69,
|
||||||
"versionName": "5.2.8-BETA01",
|
"versionName": "5.2.8",
|
||||||
"outputFile": "app-release.apk"
|
"outputFile": "app-release.apk"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -39,7 +39,6 @@ import com.github.piasy.biv.loader.fresco.FrescoImageLoader
|
|||||||
import com.google.android.gms.common.GooglePlayServicesNotAvailableException
|
import com.google.android.gms.common.GooglePlayServicesNotAvailableException
|
||||||
import com.google.android.gms.common.GooglePlayServicesRepairableException
|
import com.google.android.gms.common.GooglePlayServicesRepairableException
|
||||||
import com.google.android.gms.security.ProviderInstaller
|
import com.google.android.gms.security.ProviderInstaller
|
||||||
import com.google.android.material.snackbar.Snackbar
|
|
||||||
import com.google.firebase.crashlytics.FirebaseCrashlytics
|
import com.google.firebase.crashlytics.FirebaseCrashlytics
|
||||||
import kotlinx.coroutines.*
|
import kotlinx.coroutines.*
|
||||||
import kotlinx.coroutines.channels.BufferOverflow
|
import kotlinx.coroutines.channels.BufferOverflow
|
||||||
@@ -47,13 +46,12 @@ import kotlinx.coroutines.flow.MutableSharedFlow
|
|||||||
import kotlinx.coroutines.flow.asSharedFlow
|
import kotlinx.coroutines.flow.asSharedFlow
|
||||||
import okhttp3.*
|
import okhttp3.*
|
||||||
import xyz.quaver.io.FileX
|
import xyz.quaver.io.FileX
|
||||||
import xyz.quaver.pupil.hitomi.evaluations
|
import xyz.quaver.pupil.hitomi.evaluationContext
|
||||||
import xyz.quaver.pupil.types.Tag
|
import xyz.quaver.pupil.types.Tag
|
||||||
import xyz.quaver.pupil.util.*
|
import xyz.quaver.pupil.util.*
|
||||||
import java.io.File
|
import java.io.File
|
||||||
import java.net.URL
|
import java.net.URL
|
||||||
import java.util.*
|
import java.util.*
|
||||||
import java.util.concurrent.ConcurrentHashMap
|
|
||||||
import java.util.concurrent.Executors
|
import java.util.concurrent.Executors
|
||||||
import java.util.concurrent.TimeUnit
|
import java.util.concurrent.TimeUnit
|
||||||
import kotlin.reflect.KClass
|
import kotlin.reflect.KClass
|
||||||
@@ -96,31 +94,29 @@ fun reloadWebView() {
|
|||||||
if (reloadJob?.isActive == true) return
|
if (reloadJob?.isActive == true) return
|
||||||
|
|
||||||
reloadJob = CoroutineScope(Dispatchers.IO).launch {
|
reloadJob = CoroutineScope(Dispatchers.IO).launch {
|
||||||
if (evaluations.isEmpty()) {
|
webViewReady = false
|
||||||
webViewReady = false
|
webViewFailed = false
|
||||||
webViewFailed = false
|
|
||||||
|
|
||||||
while (evaluations.isNotEmpty()) yield()
|
evaluationContext.cancelChildren()
|
||||||
|
|
||||||
runCatching {
|
runCatching {
|
||||||
URL(
|
URL(
|
||||||
if (isDebugBuild)
|
if (isDebugBuild)
|
||||||
"https://tom5079.github.io/Pupil/hitomi-dev.html"
|
"https://tom5079.github.io/Pupil/hitomi-dev.html"
|
||||||
else
|
else
|
||||||
"https://tom5079.github.io/Pupil/hitomi.html"
|
"https://tom5079.github.io/Pupil/hitomi.html"
|
||||||
).readText()
|
).readText()
|
||||||
}.onFailure {
|
}.onFailure {
|
||||||
webViewFailed = true
|
webViewFailed = true
|
||||||
}.getOrNull()?.let { html ->
|
}.getOrNull()?.let { html ->
|
||||||
launch(Dispatchers.Main) {
|
launch(Dispatchers.Main) {
|
||||||
webView.loadDataWithBaseURL(
|
webView.loadDataWithBaseURL(
|
||||||
"https://hitomi.la/",
|
"https://hitomi.la/",
|
||||||
html,
|
html,
|
||||||
"text/html",
|
"text/html",
|
||||||
null,
|
null,
|
||||||
null
|
null
|
||||||
)
|
)
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -191,7 +187,6 @@ class Pupil : Application() {
|
|||||||
"onReceivedError: ${error?.description}"
|
"onReceivedError: ${error?.description}"
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
webViewFailed = true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -16,6 +16,7 @@
|
|||||||
|
|
||||||
package xyz.quaver.pupil.hitomi
|
package xyz.quaver.pupil.hitomi
|
||||||
|
|
||||||
|
import android.util.Log
|
||||||
import android.webkit.WebView
|
import android.webkit.WebView
|
||||||
import android.widget.Toast
|
import android.widget.Toast
|
||||||
import com.google.common.collect.ConcurrentHashMultiset
|
import com.google.common.collect.ConcurrentHashMultiset
|
||||||
@@ -37,21 +38,25 @@ import kotlin.coroutines.suspendCoroutine
|
|||||||
|
|
||||||
const val protocol = "https:"
|
const val protocol = "https:"
|
||||||
|
|
||||||
val evaluations = Collections.newSetFromMap<String>(ConcurrentHashMap())
|
val evaluationContext = Dispatchers.Main + Job()
|
||||||
|
|
||||||
suspend fun WebView.evaluate(script: String): String = withContext(Dispatchers.Main) {
|
suspend fun WebView.evaluate(script: String): String = coroutineScope {
|
||||||
val result: String = withTimeout(10000) {
|
var result: String? = null
|
||||||
while (webViewFailed || !webViewReady) yield()
|
|
||||||
|
|
||||||
val uid = UUID.randomUUID().toString()
|
while (result == null) {
|
||||||
|
try {
|
||||||
|
result = withContext(evaluationContext) {
|
||||||
|
while (webViewFailed || !webViewReady) yield()
|
||||||
|
|
||||||
evaluations.add(uid)
|
suspendCoroutine { continuation ->
|
||||||
|
evaluateJavascript(script) {
|
||||||
|
continuation.resume(it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
suspendCoroutine { continuation ->
|
|
||||||
evaluateJavascript(script) {
|
|
||||||
evaluations.remove(uid)
|
|
||||||
continuation.resume(it)
|
|
||||||
}
|
}
|
||||||
|
} catch (e: CancellationException) {
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -59,26 +64,36 @@ suspend fun WebView.evaluate(script: String): String = withContext(Dispatchers.M
|
|||||||
}
|
}
|
||||||
|
|
||||||
@OptIn(ExperimentalCoroutinesApi::class)
|
@OptIn(ExperimentalCoroutinesApi::class)
|
||||||
suspend fun WebView.evaluatePromise(script: String, then: String = ".then(result => Callback.onResult(%uid, JSON.stringify(result))).catch(err => Callback.onError(%uid, JSON.stringify(error)))"): String? = withContext(Dispatchers.Main) {
|
suspend fun WebView.evaluatePromise(
|
||||||
val flow: Flow<Pair<String, String?>> = withTimeout(10000) {
|
script: String,
|
||||||
while (webViewFailed || !webViewReady) yield()
|
then: String = ".then(result => Callback.onResult(%uid, JSON.stringify(result))).catch(err => Callback.onError(%uid, JSON.stringify(error)))"
|
||||||
|
): String? = coroutineScope {
|
||||||
|
var result: String? = null
|
||||||
|
|
||||||
val uid = UUID.randomUUID().toString()
|
while (result == null) {
|
||||||
|
try {
|
||||||
|
result = withContext(evaluationContext) {
|
||||||
|
while (webViewFailed || !webViewReady) yield()
|
||||||
|
|
||||||
evaluations.add(uid)
|
val uid = UUID.randomUUID().toString()
|
||||||
|
|
||||||
evaluateJavascript((script+then).replace("%uid", "'$uid'"), null)
|
evaluateJavascript((script + then).replace("%uid", "'$uid'"), null)
|
||||||
|
|
||||||
webViewFlow.transformWhile { (currentUid, result) ->
|
val flow: Flow<Pair<String, String?>> = webViewFlow.transformWhile { (currentUid, result) ->
|
||||||
if (currentUid == uid) {
|
if (currentUid == uid) {
|
||||||
evaluations.remove(uid)
|
emit(currentUid to result)
|
||||||
emit(currentUid to result)
|
}
|
||||||
|
currentUid != uid
|
||||||
|
}
|
||||||
|
|
||||||
|
flow.first().second
|
||||||
}
|
}
|
||||||
currentUid != uid
|
} catch (e: CancellationException) {
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
flow.first().second
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
@Suppress("EXPERIMENTAL_API_USAGE")
|
@Suppress("EXPERIMENTAL_API_USAGE")
|
||||||
@@ -108,16 +123,6 @@ suspend fun urlFromUrlFromHash(galleryID: Int, image: GalleryFiles, dir: String?
|
|||||||
""".trimIndent()
|
""".trimIndent()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
FirebaseCrashlytics.getInstance().log(
|
|
||||||
"""
|
|
||||||
url_from_url_from_hash(
|
|
||||||
${galleryID.toString().js},
|
|
||||||
${Json.encodeToString(image)},
|
|
||||||
${dir.js}, ${ext.js}, ${base.js}
|
|
||||||
)
|
|
||||||
""".trimIndent()
|
|
||||||
)
|
|
||||||
return Json.decodeFromString(result)
|
return Json.decodeFromString(result)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user