- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
所以我目前正在 Android 中为双 SIM 卡设备实现调用转移功能。为了读取 SIM 卡调用转接的当前状态(启用/禁用),我执行以下操作:
TelephonyManager
对象:val telephonyManager = getSystemService(TELEPHONY_SERVICE) 作为 TelephonyManager
我创建一个 PhoneStateListener
对象并重写 onCallForwardingIndicatorChanged
方法:
val myPhoneStateListener = object: PhoneStateListener() {
override fun onCallForwardingIndicatorChanged(isCallForwardingEnabled: Boolean) {
if(isCallForwardingEnabled) println("Call forwarding enabled!")
else println("Call forwarding disabled!")
}
}
我注册了PhoneStateListener
:
telephonyManager.listen(myPhoneStateListener, LISTEN_CALL_FORWARDING_INDICATOR)
这对于主(第一张)SIM 卡来说效果非常好。
但我在对第二张 SIM 卡执行同样的操作时遇到困难。我正在尝试这样做:
我使用 SubscriptionManager
对象来检索第二张 SIM 卡的 subscriptionId:
val subscriptionManager = getSystemService(TELEPHONY_SUBSCRIPTION_SERVICE) as SubscriptionManager
val subscriptionIdOfSimCard2 = subscriptionManager.getActiveSubscriptionInfoForSimSlotIndex(1).subscriptionId
我为第二张 SIM 卡创建一个单独的 TelephonyManager
,并使用正确的 subscriptionId:
val secondTelephonyManager = (getSystemService(TELEPHONY_SERVICE) as TelephonyManager).createForSubscriptionId(subscriptionIdOfSimCard2)
我创建了第二个 PhoneStateListener
,就像第一个 SIM 卡的那样,我们将其命名为 mySecondPhoneStateListener
并将其注册到第二个 TelephonyManager
:
secondTelephonyManager.listen(mySecondPhoneStateListener, LISTEN_CALL_FORWARDING_INDICATOR)
现在的问题是,在 mySecondPhoneStateListener 中,我没有收到第二张 SIM 卡的回调,但仍然收到第一张主要卡的回调。在深入研究 Android 源代码后,我发现了原因:在 TelephonyManager
的 listen(PhoneStateListenerlistener, int events)
方法中,使用了错误的 subscriptionId,即不是 TelephonyManager
中设置的,而是 PhoneStateListener
对象中设置的,默认为第一张 SIM 卡的 subscriptionId:
public void listen(PhoneStateListener listener, int events) {
if (mContext == null) return;
try {
Boolean notifyNow = (getITelephony() != null);
sRegistry.listenForSubscriber(listener.mSubId, */ HERE: listener.mSubId is used instead of this.mSubId */
getOpPackageName(), listener.callback, events, notifyNow);
} catch (RemoteException ex) {
// system process dead
} catch (NullPointerException ex) {
// system process dead
}
}
这个问题可以通过为 PhoneStateListener
对象设置正确的 subscriptionId 来解决,但是适当的构造函数被隐藏了:
/**
* Create a PhoneStateListener for the Phone using the specified subscription.
* This class requires Looper.myLooper() not return null. To supply your
* own non-null Looper use PhoneStateListener(int subId, Looper looper) below.
* @hide */<-- HIDDEN, NOT ACCESSIBLE*/
*/
public PhoneStateListener(int subId) {
this(subId, Looper.myLooper());
}
我能够通过反射“解决”这个问题,方法是将 PhoneStateListener
对象的 mSubId 字段设置为第二张 SIM 卡的相应 subscriptionId。
但是必须有更好的方法来做到这一点,我错过了什么吗?
最佳答案
我用 Listeners 制作了简单的 ArrayList,它对我来说很好用(顺便说一句,它是 Kotlin)
在我的 Activity 中:
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.sim_selector)
checkForForwarding()
}
fun getSimsCount(): Int {
val subscriptionManager = SubscriptionManager.from(this)
val activeSubscriptionInfoList = subscriptionManager.activeSubscriptionInfoList
return activeSubscriptionInfoList.size
}
class SimForwardListeners{
var subscriptionId: Int = 0
lateinit var manager: TelephonyManager
lateinit var phoneStateListener: MyPhoneStateListener
}
private val simsForwardListeners: ArrayList<SimForwardListeners> = arrayListOf()
fun checkForForwarding() {
val subscriptionManager = getSystemService(TELEPHONY_SUBSCRIPTION_SERVICE) as SubscriptionManager
for (slotIndex in 0 until getSimsCount()) {
val z = SimForwardListeners()
z.phoneStateListener = MyPhoneStateListener(slotIndex)
z.phoneStateListener.setView(this)
z.subscriptionId = subscriptionManager.getActiveSubscriptionInfoForSimSlotIndex(slotIndex).subscriptionId
z.manager = (getSystemService(TELEPHONY_SERVICE) as TelephonyManager).createForSubscriptionId(z.subscriptionId)
z.manager.listen(z.phoneStateListener, PhoneStateListener.LISTEN_CALL_FORWARDING_INDICATOR)
simsForwardListeners.add(z)
}
}
关于android - 使用双 SIM 卡功能时如何获取 PhoneStateListener,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47352840/
我的应用程序处理传入/传出调用并显示一些 UI。所以我使用 PhoneStateListener 来监听 CALL_STATE_IDLE、CALL_STATE_RINGING、CALL_STATE_O
当设备进入打盹模式或省电模式时,我的 PhoneStateListener 停止工作。我还需要在设备重启时注册 PhoneStateListener。我不想使用 BroadcastReceiver,因
我使用 PhoneStateListener 来查明电话何时处于 Activity 通话中以及通话何时结束并记录它是在来电还是去电中。当我安装监听器时,它在第一次调用时按预期工作。之后,尽管它仍然有效
我实现了一个 PhoneStateListener 如何在您被调用时停止音频并在您完成后再次启动它。当我通过 finish() 退出我的应用程序时,PhoneStateListener 似乎仍然处于
我正在尝试构建一个应用程序,在我接听和拒绝来电时进行记录。因此,我正在使用 PhoneStateListener。 当我在 onCreate() 方法中启动监听器时,它会在一段时间后停止 Activi
我正在尝试设置 PhoneStateListener 但我得到一个 PhoneCallListener cannot be resolved to a type。 public class Butto
我正在尝试使用 Broadcast Receiver 测试 PhoneStateListener。但我无法这样做。接收类中的日志语句也没有在 Logcat 中打印。 这是 list 文件
这是我的全部代码...... 广播示例.java package com.example.broadcast; > import android.app.Activity; import > and
我想在 android 应用程序中检测拨出电话。实际上我可以检测到拨出电话,但它也适用于来电,我不希望这样。我需要它只为传出工作。这是我的代码.... boolean ringing = false;
我目前正在开发一个 android 应用程序来监视来自电话的传入和传出调用并将调用信息注册到文件中,从我读过的内容来看 PhoneStateListener 似乎做了什么我需要。 问题是我需要应用程序
我正在使用 PhoneStateListener,因此当用户接到电话或开始通话时,扬声器将立即打开。 PhoneStateListener phoneStateListener = new P
我写了一个应用程序,它通过 PhoneStateListener 监控我的信号强度。我希望此应用在启动时启动并永远运行。 我的管理方法如下,但我想知道是否有人可以推荐更好的方法。 我已经注册了一个在
我在 android 上遇到问题。我想知道设备是否已连接到 GSM 网络,以便它可以发送短信。 两种方式我都试过了,结果都是一样的。在 Moto G 4G 上,我能够确定它是否可以访问。在 Redmi
我创建了一个 BroadcastReceiver 来监听他的电话状态并采取相应的行动。 这里的问题是它似乎没有从我的数据库中获取正确的值,并且从一开始处于空闲状态的 toast 显示两次,第二次调用后
我的应用程序运行多次后出现内存问题。 java.lang.OutOfMemoryError: 位图大小超出 VM 预算 我想我不知何故泄漏了内存,所以我做了一个 DUMP HPROF 文件并使用 MA
我需要从几个类中调用方法,但我不知道如何获得正确的上下文 持有类: public class SharedData { ...... ...... public static void sto
在某些情况下,我不想听手机的状态。如何销毁 PhoneStateListener 类的对象? 我这样创建对象 try { phoneCallListener = new WnetPlaye
在 Android(目标 25)中,我有一个后台服务,在 onCreate 函数中我已经初始化了一个电话状态监听器。它在 Nougat 之前的 Android 版本上运行良好,但在 Nougat 中它
我正在编写包含两个元素的简单应用程序 - 服务和 Activity 。 Activity 仅用于启动和停止服务。服务使用 PhoneStateListener 获取有关当前 CellID、LAC 和
我刚刚接触 Android 开发,并决定我在这个新领域的第一个征服将是掌握手机对来电的 react 。 后来我用谷歌搜索了一下 http://www.compiletimeerror.com/2013
我是一名优秀的程序员,十分优秀!