mirror of
https://github.com/SagerNet/sing-box-for-android.git
synced 2025-04-04 20:37:40 +03:00
Remove in-app sponsor
This commit is contained in:
parent
557ed643e0
commit
a35318ded2
8 changed files with 3 additions and 201 deletions
|
@ -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") != "") {
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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)
|
|
||||||
}
|
}
|
|
@ -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()
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -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()
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -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
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue