Remove in-app sponsor

This commit is contained in:
世界 2024-02-17 10:02:58 +08:00
parent 557ed643e0
commit a35318ded2
No known key found for this signature in database
GPG key ID: CD109927C34A63C4
8 changed files with 3 additions and 201 deletions

View file

@ -113,8 +113,6 @@ dependencies {
}
implementation 'com.google.guava:guava:32.1.2-android'
playImplementation 'com.google.android.play:app-update-ktx:2.1.0'
playImplementation "com.android.billingclient:billing:6.1.0"
playImplementation "com.android.billingclient:billing-ktx:6.1.0"
}
if (getProps("APPCENTER_TOKEN") != "") {

View file

@ -21,7 +21,6 @@ import io.nekohasekai.libbox.Libbox
import io.nekohasekai.libbox.PlatformInterface
import io.nekohasekai.libbox.SystemProxyStatus
import io.nekohasekai.sfa.Application
import io.nekohasekai.sfa.BuildConfig
import io.nekohasekai.sfa.R
import io.nekohasekai.sfa.constant.Action
import io.nekohasekai.sfa.constant.Alert
@ -53,9 +52,7 @@ class BoxService(
val tempDir = Application.application.cacheDir
tempDir.mkdirs()
Libbox.setup(baseDir.path, workingDir.path, tempDir.path, false)
if (!BuildConfig.DEBUG) {
Libbox.redirectStderr(File(workingDir, "stderr.log").path)
}
Libbox.redirectStderr(File(workingDir, "stderr.log").path)
initializeOnce = true
return
}

View file

@ -81,8 +81,6 @@ class MainActivity : AbstractActivity(), ServiceConnection.Callback {
startIntegration()
onNewIntent(intent)
Vendor.initializeBillingClient(this)
}
override fun onNewIntent(intent: Intent) {

View file

@ -104,9 +104,7 @@ class SettingsFragment : Fragment() {
startActivity(Intent(requireContext(), DebugActivity::class.java))
}
binding.startSponserButton.setOnClickListener {
Vendor.startSponsor(requireActivity()) {
activity.launchCustomTab("https://sekai.icu/sponsor/")
}
activity.launchCustomTab("https://sekai.icu/sponsors/")
}
lifecycleScope.launch(Dispatchers.IO) {
reloadSettings()

View file

@ -5,6 +5,4 @@ import android.app.Activity
interface VendorInterface {
fun checkUpdateAvailable(): Boolean
fun checkUpdate(activity: Activity, byUser: Boolean)
fun initializeBillingClient(activity: Activity)
fun startSponsor(activity: Activity, fallback: () -> Unit)
}

View file

@ -11,11 +11,4 @@ object Vendor : VendorInterface {
override fun checkUpdate(activity: Activity, byUser: Boolean) {
}
override fun initializeBillingClient(activity: Activity) {
}
override fun startSponsor(activity: Activity, fallback: () -> Unit) {
fallback()
}
}

View file

@ -1,40 +1,15 @@
package io.nekohasekai.sfa.vendor
import android.app.Activity
import android.app.ProgressDialog
import android.content.Context
import android.util.Log
import com.android.billingclient.api.AcknowledgePurchaseParams
import com.android.billingclient.api.BillingClient
import com.android.billingclient.api.BillingClientStateListener
import com.android.billingclient.api.BillingFlowParams
import com.android.billingclient.api.BillingResult
import com.android.billingclient.api.ProductDetails
import com.android.billingclient.api.Purchase
import com.android.billingclient.api.QueryProductDetailsParams
import com.android.billingclient.api.QueryPurchasesParams
import com.android.billingclient.api.queryProductDetails
import com.android.billingclient.api.queryPurchasesAsync
import com.google.android.material.dialog.MaterialAlertDialogBuilder
import com.google.android.play.core.appupdate.AppUpdateManagerFactory
import com.google.android.play.core.appupdate.AppUpdateOptions
import com.google.android.play.core.install.model.AppUpdateType
import com.google.android.play.core.install.model.InstallStatus
import com.google.android.play.core.install.model.UpdateAvailability
import io.nekohasekai.sfa.Application
import io.nekohasekai.sfa.R
import io.nekohasekai.sfa.ktx.errorDialogBuilder
import kotlinx.coroutines.Deferred
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.async
import kotlinx.coroutines.awaitAll
import kotlinx.coroutines.coroutineScope
import kotlinx.coroutines.launch
import kotlinx.coroutines.withContext
import java.util.concurrent.atomic.AtomicInteger
import kotlin.coroutines.resume
import kotlin.coroutines.suspendCoroutine
object Vendor : VendorInterface {
@ -94,164 +69,9 @@ object Vendor : VendorInterface {
}
}
private lateinit var billingClient: BillingClient
override fun initializeBillingClient(activity: Activity) {
billingClient =
BillingClient.newBuilder(Application.application).setListener { result, purchases ->
onPurchasesUpdated(activity, result, purchases)
}.enablePendingPurchases().build()
}
private fun requireConnection(callback: (String?) -> Unit) {
when (billingClient.connectionState) {
BillingClient.ConnectionState.CONNECTED -> callback(null)
else -> {
billingClient.startConnection(object : BillingClientStateListener {
override fun onBillingSetupFinished(result: BillingResult) {
if (result.responseCode == BillingClient.BillingResponseCode.OK) {
callback(null)
} else {
callback(result.toString())
}
}
override fun onBillingServiceDisconnected() {
}
})
}
}
}
override fun startSponsor(activity: Activity, fallback: () -> Unit) {
val dialog = ProgressDialog(activity)
dialog.setMessage(activity.getString(R.string.loading))
dialog.show()
requireConnection {
if (it != null) {
activity.errorDialogBuilder(it).show()
return@requireConnection
}
GlobalScope.launch(Dispatchers.IO) {
acknowledgeSponsor()
startSponsor0(activity, dialog, fallback)
}
}
}
private suspend fun startSponsor0(
activity: Activity,
dialog: ProgressDialog,
fallback: () -> Unit,
) {
val params = QueryProductDetailsParams.newBuilder().setProductList(
listOf(
QueryProductDetailsParams.Product.newBuilder().setProductId("sponsor_circle_1")
.setProductType(BillingClient.ProductType.SUBS).build(),
QueryProductDetailsParams.Product.newBuilder().setProductId("sponsor_circle_10")
.setProductType(BillingClient.ProductType.SUBS).build(),
QueryProductDetailsParams.Product.newBuilder()
.setProductId("sponsor_circle_100")
.setProductType(BillingClient.ProductType.SUBS).build(),
)
).build()
val (result, products) = billingClient.queryProductDetails(params)
withContext(Dispatchers.Main) {
dialog.dismiss()
}
if (result.responseCode != BillingClient.BillingResponseCode.OK || products.isNullOrEmpty()) {
withContext(Dispatchers.Main) {
activity.errorDialogBuilder(result.toString()).show()
}
return
}
val selecting = products.sortedBy { it.productId.substringAfterLast("_").toInt() }
val selected = AtomicInteger(0)
withContext(Dispatchers.Main) {
MaterialAlertDialogBuilder(activity).setTitle(R.string.sponsor_play)
.setSingleChoiceItems(selecting.map { it.title.removeSuffix(" (sing-box)") }
.toMutableList().also {
it.add(activity.getString(R.string.other_methods))
}.toTypedArray(), 0
) { _, which ->
selected.set(which)
}.setNeutralButton(android.R.string.cancel, null)
.setPositiveButton(R.string.action_start) { _, _ ->
if (selected.get() == selecting.size) {
fallback()
return@setPositiveButton
}
startSponsor1(activity, selecting[selected.get()])
}.show()
}
}
private fun startSponsor1(activity: Activity, product: ProductDetails) {
val paramsList = listOf(
BillingFlowParams.ProductDetailsParams.newBuilder().setProductDetails(product)
.setOfferToken(product.subscriptionOfferDetails!![0].offerToken).build()
)
val flowParams =
BillingFlowParams.newBuilder().setProductDetailsParamsList(paramsList).build()
val result = billingClient.launchBillingFlow(activity, flowParams)
if (result.responseCode != BillingClient.BillingResponseCode.OK) {
activity.errorDialogBuilder(result.toString()).show()
}
}
private fun Context.showNoUpdatesDialog() {
MaterialAlertDialogBuilder(this).setTitle(io.nekohasekai.sfa.R.string.check_update)
.setMessage(R.string.no_updates_available).setPositiveButton(R.string.ok, null).show()
}
private fun onPurchasesUpdated(
activity: Activity, result: BillingResult, purchases: List<Purchase>?
) {
if (result.responseCode != BillingClient.BillingResponseCode.OK || purchases.isNullOrEmpty()) {
return
}
requireConnection {
if (it != null) GlobalScope.launch(Dispatchers.Main) {
val dialog = ProgressDialog(activity)
dialog.setMessage(activity.getString(R.string.loading))
dialog.show()
val errorMessage = acknowledgeSponsor0(purchases)
dialog.dismiss()
if (errorMessage != null) {
activity.errorDialogBuilder(errorMessage).show()
}
}
}
}
private suspend fun acknowledgeSponsor() {
val result = billingClient.queryPurchasesAsync(
QueryPurchasesParams.newBuilder().setProductType(BillingClient.ProductType.SUBS)
.build()
)
if (result.purchasesList.isNotEmpty()) {
acknowledgeSponsor0(result.purchasesList)
}
}
private suspend fun acknowledgeSponsor0(purchases: List<Purchase>): String? = coroutineScope {
val deferred = mutableListOf<Deferred<String?>>()
for (purchase in purchases) {
deferred += async(Dispatchers.IO) {
suspendCoroutine { continuation ->
billingClient.acknowledgePurchase(
AcknowledgePurchaseParams.newBuilder()
.setPurchaseToken(purchase.purchaseToken).build()
) {
if (it.responseCode == BillingClient.BillingResponseCode.OK) {
continuation.resume(null)
} else {
continuation.resume(it.toString())
}
}
}
}
}
deferred.awaitAll().filterNotNull().firstOrNull()
}
}

View file

@ -1,3 +1,3 @@
VERSION_CODE=260
VERSION_CODE=262
VERSION_NAME=1.8.6
GO_VERSION=go1.22.0