mirror of
https://github.com/SagerNet/sing-box-for-android.git
synced 2025-04-05 12:57:38 +03:00
Add dynamic notification
Co-authored-by: Moe <moe@example.com> Co-authored-by: 世界 <i@sekai.icu>
This commit is contained in:
parent
71acd29526
commit
4295c30503
8 changed files with 122 additions and 28 deletions
|
@ -45,6 +45,7 @@ class Application : Application() {
|
||||||
val connectivity by lazy { application.getSystemService<ConnectivityManager>()!! }
|
val connectivity by lazy { application.getSystemService<ConnectivityManager>()!! }
|
||||||
val packageManager by lazy { application.packageManager }
|
val packageManager by lazy { application.packageManager }
|
||||||
val powerManager by lazy { application.getSystemService<PowerManager>()!! }
|
val powerManager by lazy { application.getSystemService<PowerManager>()!! }
|
||||||
|
val notificationManager by lazy { application.getSystemService<NotificationManager>()!! }
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -86,7 +86,7 @@ class BoxService(
|
||||||
|
|
||||||
private val status = MutableLiveData(Status.Stopped)
|
private val status = MutableLiveData(Status.Stopped)
|
||||||
private val binder = ServiceBinder(status)
|
private val binder = ServiceBinder(status)
|
||||||
private val notification = ServiceNotification(service)
|
private val notification = ServiceNotification(status, service)
|
||||||
private var boxService: BoxService? = null
|
private var boxService: BoxService? = null
|
||||||
private var commandServer: CommandServer? = null
|
private var commandServer: CommandServer? = null
|
||||||
private var pprofServer: PProfServer? = null
|
private var pprofServer: PProfServer? = null
|
||||||
|
@ -118,6 +118,7 @@ class BoxService(
|
||||||
this.commandServer = commandServer
|
this.commandServer = commandServer
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private var lastProfileName = ""
|
||||||
private suspend fun startService(delayStart: Boolean = false) {
|
private suspend fun startService(delayStart: Boolean = false) {
|
||||||
try {
|
try {
|
||||||
val selectedProfileId = Settings.selectedProfile
|
val selectedProfileId = Settings.selectedProfile
|
||||||
|
@ -138,6 +139,7 @@ class BoxService(
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
lastProfileName = profile.name
|
||||||
withContext(Dispatchers.Main) {
|
withContext(Dispatchers.Main) {
|
||||||
binder.broadcast {
|
binder.broadcast {
|
||||||
it.onServiceResetLogs(listOf())
|
it.onServiceResetLogs(listOf())
|
||||||
|
@ -163,6 +165,10 @@ class BoxService(
|
||||||
boxService = newService
|
boxService = newService
|
||||||
commandServer?.setService(boxService)
|
commandServer?.setService(boxService)
|
||||||
status.postValue(Status.Started)
|
status.postValue(Status.Started)
|
||||||
|
|
||||||
|
withContext(Dispatchers.Main) {
|
||||||
|
notification.show(lastProfileName)
|
||||||
|
}
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
stopAndAlert(Alert.StartService, e.message)
|
stopAndAlert(Alert.StartService, e.message)
|
||||||
return
|
return
|
||||||
|
@ -170,8 +176,8 @@ class BoxService(
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun serviceReload() {
|
override fun serviceReload() {
|
||||||
|
notification.close()
|
||||||
status.postValue(Status.Starting)
|
status.postValue(Status.Starting)
|
||||||
runBlocking {
|
|
||||||
val pfd = fileDescriptor
|
val pfd = fileDescriptor
|
||||||
if (pfd != null) {
|
if (pfd != null) {
|
||||||
pfd.close()
|
pfd.close()
|
||||||
|
@ -187,6 +193,7 @@ class BoxService(
|
||||||
Seq.destroyRef(refnum)
|
Seq.destroyRef(refnum)
|
||||||
}
|
}
|
||||||
boxService = null
|
boxService = null
|
||||||
|
runBlocking {
|
||||||
startService(true)
|
startService(true)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -283,7 +290,6 @@ class BoxService(
|
||||||
receiverRegistered = true
|
receiverRegistered = true
|
||||||
}
|
}
|
||||||
|
|
||||||
notification.show()
|
|
||||||
GlobalScope.launch(Dispatchers.IO) {
|
GlobalScope.launch(Dispatchers.IO) {
|
||||||
Settings.startedByUser = true
|
Settings.startedByUser = true
|
||||||
initialize()
|
initialize()
|
||||||
|
|
|
@ -4,16 +4,30 @@ import android.app.NotificationChannel
|
||||||
import android.app.NotificationManager
|
import android.app.NotificationManager
|
||||||
import android.app.PendingIntent
|
import android.app.PendingIntent
|
||||||
import android.app.Service
|
import android.app.Service
|
||||||
|
import android.content.BroadcastReceiver
|
||||||
|
import android.content.Context
|
||||||
import android.content.Intent
|
import android.content.Intent
|
||||||
|
import android.content.IntentFilter
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
import androidx.core.app.NotificationCompat
|
import androidx.core.app.NotificationCompat
|
||||||
import androidx.core.app.ServiceCompat
|
import androidx.core.app.ServiceCompat
|
||||||
|
import androidx.lifecycle.MutableLiveData
|
||||||
|
import io.nekohasekai.libbox.Libbox
|
||||||
|
import io.nekohasekai.libbox.StatusMessage
|
||||||
import io.nekohasekai.sfa.Application
|
import io.nekohasekai.sfa.Application
|
||||||
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.Status
|
||||||
|
import io.nekohasekai.sfa.database.Settings
|
||||||
import io.nekohasekai.sfa.ui.MainActivity
|
import io.nekohasekai.sfa.ui.MainActivity
|
||||||
|
import io.nekohasekai.sfa.utils.CommandClient
|
||||||
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
import kotlinx.coroutines.GlobalScope
|
||||||
|
import kotlinx.coroutines.withContext
|
||||||
|
|
||||||
class ServiceNotification(private val service: Service) {
|
class ServiceNotification(
|
||||||
|
private val status: MutableLiveData<Status>, private val service: Service
|
||||||
|
) : BroadcastReceiver(), CommandClient.Handler {
|
||||||
companion object {
|
companion object {
|
||||||
private const val notificationId = 1
|
private const val notificationId = 1
|
||||||
private const val notificationChannel = "service"
|
private const val notificationChannel = "service"
|
||||||
|
@ -28,11 +42,12 @@ class ServiceNotification(private val service: Service) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private val commandClient =
|
||||||
|
CommandClient(GlobalScope, CommandClient.ConnectionType.Status, this)
|
||||||
|
|
||||||
private val notification by lazy {
|
private val notificationBuilder by lazy {
|
||||||
NotificationCompat.Builder(service, notificationChannel).setWhen(0)
|
NotificationCompat.Builder(service, notificationChannel).setShowWhen(false).setOngoing(true)
|
||||||
.setContentTitle("sing-box")
|
.setContentTitle("sing-box").setOnlyAlertOnce(true)
|
||||||
.setContentText("service started").setOnlyAlertOnce(true)
|
|
||||||
.setSmallIcon(R.drawable.ic_menu)
|
.setSmallIcon(R.drawable.ic_menu)
|
||||||
.setCategory(NotificationCompat.CATEGORY_SERVICE)
|
.setCategory(NotificationCompat.CATEGORY_SERVICE)
|
||||||
.setContentIntent(
|
.setContentIntent(
|
||||||
|
@ -60,7 +75,7 @@ class ServiceNotification(private val service: Service) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun show() {
|
suspend fun show(lastProfileName: String) {
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
||||||
Application.notification.createNotificationChannel(
|
Application.notification.createNotificationChannel(
|
||||||
NotificationChannel(
|
NotificationChannel(
|
||||||
|
@ -68,10 +83,52 @@ class ServiceNotification(private val service: Service) {
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
service.startForeground(notificationId, notification.build())
|
service.startForeground(
|
||||||
|
notificationId, notificationBuilder
|
||||||
|
.setContentTitle(lastProfileName.takeIf { it.isNotBlank() } ?: "sing-box")
|
||||||
|
.setContentText("service started").build()
|
||||||
|
)
|
||||||
|
withContext(Dispatchers.IO) {
|
||||||
|
if (Settings.dynamicNotification) {
|
||||||
|
commandClient.connect()
|
||||||
|
withContext(Dispatchers.Main) {
|
||||||
|
registerReceiver()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun registerReceiver() {
|
||||||
|
service.registerReceiver(this, IntentFilter().apply {
|
||||||
|
addAction(Intent.ACTION_SCREEN_ON)
|
||||||
|
addAction(Intent.ACTION_SCREEN_OFF)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun updateStatus(status: StatusMessage) {
|
||||||
|
val content =
|
||||||
|
Libbox.formatBytes(status.uplink) + "/s ↑\t" + Libbox.formatBytes(status.downlink) + "/s ↓"
|
||||||
|
Application.notificationManager.notify(
|
||||||
|
notificationId,
|
||||||
|
notificationBuilder.setContentText(content).build()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onReceive(context: Context, intent: Intent) {
|
||||||
|
when (intent.action) {
|
||||||
|
Intent.ACTION_SCREEN_ON -> {
|
||||||
|
commandClient.connect()
|
||||||
|
}
|
||||||
|
|
||||||
|
Intent.ACTION_SCREEN_OFF -> {
|
||||||
|
commandClient.disconnect()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun close() {
|
fun close() {
|
||||||
|
commandClient.disconnect()
|
||||||
ServiceCompat.stopForeground(service, ServiceCompat.STOP_FOREGROUND_REMOVE)
|
ServiceCompat.stopForeground(service, ServiceCompat.STOP_FOREGROUND_REMOVE)
|
||||||
|
service.unregisterReceiver(this)
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -6,6 +6,7 @@ object SettingsKey {
|
||||||
const val SERVICE_MODE = "service_mode"
|
const val SERVICE_MODE = "service_mode"
|
||||||
const val CHECK_UPDATE_ENABLED = "check_update_enabled"
|
const val CHECK_UPDATE_ENABLED = "check_update_enabled"
|
||||||
const val DISABLE_MEMORY_LIMIT = "disable_memory_limit"
|
const val DISABLE_MEMORY_LIMIT = "disable_memory_limit"
|
||||||
|
const val DYNAMIC_NOTIFICATION = "dynamic_notification"
|
||||||
|
|
||||||
const val PER_APP_PROXY_ENABLED = "per_app_proxy_enabled"
|
const val PER_APP_PROXY_ENABLED = "per_app_proxy_enabled"
|
||||||
const val PER_APP_PROXY_MODE = "per_app_proxy_mode"
|
const val PER_APP_PROXY_MODE = "per_app_proxy_mode"
|
||||||
|
|
|
@ -40,6 +40,7 @@ object Settings {
|
||||||
|
|
||||||
var checkUpdateEnabled by dataStore.boolean(SettingsKey.CHECK_UPDATE_ENABLED) { true }
|
var checkUpdateEnabled by dataStore.boolean(SettingsKey.CHECK_UPDATE_ENABLED) { true }
|
||||||
var disableMemoryLimit by dataStore.boolean(SettingsKey.DISABLE_MEMORY_LIMIT)
|
var disableMemoryLimit by dataStore.boolean(SettingsKey.DISABLE_MEMORY_LIMIT)
|
||||||
|
var dynamicNotification by dataStore.boolean(SettingsKey.DYNAMIC_NOTIFICATION) { true }
|
||||||
|
|
||||||
|
|
||||||
const val PER_APP_PROXY_DISABLED = 0
|
const val PER_APP_PROXY_DISABLED = 0
|
||||||
|
|
|
@ -61,7 +61,8 @@ class SettingsFragment : Fragment() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!Vendor.checkUpdateAvailable()) {
|
if (!Vendor.checkUpdateAvailable()) {
|
||||||
binding.appSettingsCard.isVisible = false
|
binding.checkUpdateEnabled.isVisible = false
|
||||||
|
binding.checkUpdateButton.isVisible = false
|
||||||
}
|
}
|
||||||
binding.checkUpdateEnabled.addTextChangedListener {
|
binding.checkUpdateEnabled.addTextChangedListener {
|
||||||
lifecycleScope.launch(Dispatchers.IO) {
|
lifecycleScope.launch(Dispatchers.IO) {
|
||||||
|
@ -78,6 +79,13 @@ class SettingsFragment : Fragment() {
|
||||||
Settings.disableMemoryLimit = !newValue
|
Settings.disableMemoryLimit = !newValue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
binding.dynamicNotificationEnabled.addTextChangedListener {
|
||||||
|
lifecycleScope.launch(Dispatchers.IO) {
|
||||||
|
val newValue = EnabledType.valueOf(it).boolValue
|
||||||
|
Settings.dynamicNotification = newValue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
binding.dontKillMyAppButton.setOnClickListener {
|
binding.dontKillMyAppButton.setOnClickListener {
|
||||||
it.context.launchCustomTab("https://dontkillmyapp.com/")
|
it.context.launchCustomTab("https://dontkillmyapp.com/")
|
||||||
}
|
}
|
||||||
|
@ -119,6 +127,7 @@ class SettingsFragment : Fragment() {
|
||||||
} else {
|
} else {
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
val dynamicNotification = Settings.dynamicNotification
|
||||||
withContext(Dispatchers.Main) {
|
withContext(Dispatchers.Main) {
|
||||||
binding.dataSizeText.text = dataSize
|
binding.dataSizeText.text = dataSize
|
||||||
binding.checkUpdateEnabled.text = EnabledType.from(checkUpdateEnabled).name
|
binding.checkUpdateEnabled.text = EnabledType.from(checkUpdateEnabled).name
|
||||||
|
@ -126,6 +135,8 @@ class SettingsFragment : Fragment() {
|
||||||
binding.disableMemoryLimit.text = EnabledType.from(!Settings.disableMemoryLimit).name
|
binding.disableMemoryLimit.text = EnabledType.from(!Settings.disableMemoryLimit).name
|
||||||
binding.disableMemoryLimit.setSimpleItems(R.array.enabled)
|
binding.disableMemoryLimit.setSimpleItems(R.array.enabled)
|
||||||
binding.backgroundPermissionCard.isGone = removeBackgroundPermissionPage
|
binding.backgroundPermissionCard.isGone = removeBackgroundPermissionPage
|
||||||
|
binding.dynamicNotificationEnabled.text = EnabledType.from(dynamicNotification).name
|
||||||
|
binding.dynamicNotificationEnabled.setSimpleItems(R.array.enabled)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -16,7 +16,6 @@
|
||||||
|
|
||||||
|
|
||||||
<com.google.android.material.card.MaterialCardView
|
<com.google.android.material.card.MaterialCardView
|
||||||
android:id="@+id/appSettingsCard"
|
|
||||||
style="?attr/materialCardViewElevatedStyle"
|
style="?attr/materialCardViewElevatedStyle"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
|
@ -38,11 +37,28 @@
|
||||||
android:textAppearance="?attr/textAppearanceTitleLarge" />
|
android:textAppearance="?attr/textAppearanceTitleLarge" />
|
||||||
|
|
||||||
<com.google.android.material.textfield.TextInputLayout
|
<com.google.android.material.textfield.TextInputLayout
|
||||||
android:id="@+id/checkUpdateEnabled"
|
android:id="@+id/dynamicNotificationEnabled"
|
||||||
style="@style/Widget.Material3.TextInputLayout.FilledBox.ExposedDropdownMenu"
|
style="@style/Widget.Material3.TextInputLayout.FilledBox.ExposedDropdownMenu"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="wrap_content"
|
android:layout_height="wrap_content"
|
||||||
android:layout_marginTop="8dp"
|
android:layout_marginTop="8dp"
|
||||||
|
android:layout_marginBottom="8dp"
|
||||||
|
android:hint="@string/dynamic_notification">
|
||||||
|
|
||||||
|
<AutoCompleteTextView
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:inputType="none"
|
||||||
|
android:text="@string/disabled"
|
||||||
|
app:simpleItems="@array/enabled" />
|
||||||
|
|
||||||
|
</com.google.android.material.textfield.TextInputLayout>
|
||||||
|
|
||||||
|
<com.google.android.material.textfield.TextInputLayout
|
||||||
|
android:id="@+id/checkUpdateEnabled"
|
||||||
|
style="@style/Widget.Material3.TextInputLayout.FilledBox.ExposedDropdownMenu"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
android:hint="@string/check_update_atomic">
|
android:hint="@string/check_update_atomic">
|
||||||
|
|
||||||
<AutoCompleteTextView
|
<AutoCompleteTextView
|
||||||
|
|
|
@ -81,10 +81,11 @@
|
||||||
<string name="profile">Profile</string>
|
<string name="profile">Profile</string>
|
||||||
<string name="core_version">Version</string>
|
<string name="core_version">Version</string>
|
||||||
<string name="core">Core</string>
|
<string name="core">Core</string>
|
||||||
|
<string name="dynamic_notification">Display realtime speed in notification</string>
|
||||||
<string name="core_data_size">Data Size</string>
|
<string name="core_data_size">Data Size</string>
|
||||||
<string name="check_update_atomic">Atomic Check Update</string>
|
<string name="check_update_atomic">Atomic Check Update</string>
|
||||||
<string name="check_update">Check Update</string>
|
<string name="check_update">Check Update</string>
|
||||||
<string name="title_app_settings">App Settings</string>
|
<string name="title_app_settings">App</string>
|
||||||
<string name="about_title">About</string>
|
<string name="about_title">About</string>
|
||||||
<string name="app_description">Android client for sing-box, the universal proxy platform.</string>
|
<string name="app_description">Android client for sing-box, the universal proxy platform.</string>
|
||||||
<string name="documentation_button">Documentation</string>
|
<string name="documentation_button">Documentation</string>
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue