diff --git a/app/src/main/java/io/nekohasekai/sfa/Application.kt b/app/src/main/java/io/nekohasekai/sfa/Application.kt index 298e231..51cfa5d 100644 --- a/app/src/main/java/io/nekohasekai/sfa/Application.kt +++ b/app/src/main/java/io/nekohasekai/sfa/Application.kt @@ -11,11 +11,13 @@ import android.net.wifi.WifiManager import android.os.PowerManager import androidx.core.content.getSystemService import go.Seq +import io.nekohasekai.libbox.Libbox import io.nekohasekai.sfa.bg.AppChangeReceiver import io.nekohasekai.sfa.bg.UpdateProfileWork import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.GlobalScope import kotlinx.coroutines.launch +import java.util.Locale import io.nekohasekai.sfa.Application as BoxApplication class Application : Application() { @@ -29,6 +31,7 @@ class Application : Application() { super.onCreate() Seq.setContext(this) + Libbox.setLocale(Locale.getDefault().toLanguageTag().replace("-", "_")) @Suppress("OPT_IN_USAGE") GlobalScope.launch(Dispatchers.IO) { diff --git a/app/src/main/java/io/nekohasekai/sfa/constant/EnabledType.kt b/app/src/main/java/io/nekohasekai/sfa/constant/EnabledType.kt index 750431e..1bf8527 100644 --- a/app/src/main/java/io/nekohasekai/sfa/constant/EnabledType.kt +++ b/app/src/main/java/io/nekohasekai/sfa/constant/EnabledType.kt @@ -1,11 +1,30 @@ package io.nekohasekai.sfa.constant +import android.content.Context +import io.nekohasekai.sfa.R + enum class EnabledType(val boolValue: Boolean) { Enabled(true), Disabled(false); + fun getString(context: Context): String { + return when (this) { + Enabled -> context.getString(R.string.enabled) + Disabled -> context.getString(R.string.disabled) + } + } + + companion object { fun from(value: Boolean): EnabledType { return if (value) Enabled else Disabled } + + fun valueOf(context: Context, value: String): EnabledType { + return when (value) { + context.getString(R.string.enabled) -> Enabled + context.getString(R.string.disabled) -> Disabled + else -> Disabled + } + } } } \ No newline at end of file diff --git a/app/src/main/java/io/nekohasekai/sfa/constant/PerAppProxyUpdateType.kt b/app/src/main/java/io/nekohasekai/sfa/constant/PerAppProxyUpdateType.kt index aed1f19..00e8110 100644 --- a/app/src/main/java/io/nekohasekai/sfa/constant/PerAppProxyUpdateType.kt +++ b/app/src/main/java/io/nekohasekai/sfa/constant/PerAppProxyUpdateType.kt @@ -1,5 +1,7 @@ package io.nekohasekai.sfa.constant +import android.content.Context +import io.nekohasekai.sfa.R import io.nekohasekai.sfa.database.Settings enum class PerAppProxyUpdateType { @@ -11,6 +13,14 @@ enum class PerAppProxyUpdateType { Deselect -> Settings.PER_APP_PROXY_EXCLUDE } + fun getString(context: Context): String { + return when (this) { + Disabled -> context.getString(R.string.disabled) + Select -> context.getString(R.string.action_select) + Deselect -> context.getString(R.string.action_deselect) + } + } + companion object { fun valueOf(value: Int): PerAppProxyUpdateType = when (value) { Settings.PER_APP_PROXY_DISABLED -> Disabled @@ -18,5 +28,14 @@ enum class PerAppProxyUpdateType { Settings.PER_APP_PROXY_EXCLUDE -> Deselect else -> throw IllegalArgumentException() } + + fun valueOf(context: Context, value: String): PerAppProxyUpdateType { + return when (value) { + context.getString(R.string.disabled) -> Disabled + context.getString(R.string.action_select) -> Select + context.getString(R.string.action_deselect) -> Deselect + else -> Disabled + } + } } } \ No newline at end of file diff --git a/app/src/main/java/io/nekohasekai/sfa/database/TypedProfile.kt b/app/src/main/java/io/nekohasekai/sfa/database/TypedProfile.kt index 8481238..6350b56 100644 --- a/app/src/main/java/io/nekohasekai/sfa/database/TypedProfile.kt +++ b/app/src/main/java/io/nekohasekai/sfa/database/TypedProfile.kt @@ -1,8 +1,11 @@ package io.nekohasekai.sfa.database +import android.content.Context import android.os.Parcel import android.os.Parcelable import androidx.room.TypeConverter +import io.nekohasekai.sfa.R +import io.nekohasekai.sfa.database.TypedProfile.Type.values import io.nekohasekai.sfa.ktx.marshall import io.nekohasekai.sfa.ktx.unmarshall import java.util.Date @@ -12,6 +15,13 @@ class TypedProfile() : Parcelable { enum class Type { Local, Remote; + fun getString(context: Context): String { + return when (this) { + Local -> context.getString(R.string.profile_type_local) + Remote -> context.getString(R.string.profile_type_remote) + } + } + companion object { fun valueOf(value: Int): Type { for (it in values()) { diff --git a/app/src/main/java/io/nekohasekai/sfa/ui/main/SettingsFragment.kt b/app/src/main/java/io/nekohasekai/sfa/ui/main/SettingsFragment.kt index ae6ff69..93f7d38 100644 --- a/app/src/main/java/io/nekohasekai/sfa/ui/main/SettingsFragment.kt +++ b/app/src/main/java/io/nekohasekai/sfa/ui/main/SettingsFragment.kt @@ -69,7 +69,7 @@ class SettingsFragment : Fragment() { } binding.checkUpdateEnabled.addTextChangedListener { lifecycleScope.launch(Dispatchers.IO) { - val newValue = EnabledType.valueOf(it).boolValue + val newValue = EnabledType.valueOf(requireContext(), it).boolValue Settings.checkUpdateEnabled = newValue } } @@ -81,13 +81,13 @@ class SettingsFragment : Fragment() { } binding.disableMemoryLimit.addTextChangedListener { lifecycleScope.launch(Dispatchers.IO) { - val newValue = EnabledType.valueOf(it).boolValue + val newValue = EnabledType.valueOf(requireContext(), it).boolValue Settings.disableMemoryLimit = !newValue } } binding.dynamicNotificationEnabled.addTextChangedListener { lifecycleScope.launch(Dispatchers.IO) { - val newValue = EnabledType.valueOf(it).boolValue + val newValue = EnabledType.valueOf(requireContext(), it).boolValue Settings.dynamicNotification = newValue } } @@ -136,12 +136,15 @@ class SettingsFragment : Fragment() { val dynamicNotification = Settings.dynamicNotification withContext(Dispatchers.Main) { binding.dataSizeText.text = dataSize - binding.checkUpdateEnabled.text = EnabledType.from(checkUpdateEnabled).name + binding.checkUpdateEnabled.text = + EnabledType.from(checkUpdateEnabled).getString(requireContext()) binding.checkUpdateEnabled.setSimpleItems(R.array.enabled) - binding.disableMemoryLimit.text = EnabledType.from(!Settings.disableMemoryLimit).name + binding.disableMemoryLimit.text = + EnabledType.from(!Settings.disableMemoryLimit).getString(requireContext()) binding.disableMemoryLimit.setSimpleItems(R.array.enabled) binding.backgroundPermissionCard.isGone = removeBackgroundPermissionPage - binding.dynamicNotificationEnabled.text = EnabledType.from(dynamicNotification).name + binding.dynamicNotificationEnabled.text = + EnabledType.from(dynamicNotification).getString(requireContext()) binding.dynamicNotificationEnabled.setSimpleItems(R.array.enabled) } } diff --git a/app/src/main/java/io/nekohasekai/sfa/ui/profile/EditProfileActivity.kt b/app/src/main/java/io/nekohasekai/sfa/ui/profile/EditProfileActivity.kt index 29924c8..da26bbe 100644 --- a/app/src/main/java/io/nekohasekai/sfa/ui/profile/EditProfileActivity.kt +++ b/app/src/main/java/io/nekohasekai/sfa/ui/profile/EditProfileActivity.kt @@ -67,7 +67,7 @@ class EditProfileActivity : AbstractActivity() { } } } - binding.type.text = profile.typed.type.name + binding.type.text = profile.typed.type.getString(this@EditProfileActivity) binding.editButton.setOnClickListener { startActivity( Intent( @@ -89,7 +89,8 @@ class EditProfileActivity : AbstractActivity() { binding.remoteURL.text = profile.typed.remoteURL binding.lastUpdated.text = DateFormat.getDateTimeInstance().format(profile.typed.lastUpdated) - binding.autoUpdate.text = EnabledType.from(profile.typed.autoUpdate).name + binding.autoUpdate.text = EnabledType.from(profile.typed.autoUpdate) + .getString(this@EditProfileActivity) binding.autoUpdate.setSimpleItems(R.array.enabled) binding.autoUpdateInterval.isVisible = profile.typed.autoUpdate binding.autoUpdateInterval.text = profile.typed.autoUpdateInterval.toString() @@ -111,7 +112,7 @@ class EditProfileActivity : AbstractActivity() { } private fun updateAutoUpdate(newValue: String) { - val boolValue = EnabledType.valueOf(newValue).boolValue + val boolValue = EnabledType.valueOf(this, newValue).boolValue if (profile.typed.autoUpdate == boolValue) { return } diff --git a/app/src/main/java/io/nekohasekai/sfa/ui/profile/NewProfileActivity.kt b/app/src/main/java/io/nekohasekai/sfa/ui/profile/NewProfileActivity.kt index 847924e..75672f3 100644 --- a/app/src/main/java/io/nekohasekai/sfa/ui/profile/NewProfileActivity.kt +++ b/app/src/main/java/io/nekohasekai/sfa/ui/profile/NewProfileActivity.kt @@ -1,9 +1,11 @@ package io.nekohasekai.sfa.ui.profile +import android.content.Context import android.net.Uri import android.os.Bundle import android.view.View import androidx.activity.result.contract.ActivityResultContracts +import androidx.annotation.StringRes import androidx.core.view.isVisible import androidx.lifecycle.lifecycleScope import io.nekohasekai.libbox.Libbox @@ -29,9 +31,13 @@ import java.io.InputStream import java.util.Date class NewProfileActivity : AbstractActivity() { - enum class FileSource(val formatted: String) { - CreateNew("Create New"), - Import("Import"); + enum class FileSource(@StringRes val formattedRes: Int) { + CreateNew(R.string.profile_source_create_new), + Import(R.string.profile_source_import); + + fun formatted(context: Context): String { + return context.getString(formattedRes) + } } private val importFile = @@ -49,7 +55,7 @@ class NewProfileActivity : AbstractActivity() { intent.getStringExtra("importName")?.also { importName -> intent.getStringExtra("importURL")?.also { importURL -> binding.name.editText?.setText(importName) - binding.type.text = TypedProfile.Type.Remote.name + binding.type.text = TypedProfile.Type.Remote.getString(this) binding.remoteURL.editText?.setText(importURL) binding.localFields.isVisible = false binding.remoteFields.isVisible = true @@ -60,12 +66,12 @@ class NewProfileActivity : AbstractActivity() { binding.name.removeErrorIfNotEmpty() binding.type.addTextChangedListener { when (it) { - TypedProfile.Type.Local.name -> { + TypedProfile.Type.Local.getString(this) -> { binding.localFields.isVisible = true binding.remoteFields.isVisible = false } - TypedProfile.Type.Remote.name -> { + TypedProfile.Type.Remote.getString(this) -> { binding.localFields.isVisible = false binding.remoteFields.isVisible = true if (binding.autoUpdateInterval.text.toIntOrNull() == null) { @@ -76,12 +82,12 @@ class NewProfileActivity : AbstractActivity() { } binding.fileSourceMenu.addTextChangedListener { when (it) { - FileSource.CreateNew.formatted -> { + FileSource.CreateNew.formatted(this) -> { binding.importFileButton.isVisible = false binding.sourceURL.isVisible = false } - FileSource.Import.formatted -> { + FileSource.Import.formatted(this) -> { binding.importFileButton.isVisible = true binding.sourceURL.isVisible = true } @@ -99,9 +105,9 @@ class NewProfileActivity : AbstractActivity() { return } when (binding.type.text) { - TypedProfile.Type.Local.name -> { + TypedProfile.Type.Local.getString(this) -> { when (binding.fileSourceMenu.text) { - FileSource.Import.formatted -> { + FileSource.Import.formatted(this) -> { if (binding.sourceURL.showErrorIfEmpty()) { return } @@ -109,7 +115,7 @@ class NewProfileActivity : AbstractActivity() { } } - TypedProfile.Type.Remote.name -> { + TypedProfile.Type.Remote.getString(this) -> { if (binding.remoteURL.showErrorIfEmpty()) { return } @@ -138,15 +144,15 @@ class NewProfileActivity : AbstractActivity() { typedProfile.path = configFile.path when (binding.type.text) { - TypedProfile.Type.Local.name -> { + TypedProfile.Type.Local.getString(this) -> { typedProfile.type = TypedProfile.Type.Local when (binding.fileSourceMenu.text) { - FileSource.CreateNew.formatted -> { + FileSource.CreateNew.formatted(this) -> { configFile.writeText("{}") } - FileSource.Import.formatted -> { + FileSource.Import.formatted(this) -> { val sourceURL = binding.sourceURL.text val content = if (sourceURL.startsWith("content://")) { val inputStream = @@ -165,7 +171,7 @@ class NewProfileActivity : AbstractActivity() { } } - TypedProfile.Type.Remote.name -> { + TypedProfile.Type.Remote.getString(this) -> { typedProfile.type = TypedProfile.Type.Remote val remoteURL = binding.remoteURL.text val content = HTTPClient().use { it.getString(remoteURL) } @@ -173,7 +179,8 @@ class NewProfileActivity : AbstractActivity() { configFile.writeText(content) typedProfile.remoteURL = remoteURL typedProfile.lastUpdated = Date() - typedProfile.autoUpdate = EnabledType.valueOf(binding.autoUpdate.text).boolValue + typedProfile.autoUpdate = + EnabledType.valueOf(this, binding.autoUpdate.text).boolValue binding.autoUpdateInterval.text.toIntOrNull()?.also { typedProfile.autoUpdateInterval = it } diff --git a/app/src/main/java/io/nekohasekai/sfa/ui/profileoverride/PerAppProxyActivity.kt b/app/src/main/java/io/nekohasekai/sfa/ui/profileoverride/PerAppProxyActivity.kt index 990d304..ea5cd09 100644 --- a/app/src/main/java/io/nekohasekai/sfa/ui/profileoverride/PerAppProxyActivity.kt +++ b/app/src/main/java/io/nekohasekai/sfa/ui/profileoverride/PerAppProxyActivity.kt @@ -304,7 +304,7 @@ class PerAppProxyActivity : AbstractActivity() { } override fun onCreateOptionsMenu(menu: Menu?): Boolean { - menuInflater.inflate(R.menu.per_app_menu0, menu) + menuInflater.inflate(R.menu.per_app_menu, menu) if (menu != null) { val searchView = menu.findItem(R.id.action_search).actionView as SearchView diff --git a/app/src/main/java/io/nekohasekai/sfa/ui/profileoverride/ProfileOverrideActivity.kt b/app/src/main/java/io/nekohasekai/sfa/ui/profileoverride/ProfileOverrideActivity.kt index 8c8d1c4..4be88f2 100644 --- a/app/src/main/java/io/nekohasekai/sfa/ui/profileoverride/ProfileOverrideActivity.kt +++ b/app/src/main/java/io/nekohasekai/sfa/ui/profileoverride/ProfileOverrideActivity.kt @@ -33,7 +33,8 @@ class ProfileOverrideActivity : binding.perAppProxyUpdateOnChange.addTextChangedListener { lifecycleScope.launch(Dispatchers.IO) { - Settings.perAppProxyUpdateOnChange = PerAppProxyUpdateType.valueOf(it).value() + Settings.perAppProxyUpdateOnChange = + PerAppProxyUpdateType.valueOf(this@ProfileOverrideActivity, it).value() } } @@ -49,7 +50,8 @@ class ProfileOverrideActivity : val perAppUpdateOnChange = Settings.perAppProxyUpdateOnChange withContext(Dispatchers.Main) { binding.perAppProxyUpdateOnChange.text = - PerAppProxyUpdateType.valueOf(perAppUpdateOnChange).name + PerAppProxyUpdateType.valueOf(perAppUpdateOnChange) + .getString(this@ProfileOverrideActivity) binding.perAppProxyUpdateOnChange.setSimpleItems(R.array.per_app_proxy_update_on_change_value) } } diff --git a/app/src/main/java/io/nekohasekai/sfa/utils/HTTPClient.kt b/app/src/main/java/io/nekohasekai/sfa/utils/HTTPClient.kt index baab03f..64f785b 100644 --- a/app/src/main/java/io/nekohasekai/sfa/utils/HTTPClient.kt +++ b/app/src/main/java/io/nekohasekai/sfa/utils/HTTPClient.kt @@ -4,6 +4,7 @@ import io.nekohasekai.libbox.Libbox import io.nekohasekai.sfa.BuildConfig import io.nekohasekai.sfa.ktx.unwrap import java.io.Closeable +import java.util.Locale class HTTPClient : Closeable { @@ -15,6 +16,8 @@ class HTTPClient : Closeable { userAgent += BuildConfig.VERSION_CODE userAgent += "; sing-box " userAgent += Libbox.version() + userAgent += "; language " + userAgent += Locale.getDefault().toLanguageTag().replace("-", "_") userAgent += ")" userAgent } diff --git a/app/src/main/res/layout/fragment_dashboard_overview.xml b/app/src/main/res/layout/fragment_dashboard_overview.xml index 0f94137..c478a16 100644 --- a/app/src/main/res/layout/fragment_dashboard_overview.xml +++ b/app/src/main/res/layout/fragment_dashboard_overview.xml @@ -32,18 +32,18 @@ android:id="@+id/statusContainer" android:layout_width="match_parent" android:layout_height="wrap_content" - android:orientation="vertical" - android:visibility="gone" android:clipChildren="false" android:clipToPadding="false" + android:orientation="vertical" + android:visibility="gone" tools:visibility="visible"> @@ -150,7 +151,9 @@ @@ -251,6 +254,7 @@ @@ -341,6 +345,7 @@ diff --git a/app/src/main/res/menu/per_app_menu.xml b/app/src/main/res/menu/per_app_menu.xml index 7ade0dc..96d1468 100644 --- a/app/src/main/res/menu/per_app_menu.xml +++ b/app/src/main/res/menu/per_app_menu.xml @@ -10,20 +10,90 @@ app:iconTint="?colorControlNormal" app:showAsAction="collapseActionView|always" /> - + + + + + + + + - + + + + + + + + + + + + - + + + + + + + - + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/menu/per_app_menu0.xml b/app/src/main/res/menu/per_app_menu0.xml deleted file mode 100644 index 96d1468..0000000 --- a/app/src/main/res/menu/per_app_menu0.xml +++ /dev/null @@ -1,99 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/app/src/main/res/values-zh-rCN/strings.xml b/app/src/main/res/values-zh-rCN/strings.xml index 1098170..76294e3 100644 --- a/app/src/main/res/values-zh-rCN/strings.xml +++ b/app/src/main/res/values-zh-rCN/strings.xml @@ -1,3 +1,164 @@ + 停止 + 不,谢谢 + 仪表 + 配置 + 日志 + 设置 + 新建配置 + 编辑配置 + 编辑配置 + 概述 + + 调试 + 切换 + 名称 + 类型 + + 导入文件 + 创建 + 编辑内容 + 检查 + 分享 + 通过二维码分享 URL + 必须 + 无配置 + 最后更新:%s + 最后更新 + 更新 + 自动更新 + 自动更新间隔 (分) + 最低值为 15 + 本地 + 远程 + 创建 + 导入 + 从文件导入 + 扫描二维码 + 前置摄像头 + 使用 MLKit 扫描 + 灯光 + 手动创建 + 撤销 + 重做 + 格式化 + 删除 + 分享 + 服务未启动 + 服务启动中... + 服务停止中... + 服务已启动 + 启用 + 禁用 + 清理工作目录 + 错误 + 您的设备缺少 Android 标准文件选择器,请安装一个,例如 Material Files。 + 加载中... + 缺少 VPN 权限 + 缺少通知权限 + 空配置 + 启动命令服务器 + 创建服务 + 启动服务 + 弃用警告 + 文档 + 状态 + 内存 + 连接 + 入站 + 出站 + 上传 + 下载 + 速率 + 流量 + 选中 + 配置 + 版本 + 核心 + 在通知中显示实时速度 + 数据大小 + 自动检查更新 + 检查更新 + 隐私政策 + 应用 + 内存限制 + 后台权限 + 申请必要的权限以使 VPN 正常运行。 如果您使用的是中国公司生产的设备,则授予权限后该卡可能不会消失。 + 阅读更多 + 忽略电池优化 + 导入远程配置 + 您确定要导入远程配置文件 %1$s 吗?您将连接到 %2$s 来下载配置。 + 配置覆盖 + 使用平台特定的值覆盖文件配置项。 + 配置 + 分应用代理 + 覆盖配置中的 include_package 和 exclude_package。 + 代理模式 + 白名单 + 仅允许选定的应用程序通过 VPN + 黑名单 + 选定的应用将从 VPN 中排除 + 复制 + 名称 + 包名 + UID + 排序 + 名称 + 包名 + UID + 安装时间 + 更新shijian + 反转 + 过滤 + 隐藏系统应用 + 隐藏离线应用 + 隐藏禁用应用 + 选择 + 全选 + 全不选 + 备份 + 从剪切板导入 + 导出到剪切板 + 扫描 + 中国应用 + App 图标 + 剪切板为空 + 应用列表为空 + 已导出到剪切板 + 已从剪贴板导入 + 从剪贴板导入应用列表将覆盖当前列表。您确定要继续吗? + 扫描中... + 扫描应用程序时出错 + 未找到匹配的应用 + 找到以下应用程序,请选择您想要的操作。 + 扫描结果 + 选择 + 取消选择 + 新中国应用安装时更新 + 导入配置 + 您确定要导入配置文件 %s 吗? + 当前平台不支持 iCloud 配置文件 + 搜索 + 展开 + HTTP 代理 + 扫描 VPN 应用 + 检查设备上安装的 VPN 及其内容 + 扫描 + 一些调试工具 + 打开 + 应用类型 + 核心类型 + 发现路径 + Go 版本 + 其他 + 未知 + 没有可用更新 + 赞助 + 支持我的工作 + 启动 + 位置权限 + wifi_ssidwifi_bssid 路由规则。为了使它们正常工作,sing-box 在后台使用 位置 权限来获取有关所连接 Wi-Fi 网络的信息。该信息将仅用于路由目的。]]> + 后台位置权限。选择始终允许以授予权限。]]> + 打开设置 \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 0172808..978c460 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -1,5 +1,5 @@ - sing-box + sing-box Stop OK @@ -20,7 +20,7 @@ Name Type Source - URL + URL Import File Create Edit Content @@ -69,8 +69,8 @@ Your device lacks an Android standard file selector, please install one, such as Material Files. Loading… - Failed to request VPN permission - Failed to request notification permission + Missiong VPN permission + Missing notification permission Empty configuration Start command server Create service @@ -80,7 +80,7 @@ Status Memory - Goroutines + Goroutines Connections Inbound Outbound @@ -124,7 +124,7 @@ Package Name UID - Sort Mode + Sort By By name By package name By UID @@ -149,9 +149,6 @@ China apps App icon - Hide system apps - Show system apps - Scan China apps Clipboard is empty App list is empty Exported to clipboard @@ -169,7 +166,7 @@ Are you sure to import profile %s? iCloud profile is not support on current platform Search - URLTest + URLTest Expand HTTP Proxy Scan VPN apps @@ -185,9 +182,7 @@ Unknown No updates available Sponsor - Sponsor via Play Store If I\'ve defended your modern life, please consider sponsoring me. - Other methods Start Location permission wifi_ssid or wifi_bssid routing rules. To make them work, sing-box uses the location permission in the background to get information about the connected Wi-Fi network. The information will be used for routing purposes only.]]>