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' implementation 'com.google.guava:guava:32.1.2-android'
playImplementation 'com.google.android.play:app-update-ktx:2.1.0' 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") != "") { if (getProps("APPCENTER_TOKEN") != "") {

View file

@ -21,7 +21,6 @@ import io.nekohasekai.libbox.Libbox
import io.nekohasekai.libbox.PlatformInterface import io.nekohasekai.libbox.PlatformInterface
import io.nekohasekai.libbox.SystemProxyStatus import io.nekohasekai.libbox.SystemProxyStatus
import io.nekohasekai.sfa.Application import io.nekohasekai.sfa.Application
import io.nekohasekai.sfa.BuildConfig
import io.nekohasekai.sfa.R import io.nekohasekai.sfa.R
import io.nekohasekai.sfa.constant.Action import io.nekohasekai.sfa.constant.Action
import io.nekohasekai.sfa.constant.Alert import io.nekohasekai.sfa.constant.Alert
@ -53,9 +52,7 @@ class BoxService(
val tempDir = Application.application.cacheDir val tempDir = Application.application.cacheDir
tempDir.mkdirs() tempDir.mkdirs()
Libbox.setup(baseDir.path, workingDir.path, tempDir.path, false) 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 initializeOnce = true
return return
} }

View file

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

View file

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

View file

@ -5,6 +5,4 @@ import android.app.Activity
interface VendorInterface { interface VendorInterface {
fun checkUpdateAvailable(): Boolean fun checkUpdateAvailable(): Boolean
fun checkUpdate(activity: Activity, byUser: 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 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 package io.nekohasekai.sfa.vendor
import android.app.Activity import android.app.Activity
import android.app.ProgressDialog
import android.content.Context import android.content.Context
import android.util.Log 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.material.dialog.MaterialAlertDialogBuilder
import com.google.android.play.core.appupdate.AppUpdateManagerFactory import com.google.android.play.core.appupdate.AppUpdateManagerFactory
import com.google.android.play.core.appupdate.AppUpdateOptions 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.AppUpdateType
import com.google.android.play.core.install.model.InstallStatus import com.google.android.play.core.install.model.InstallStatus
import com.google.android.play.core.install.model.UpdateAvailability import com.google.android.play.core.install.model.UpdateAvailability
import io.nekohasekai.sfa.Application
import io.nekohasekai.sfa.R 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 { 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() { private fun Context.showNoUpdatesDialog() {
MaterialAlertDialogBuilder(this).setTitle(io.nekohasekai.sfa.R.string.check_update) MaterialAlertDialogBuilder(this).setTitle(io.nekohasekai.sfa.R.string.check_update)
.setMessage(R.string.no_updates_available).setPositiveButton(R.string.ok, null).show() .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 VERSION_NAME=1.8.6
GO_VERSION=go1.22.0 GO_VERSION=go1.22.0