mirror of
https://github.com/SagerNet/sing-box-for-android.git
synced 2025-04-04 20:37:40 +03:00
Fix sponsor not acknowledged
This commit is contained in:
parent
4e4b01b7b9
commit
550e3b690f
3 changed files with 131 additions and 72 deletions
|
@ -80,6 +80,8 @@ class MainActivity : AbstractActivity(), ServiceConnection.Callback {
|
||||||
startIntegration()
|
startIntegration()
|
||||||
|
|
||||||
onNewIntent(intent)
|
onNewIntent(intent)
|
||||||
|
|
||||||
|
Vendor.initializeBillingClient(this)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onNewIntent(intent: Intent) {
|
override fun onNewIntent(intent: Intent) {
|
||||||
|
|
|
@ -5,5 +5,6 @@ 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)
|
fun startSponsor(activity: Activity, fallback: () -> Unit)
|
||||||
}
|
}
|
|
@ -4,13 +4,17 @@ import android.app.Activity
|
||||||
import android.app.ProgressDialog
|
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.BillingClient
|
||||||
import com.android.billingclient.api.BillingClientStateListener
|
import com.android.billingclient.api.BillingClientStateListener
|
||||||
import com.android.billingclient.api.BillingFlowParams
|
import com.android.billingclient.api.BillingFlowParams
|
||||||
import com.android.billingclient.api.BillingResult
|
import com.android.billingclient.api.BillingResult
|
||||||
import com.android.billingclient.api.ProductDetails
|
import com.android.billingclient.api.ProductDetails
|
||||||
|
import com.android.billingclient.api.Purchase
|
||||||
import com.android.billingclient.api.QueryProductDetailsParams
|
import com.android.billingclient.api.QueryProductDetailsParams
|
||||||
|
import com.android.billingclient.api.QueryPurchasesParams
|
||||||
import com.android.billingclient.api.queryProductDetails
|
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
|
||||||
|
@ -20,11 +24,17 @@ import com.google.android.play.core.install.model.UpdateAvailability
|
||||||
import io.nekohasekai.sfa.Application
|
import io.nekohasekai.sfa.Application
|
||||||
import io.nekohasekai.sfa.R
|
import io.nekohasekai.sfa.R
|
||||||
import io.nekohasekai.sfa.ktx.errorDialogBuilder
|
import io.nekohasekai.sfa.ktx.errorDialogBuilder
|
||||||
|
import kotlinx.coroutines.Deferred
|
||||||
import kotlinx.coroutines.Dispatchers
|
import kotlinx.coroutines.Dispatchers
|
||||||
import kotlinx.coroutines.GlobalScope
|
import kotlinx.coroutines.GlobalScope
|
||||||
|
import kotlinx.coroutines.async
|
||||||
|
import kotlinx.coroutines.awaitAll
|
||||||
|
import kotlinx.coroutines.coroutineScope
|
||||||
import kotlinx.coroutines.launch
|
import kotlinx.coroutines.launch
|
||||||
import kotlinx.coroutines.withContext
|
import kotlinx.coroutines.withContext
|
||||||
import java.util.concurrent.atomic.AtomicInteger
|
import java.util.concurrent.atomic.AtomicInteger
|
||||||
|
import kotlin.coroutines.resume
|
||||||
|
import kotlin.coroutines.suspendCoroutine
|
||||||
|
|
||||||
object Vendor : VendorInterface {
|
object Vendor : VendorInterface {
|
||||||
|
|
||||||
|
@ -85,105 +95,103 @@ object Vendor : VendorInterface {
|
||||||
}
|
}
|
||||||
|
|
||||||
private lateinit var billingClient: BillingClient
|
private lateinit var billingClient: BillingClient
|
||||||
override fun startSponsor(activity: Activity, fallback: () -> Unit) {
|
override fun initializeBillingClient(activity: Activity) {
|
||||||
if (!::billingClient.isInitialized) {
|
billingClient =
|
||||||
billingClient = BillingClient.newBuilder(Application.application)
|
BillingClient.newBuilder(Application.application).setListener { result, purchases ->
|
||||||
.setListener { _, _ ->
|
onPurchasesUpdated(activity, result, purchases)
|
||||||
}
|
}.enablePendingPurchases().build()
|
||||||
.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)
|
val dialog = ProgressDialog(activity)
|
||||||
dialog.setMessage(activity.getString(R.string.loading))
|
dialog.setMessage(activity.getString(R.string.loading))
|
||||||
dialog.show()
|
dialog.show()
|
||||||
billingClient.startConnection(object : BillingClientStateListener {
|
requireConnection {
|
||||||
override fun onBillingSetupFinished(result: BillingResult) {
|
if (it != null) {
|
||||||
dialog.dismiss()
|
activity.errorDialogBuilder(it).show()
|
||||||
if (result.responseCode == BillingClient.BillingResponseCode.OK) {
|
return@requireConnection
|
||||||
GlobalScope.launch(Dispatchers.IO) {
|
|
||||||
runCatching {
|
|
||||||
startSponsor0(activity, fallback)
|
|
||||||
}.onFailure { exception ->
|
|
||||||
withContext(Dispatchers.Main) {
|
|
||||||
activity.errorDialogBuilder(exception).show()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
GlobalScope.launch(Dispatchers.Main) {
|
|
||||||
activity.errorDialogBuilder(result.toString()).show()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
GlobalScope.launch(Dispatchers.IO) {
|
||||||
override fun onBillingServiceDisconnected() {
|
acknowledgeSponsor()
|
||||||
|
startSponsor0(activity, dialog, fallback)
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private suspend fun startSponsor0(activity: Activity, fallback: () -> Unit) {
|
private suspend fun startSponsor0(
|
||||||
val params = QueryProductDetailsParams.newBuilder()
|
activity: Activity,
|
||||||
.setProductList(
|
dialog: ProgressDialog,
|
||||||
listOf(
|
fallback: () -> Unit,
|
||||||
QueryProductDetailsParams.Product.newBuilder()
|
) {
|
||||||
.setProductId("sponsor_circle_1")
|
val params = QueryProductDetailsParams.newBuilder().setProductList(
|
||||||
.setProductType(BillingClient.ProductType.SUBS)
|
listOf(
|
||||||
.build(),
|
QueryProductDetailsParams.Product.newBuilder().setProductId("sponsor_circle_1")
|
||||||
QueryProductDetailsParams.Product.newBuilder()
|
.setProductType(BillingClient.ProductType.SUBS).build(),
|
||||||
.setProductId("sponsor_circle_10")
|
QueryProductDetailsParams.Product.newBuilder().setProductId("sponsor_circle_10")
|
||||||
.setProductType(BillingClient.ProductType.SUBS)
|
.setProductType(BillingClient.ProductType.SUBS).build(),
|
||||||
.build(),
|
QueryProductDetailsParams.Product.newBuilder()
|
||||||
QueryProductDetailsParams.Product.newBuilder()
|
.setProductId("sponsor_circle_100")
|
||||||
.setProductId("sponsor_circle_100")
|
.setProductType(BillingClient.ProductType.SUBS).build(),
|
||||||
.setProductType(BillingClient.ProductType.SUBS)
|
)
|
||||||
.build(),
|
).build()
|
||||||
)
|
|
||||||
).build()
|
|
||||||
val (result, products) = billingClient.queryProductDetails(params)
|
val (result, products) = billingClient.queryProductDetails(params)
|
||||||
if (result.responseCode != BillingClient.BillingResponseCode.OK) {
|
withContext(Dispatchers.Main) {
|
||||||
error(result.toString())
|
dialog.dismiss()
|
||||||
}
|
}
|
||||||
if (products.isNullOrEmpty()) {
|
if (result.responseCode != BillingClient.BillingResponseCode.OK || products.isNullOrEmpty()) {
|
||||||
withContext(Dispatchers.Main) {
|
withContext(Dispatchers.Main) {
|
||||||
fallback()
|
activity.errorDialogBuilder(result.toString()).show()
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
val selecting = products.sortedBy { it.productId.substringAfterLast("_").toInt() }
|
val selecting = products.sortedBy { it.productId.substringAfterLast("_").toInt() }
|
||||||
val selected = AtomicInteger(0)
|
val selected = AtomicInteger(0)
|
||||||
withContext(Dispatchers.Main) {
|
withContext(Dispatchers.Main) {
|
||||||
MaterialAlertDialogBuilder(activity)
|
MaterialAlertDialogBuilder(activity).setTitle(R.string.sponsor_play)
|
||||||
.setTitle(R.string.sponsor_play)
|
.setSingleChoiceItems(selecting.map { it.title.removeSuffix(" (sing-box)") }
|
||||||
.setSingleChoiceItems(
|
.toMutableList().also {
|
||||||
selecting.map { it.title.removeSuffix(" (sing-box)") }.toMutableList().also {
|
|
||||||
it.add(activity.getString(R.string.other_methods))
|
it.add(activity.getString(R.string.other_methods))
|
||||||
}.toTypedArray(),
|
}.toTypedArray(), 0
|
||||||
0
|
|
||||||
) { _, which ->
|
) { _, which ->
|
||||||
selected.set(which)
|
selected.set(which)
|
||||||
}
|
}.setNeutralButton(android.R.string.cancel, null)
|
||||||
.setNeutralButton(android.R.string.cancel, null)
|
|
||||||
.setPositiveButton(R.string.action_start) { _, _ ->
|
.setPositiveButton(R.string.action_start) { _, _ ->
|
||||||
if (selected.get() == selecting.size) {
|
if (selected.get() == selecting.size) {
|
||||||
fallback()
|
fallback()
|
||||||
return@setPositiveButton
|
return@setPositiveButton
|
||||||
}
|
}
|
||||||
startSponsor1(activity, selecting[selected.get()])
|
startSponsor1(activity, selecting[selected.get()])
|
||||||
}
|
}.show()
|
||||||
.show()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun startSponsor1(activity: Activity, product: ProductDetails) {
|
private fun startSponsor1(activity: Activity, product: ProductDetails) {
|
||||||
val paramsList = listOf(
|
val paramsList = listOf(
|
||||||
BillingFlowParams.ProductDetailsParams.newBuilder()
|
BillingFlowParams.ProductDetailsParams.newBuilder().setProductDetails(product)
|
||||||
.setProductDetails(product)
|
.setOfferToken(product.subscriptionOfferDetails!![0].offerToken).build()
|
||||||
.setOfferToken(product.subscriptionOfferDetails!![0].offerToken)
|
|
||||||
.build()
|
|
||||||
)
|
)
|
||||||
val flowParams = BillingFlowParams.newBuilder()
|
val flowParams =
|
||||||
.setProductDetailsParamsList(paramsList)
|
BillingFlowParams.newBuilder().setProductDetailsParamsList(paramsList).build()
|
||||||
.build()
|
|
||||||
val result = billingClient.launchBillingFlow(activity, flowParams)
|
val result = billingClient.launchBillingFlow(activity, flowParams)
|
||||||
if (result.responseCode != BillingClient.BillingResponseCode.OK) {
|
if (result.responseCode != BillingClient.BillingResponseCode.OK) {
|
||||||
activity.errorDialogBuilder(result.toString()).show()
|
activity.errorDialogBuilder(result.toString()).show()
|
||||||
|
@ -191,11 +199,59 @@ object Vendor : VendorInterface {
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun Context.showNoUpdatesDialog() {
|
private fun Context.showNoUpdatesDialog() {
|
||||||
MaterialAlertDialogBuilder(this)
|
MaterialAlertDialogBuilder(this).setTitle(io.nekohasekai.sfa.R.string.check_update)
|
||||||
.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()
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
Loading…
Add table
Add a link
Reference in a new issue