request optimize

pull/731/head
Li ZongYing 2 years ago
parent 188431a29f
commit 7057e4a8b4
  1. 5
      README.md
  2. 84
      app/src/main/java/com/lizongying/mytv/Ext.kt
  3. 86
      app/src/main/java/com/lizongying/mytv/MainActivity.kt
  4. 75
      app/src/main/java/com/lizongying/mytv/Request.kt
  5. 39
      app/src/main/java/com/lizongying/mytv/SettingFragment.kt
  6. 14
      app/src/main/java/com/lizongying/mytv/UpdateManager.kt
  7. 7
      app/src/main/java/com/lizongying/mytv/models/TVViewModel.kt
  8. 2
      app/src/main/res/raw/channels.json

@ -20,6 +20,11 @@
* 凤凰卫视增强画质 * 凤凰卫视增强画质
* 凤凰卫视增加EPG * 凤凰卫视增加EPG
### v1.6.5(安卓5及以上专用)
* 增加CETV1图标
* 稳定性提升
### v1.6.4(通用) ### v1.6.4(通用)
* 增加CETV1 * 增加CETV1

@ -0,0 +1,84 @@
@file:Suppress("DEPRECATION")
package com.lizongying.mytv
import android.content.Context
import android.content.pm.PackageInfo
import android.content.pm.PackageManager
import android.content.pm.Signature
import android.content.pm.SigningInfo
import android.os.Build
import android.util.Log
import java.security.MessageDigest
private const val TAG = "Extensions"
private val Context.packageInfo: PackageInfo
get() {
val flag = if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P) {
PackageManager.GET_SIGNATURES
} else {
PackageManager.GET_SIGNING_CERTIFICATES
}
return if (Build.VERSION.SDK_INT < Build.VERSION_CODES.TIRAMISU) {
packageManager.getPackageInfo(packageName, flag)
} else {
packageManager.getPackageInfo(
packageName,
PackageManager.PackageInfoFlags.of(PackageManager.GET_SIGNING_CERTIFICATES.toLong())
)
}
}
/**
* Return the version code of the app which is defined in build.gradle.
* eg:100
*/
val Context.appVersionCode: Long
get() {
val packageInfo = this.packageInfo
return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
packageInfo.longVersionCode
} else {
packageInfo.versionCode.toLong()
}
}
/**
* Return the version name of the app which is defined in build.gradle.
* eg:1.0.0
*/
val Context.appVersionName: String get() = packageInfo.versionName
val Context.appSignature: String
get() {
val packageInfo = this.packageInfo
var sign: Signature? = null
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P) {
val signatures: Array<out Signature>? = packageInfo.signatures
if (signatures != null) {
sign = signatures[0]
}
} else {
val signingInfo: SigningInfo? = packageInfo.signingInfo
if (signingInfo != null) {
sign = signingInfo.apkContentsSigners[0]
}
}
if (sign == null) {
return ""
}
return hashSignature(sign)
}
private fun hashSignature(signature: Signature): String {
return try {
val md = MessageDigest.getInstance("MD5")
md.update(signature.toByteArray())
val digest = md.digest()
digest.let { it -> it.joinToString("") { "%02x".format(it) } }
} catch (e: Exception) {
Log.e(TAG, "Error hashing signature", e)
""
}
}

@ -1,10 +1,5 @@
package com.lizongying.mytv package com.lizongying.mytv
import android.content.pm.PackageInfo
import android.content.pm.PackageManager
import android.content.pm.Signature
import android.content.pm.SigningInfo
import android.os.Build
import android.os.Bundle import android.os.Bundle
import android.os.Handler import android.os.Handler
import android.os.Looper import android.os.Looper
@ -16,23 +11,17 @@ import android.view.View.SYSTEM_UI_FLAG_HIDE_NAVIGATION
import android.view.WindowManager import android.view.WindowManager
import android.widget.Toast import android.widget.Toast
import androidx.fragment.app.FragmentActivity import androidx.fragment.app.FragmentActivity
import androidx.lifecycle.lifecycleScope
import com.lizongying.mytv.models.TVViewModel import com.lizongying.mytv.models.TVViewModel
import kotlinx.coroutines.CoroutineStart
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.async
import kotlinx.coroutines.launch
import java.security.MessageDigest
class MainActivity : FragmentActivity(), Request.RequestListener { class MainActivity : FragmentActivity(), Request.RequestListener {
private var ready = 0 private var ready = 0
private var playerFragment = PlayerFragment() private val playerFragment = PlayerFragment()
private var mainFragment = MainFragment() private val mainFragment = MainFragment()
private var infoFragment = InfoFragment() private val infoFragment = InfoFragment()
private var channelFragment = ChannelFragment() private val channelFragment = ChannelFragment()
private lateinit var settingFragment: SettingFragment private val settingFragment = SettingFragment()
private var doubleBackToExitPressedOnce = false private var doubleBackToExitPressedOnce = false
@ -64,20 +53,6 @@ class MainActivity : FragmentActivity(), Request.RequestListener {
.commit() .commit()
} }
gestureDetector = GestureDetector(this, GestureListener()) gestureDetector = GestureDetector(this, GestureListener())
val packageInfo = getPackageInfo()
val versionName = packageInfo.versionName
val versionCode = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
packageInfo.longVersionCode
} else {
packageInfo.versionCode.toLong()
}
settingFragment = SettingFragment(
versionName,
versionCode,
SP.channelReversal,
SP.channelNum,
SP.bootStartup
)
} }
fun showInfoFragment(tvViewModel: TVViewModel) { fun showInfoFragment(tvViewModel: TVViewModel) {
@ -448,56 +423,7 @@ class MainActivity : FragmentActivity(), Request.RequestListener {
return super.onKeyDown(keyCode, event) return super.onKeyDown(keyCode, event)
} }
private fun getPackageInfo(): PackageInfo { private fun getAppSignature() = this.appSignature
val flag = if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P) {
PackageManager.GET_SIGNATURES
} else {
PackageManager.GET_SIGNING_CERTIFICATES
}
return if (Build.VERSION.SDK_INT < Build.VERSION_CODES.TIRAMISU) {
packageManager.getPackageInfo(packageName, flag)
} else {
packageManager.getPackageInfo(
packageName,
PackageManager.PackageInfoFlags.of(PackageManager.GET_SIGNING_CERTIFICATES.toLong())
)
}
}
private fun getAppSignature(): String {
val packageInfo = getPackageInfo()
var sign: Signature? = null
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P) {
val signatures: Array<out Signature>? = packageInfo.signatures
if (signatures != null) {
sign = signatures[0]
}
} else {
val signingInfo: SigningInfo? = packageInfo.signingInfo
if (signingInfo != null) {
sign = signingInfo.apkContentsSigners[0]
}
}
if (sign == null) {
return ""
}
return hashSignature(sign)
}
private fun hashSignature(signature: Signature): String {
return try {
val md = MessageDigest.getInstance("MD5")
md.update(signature.toByteArray())
val digest = md.digest()
digest.let { it -> it.joinToString("") { "%02x".format(it) } }
} catch (e: Exception) {
Log.e(TAG, "Error hashing signature", e)
""
}
}
override fun onStart() { override fun onStart() {
Log.i(TAG, "onStart") Log.i(TAG, "onStart")

@ -56,6 +56,9 @@ object Request {
private var listener: RequestListener? = null private var listener: RequestListener? = null
private var initRetryTimes = 0
private var initRetryMaxTimes = 1
fun onCreate() { fun onCreate() {
Log.i(TAG, "onCreate") Log.i(TAG, "onCreate")
fetchInfoV2() fetchInfoV2()
@ -66,11 +69,22 @@ object Request {
handler.removeCallbacks(tokenRunnable) handler.removeCallbacks(tokenRunnable)
} }
var call: Call<LiveInfo>? = null private var call: Call<LiveInfo>? = null
private var callAuth: Call<Auth>? = null private var callAuth: Call<Auth>? = null
private var callInfo: Call<Info>? = null
private var fAuth: Call<FAuth>? = null
private var callPage: Call<pageModel.Response>? = null
private fun fetchAuth(tvModel: TVViewModel, cookie: String) { private fun cancelCall() {
call?.cancel()
callAuth?.cancel() callAuth?.cancel()
callInfo?.cancel()
fAuth?.cancel()
callPage?.cancel()
}
private fun fetchAuth(tvModel: TVViewModel, cookie: String) {
cancelCall()
val title = tvModel.title.value val title = tvModel.title.value
@ -146,9 +160,9 @@ object Request {
}) })
} }
fun fetchVideo(tvModel: TVViewModel, cookie: String) { private fun fetchVideo(tvModel: TVViewModel, cookie: String) {
Log.i(TAG, "fetchVideo with cookie") cancelCall()
call?.cancel()
if (::btraceRunnable.isInitialized) { if (::btraceRunnable.isInitialized) {
handler.removeCallbacks(btraceRunnable) handler.removeCallbacks(btraceRunnable)
} }
@ -284,10 +298,11 @@ object Request {
}) })
} }
fun fetchAuth(tvModel: TVViewModel) { private fun fetchAuth(tvModel: TVViewModel) {
cancelCall()
if (token == "") { if (token == "") {
yspTokenService.getInfo("") callInfo = yspTokenService.getInfo("")
.enqueue(object : Callback<Info> { callInfo?.enqueue(object : Callback<Info> {
override fun onResponse(call: Call<Info>, response: Response<Info>) { override fun onResponse(call: Call<Info>, response: Response<Info>) {
if (response.isSuccessful) { if (response.isSuccessful) {
token = response.body()?.data?.token!! token = response.body()?.data?.token!!
@ -331,11 +346,12 @@ object Request {
} }
} }
fun fetchVideo(tvModel: TVViewModel) { private fun fetchVideo(tvModel: TVViewModel) {
cancelCall()
Log.d(TAG, "fetchVideo") Log.d(TAG, "fetchVideo")
if (token == "") { if (token == "") {
yspTokenService.getInfo("") callInfo = yspTokenService.getInfo("")
.enqueue(object : Callback<Info> { callInfo?.enqueue(object : Callback<Info> {
override fun onResponse(call: Call<Info>, response: Response<Info>) { override fun onResponse(call: Call<Info>, response: Response<Info>) {
if (response.isSuccessful && response.body()?.data?.token != null) { if (response.isSuccessful && response.body()?.data?.token != null) {
token = response.body()?.data?.token!! token = response.body()?.data?.token!!
@ -379,11 +395,8 @@ object Request {
} }
} }
private var fAuth: Call<FAuth>? = null private fun fetchFAuth(tvModel: TVViewModel) {
fun fetchFAuth(tvModel: TVViewModel) { cancelCall()
call?.cancel()
callAuth?.cancel()
fAuth?.cancel()
val title = tvModel.title.value val title = tvModel.title.value
@ -400,11 +413,11 @@ object Request {
// Log.d(TAG, "$title url $url") // Log.d(TAG, "$title url $url")
tvModel.addVideoUrl(url) tvModel.addVideoUrl(url)
tvModel.allReady() tvModel.allReady()
tvModel.tokenRetryTimes = 0 tvModel.tokenFHRetryTimes = 0
} else { } else {
Log.e(TAG, "auth status error ${response.code()}") Log.e(TAG, "auth status error ${response.code()}")
if (tvModel.tokenRetryTimes < tvModel.tokenRetryMaxTimes) { if (tvModel.tokenFHRetryTimes < tvModel.tokenFHRetryMaxTimes) {
tvModel.tokenRetryTimes++ tvModel.tokenFHRetryTimes++
fetchFAuth(tvModel) fetchFAuth(tvModel)
} }
} }
@ -412,8 +425,8 @@ object Request {
override fun onFailure(call: Call<FAuth>, t: Throwable) { override fun onFailure(call: Call<FAuth>, t: Throwable) {
Log.e(TAG, "auth request error $t") Log.e(TAG, "auth request error $t")
if (tvModel.tokenRetryTimes < tvModel.tokenRetryMaxTimes) { if (tvModel.tokenFHRetryTimes < tvModel.tokenFHRetryMaxTimes) {
tvModel.tokenRetryTimes++ tvModel.tokenFHRetryTimes++
fetchFAuth(tvModel) fetchFAuth(tvModel)
} }
} }
@ -421,7 +434,6 @@ object Request {
} }
fun fetchData(tvModel: TVViewModel) { fun fetchData(tvModel: TVViewModel) {
Log.i(TAG, "fetchData")
if (tvModel.getTV().channel == "港澳台") { if (tvModel.getTV().channel == "港澳台") {
fetchFAuth(tvModel) fetchFAuth(tvModel)
return return
@ -446,7 +458,7 @@ object Request {
class TokenRunnable : Runnable { class TokenRunnable : Runnable {
override fun run() { override fun run() {
fetchToken() // fetchToken()
} }
} }
@ -489,15 +501,21 @@ object Request {
Utils.setBetween(c * 1000L) Utils.setBetween(c * 1000L)
Log.i(TAG, "current time $c") Log.i(TAG, "current time $c")
} }
listener?.onRequestFinished()
} else { } else {
Log.e(TAG, "token status error") Log.e(TAG, "token status error")
handler.postDelayed( handler.postDelayed(
tokenRunnable, tokenRunnable,
30 * 60 * 1000 30 * 60 * 1000
) )
} if (initRetryTimes < initRetryMaxTimes) {
initRetryTimes++
fetchInfoV2()
} else {
listener?.onRequestFinished() listener?.onRequestFinished()
} }
}
}
override fun onFailure(call: Call<InfoV2>, t: Throwable) { override fun onFailure(call: Call<InfoV2>, t: Throwable) {
Log.e(TAG, "token request error $t") Log.e(TAG, "token request error $t")
@ -505,8 +523,14 @@ object Request {
tokenRunnable, tokenRunnable,
30 * 60 * 1000 30 * 60 * 1000
) )
if (initRetryTimes < initRetryMaxTimes) {
initRetryTimes++
fetchInfoV2()
} else {
listener?.onRequestFinished() listener?.onRequestFinished()
} }
}
}) })
} }
@ -601,7 +625,8 @@ object Request {
} }
fun fetchPage() { fun fetchPage() {
yspProtoService.getPage().enqueue(object : Callback<pageModel.Response> { callPage = yspProtoService.getPage()
callPage?.enqueue(object : Callback<pageModel.Response> {
override fun onResponse( override fun onResponse(
call: Call<pageModel.Response>, call: Call<pageModel.Response>,
response: Response<pageModel.Response> response: Response<pageModel.Response>

@ -8,14 +8,7 @@ import androidx.fragment.app.DialogFragment
import com.lizongying.mytv.databinding.DialogBinding import com.lizongying.mytv.databinding.DialogBinding
class SettingFragment( class SettingFragment : DialogFragment() {
private val versionName: String,
private val versionCode: Long,
private val channelReversal: Boolean,
private val channelNum: Boolean,
private val bootStartup: Boolean,
) :
DialogFragment() {
private var _binding: DialogBinding? = null private var _binding: DialogBinding? = null
private val binding get() = _binding!! private val binding get() = _binding!!
@ -32,33 +25,37 @@ class SettingFragment(
container: ViewGroup?, container: ViewGroup?,
savedInstanceState: Bundle? savedInstanceState: Bundle?
): View { ): View {
val context = requireContext() // It‘s safe to get context here.
_binding = DialogBinding.inflate(inflater, container, false) _binding = DialogBinding.inflate(inflater, container, false)
_binding?.version?.text = binding.version.text =
"当前版本: $versionName\n获取最新: https://github.com/lizongying/my-tv/releases/" "当前版本: ${context.appVersionName}\n获取最新: https://github.com/lizongying/my-tv/releases/"
val switchChannelReversal = _binding?.switchChannelReversal binding.switchChannelReversal.run {
switchChannelReversal?.isChecked = channelReversal isChecked = SP.channelReversal
switchChannelReversal?.setOnCheckedChangeListener { _, isChecked -> setOnCheckedChangeListener { _, isChecked ->
SP.channelReversal = isChecked SP.channelReversal = isChecked
(activity as MainActivity).settingActive() (activity as MainActivity).settingActive()
} }
}
val switchChannelNum = _binding?.switchChannelNum binding.switchChannelNum.run {
switchChannelNum?.isChecked = channelNum isChecked = SP.channelNum
switchChannelNum?.setOnCheckedChangeListener { _, isChecked -> setOnCheckedChangeListener { _, isChecked ->
SP.channelNum = isChecked SP.channelNum = isChecked
(activity as MainActivity).settingActive() (activity as MainActivity).settingActive()
} }
}
val switchBootStartup = _binding?.switchBootStartup binding.switchBootStartup.run {
switchBootStartup?.isChecked = bootStartup isChecked = SP.bootStartup
switchBootStartup?.setOnCheckedChangeListener { _, isChecked -> setOnCheckedChangeListener { _, isChecked ->
SP.bootStartup = isChecked SP.bootStartup = isChecked
(activity as MainActivity).settingActive() (activity as MainActivity).settingActive()
} }
}
updateManager = UpdateManager(context, this, versionCode) updateManager = UpdateManager(context, this, context.appVersionCode)
_binding?.checkVersion?.setOnClickListener(OnClickListenerCheckVersion(updateManager)) binding.checkVersion.setOnClickListener(OnClickListenerCheckVersion(updateManager))
return binding.root return binding.root
} }

@ -23,7 +23,7 @@ import java.io.File
class UpdateManager( class UpdateManager(
private var context: Context?, private var context: Context,
private var settingFragment: SettingFragment, private var settingFragment: SettingFragment,
private var versionCode: Long private var versionCode: Long
) : ) :
@ -67,7 +67,7 @@ class UpdateManager(
val apkFileName = "my-tv-${release.data.versionName}.apk" val apkFileName = "my-tv-${release.data.versionName}.apk"
Log.i(TAG, "apkFileName $apkFileName") Log.i(TAG, "apkFileName $apkFileName")
val downloadManager = val downloadManager =
context!!.getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager context.getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager
val request = Request(Uri.parse(release.data.downloadUrl)) val request = Request(Uri.parse(release.data.downloadUrl))
Log.i(TAG, "url ${Uri.parse(release.data.downloadUrl)}") Log.i(TAG, "url ${Uri.parse(release.data.downloadUrl)}")
request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, apkFileName) request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, apkFileName)
@ -77,22 +77,22 @@ class UpdateManager(
// 获取下载任务的引用 // 获取下载任务的引用
val downloadReference = downloadManager.enqueue(request) val downloadReference = downloadManager.enqueue(request)
downloadReceiver = DownloadReceiver(context!!, apkFileName, downloadReference) downloadReceiver = DownloadReceiver(context, apkFileName, downloadReference)
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
context!!.registerReceiver( context.registerReceiver(
downloadReceiver, downloadReceiver,
IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE), IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE),
Context.RECEIVER_NOT_EXPORTED, Context.RECEIVER_NOT_EXPORTED,
) )
} else { } else {
context!!.registerReceiver( context.registerReceiver(
downloadReceiver, downloadReceiver,
IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE) IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE)
) )
} }
getDownloadProgress(context!!, downloadReference) { progress -> getDownloadProgress(context, downloadReference) { progress ->
println("Download progress: $progress%") println("Download progress: $progress%")
} }
} }
@ -182,7 +182,7 @@ class UpdateManager(
fun destroy() { fun destroy() {
if (downloadReceiver != null) { if (downloadReceiver != null) {
context!!.unregisterReceiver(downloadReceiver) context.unregisterReceiver(downloadReceiver)
Log.i(TAG, "destroy downloadReceiver") Log.i(TAG, "destroy downloadReceiver")
} }
} }

@ -22,10 +22,11 @@ class TVViewModel(private var tv: TV) : ViewModel() {
private var itemPosition: Int = 0 private var itemPosition: Int = 0
var retryTimes = 0 var retryTimes = 0
var tokenRetryTimes = 0
var retryMaxTimes = 8 var retryMaxTimes = 8
var tokenRetryMaxTimes = 2 var tokenRetryTimes = 0
var programUpdateTime: Long = 0 var tokenRetryMaxTimes = 0
var tokenFHRetryTimes = 0
var tokenFHRetryMaxTimes = 2
private val _errInfo = MutableLiveData<String>() private val _errInfo = MutableLiveData<String>()
val errInfo: LiveData<String> val errInfo: LiveData<String>

@ -130,7 +130,7 @@
"programId": "600004078", "programId": "600004078",
"needToken": true, "needToken": true,
"mustToken": false, "mustToken": false,
"title": "CCTV9 录", "title": "CCTV9 录",
"videoUrl": [ "videoUrl": [
"http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226197/index.m3u8" "http://dbiptv.sn.chinamobile.com/PLTV/88888890/224/3221226197/index.m3u8"
] ]

Loading…
Cancel
Save