如何在Gingerbread上为应用程序授予MODIFY_PHONE_STATE权限

39

我写了一个应用程序尝试修改电话呼叫状态。它在Android 2.2或更低版本上运行良好,但在Android 2.3上由于缺乏android.permission.MODIFY_PHONE_STATE权限而抛出异常(我在AndroidManifest.xml中声明了此权限)。有任何想法吗?以下是异常日志:

01-15 09:14:23.210: ERROR/AndroidRuntime(404): FATAL EXCEPTION: main
01-15 09:14:23.210: ERROR/AndroidRuntime(404): java.lang.RuntimeException: Unable to start receiver test.PhoneReceiver: java.lang.SecurityException: Neither user 10031 nor current process has android.permission.MODIFY_PHONE_STATE.
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at android.app.ActivityThread.handleReceiver(ActivityThread.java:1780)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at android.app.ActivityThread.access$2400(ActivityThread.java:117)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:978)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at android.os.Handler.dispatchMessage(Handler.java:99)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at android.os.Looper.loop(Looper.java:123)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at android.app.ActivityThread.main(ActivityThread.java:3647)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at java.lang.reflect.Method.invokeNative(Native Method)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at java.lang.reflect.Method.invoke(Method.java:507)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
01-15 09:14:23.210: ERROR/AndroidRuntime(404):     at dalvik.system.NativeStart.main(Native Method)

请参考Shamrflow在此的帖子 - Name is Nilay
5个回答

55
你遇到的问题是在Android 2.3(姜饼)中引入的。任何需要MODIFY_PHONE_STATE权限的代码都可以在Android 2.2及以下版本上运行,但在Android 2.3+上会出现错误。
David Brown进行了更改,限制了MODIFY_PHONE_STATE权限的使用,只能用于系统应用程序。系统应用程序可以是:
  1. 预装在ROM上的系统文件夹中
  2. 由制造商使用其安全证书编译
我猜你试图使用ITelephony这样的隐藏API。我也是这样做的 - 并因此而受到了影响。 Android团队的理由是这是一个隐藏的API,你本来就不应该使用它。
话虽如此,有一个增强请求来创建一个合适的公共电信API,但Google取消了该请求。他们的立场似乎是他们不打算改变方向,这些API不适用于公众消费。

你的回答中链接已经失效了,你知道现在它可以在哪里找到吗? - donfuxx

20

“MODIFY_PHONE_STATE”是一个仅限系统使用的权限,因此应用程序不能获取它。

这可能与平台先前的版本有所不同,但没关系,因为它只是保护私有API,因此,如果您正在进行需要此权限的操作,则正使用不受支持的私有API,将导致您的应用在平台的不同版本上出现问题。

您包含的堆栈跟踪不完整,因此无法确定您实际在做什么。


1
我的应用程序试图阻止依赖于内部API的来电。这在Android 2.2及更早版本上完美运行,但在Android 2.3上却不行。你有什么想法吗? - Bao Le
7
是的,你正在使用私有API,而该API已不再可用。这就是依赖API的后果。这也是为什么我们努力让人们清楚地知道何时没有使用公共API,以免冒遇到问题的风险。 - hackbod
4
Dianne,Android dringend eine angemessene vollständige öffentliche Telefoni-API benötigt! Bitte!! :) - Donal Rafferty
1
我们为什么不能只使用反射来访问私有的API呢?如果Android默认不提供它们,是否有原因? - Tom
3
这个问题的核心就在这里。该应用程序正在使用私有实现,而这个实现已经发生了改变,导致应用程序出现故障。使用不在支持SDK中的东西意味着你的应用程序可能会在任何时间、任何设备、任何Android版本上出现故障。 - hackbod

6

试试这个。

public static void answerPhoneHeadsethook(Context context) {
    // Simulate a press of the headset button to pick up the call
    // SettingsClass.logMe(tag, "Simulating headset button");
    Intent buttonDown = new Intent(Intent.ACTION_MEDIA_BUTTON);
    buttonDown.putExtra(Intent.EXTRA_KEY_EVENT, new KeyEvent(KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_HEADSETHOOK));
    context.sendOrderedBroadcast(buttonDown, "android.permission.CALL_PRIVILEGED");

    // froyo and beyond trigger on buttonUp instead of buttonDown
    Intent buttonUp = new Intent(Intent.ACTION_MEDIA_BUTTON);
    buttonUp.putExtra(Intent.EXTRA_KEY_EVENT, new KeyEvent(KeyEvent.ACTION_UP, KeyEvent.KEYCODE_HEADSETHOOK));
    context.sendOrderedBroadcast(buttonUp, "android.permission.CALL_PRIVILEGED");
}

5

我已经找到解决方案。

不要覆盖来电屏幕,而是执行以下两个步骤。这将允许您访问接受和拒绝按钮,并允许您在来电屏幕上方显示屏幕。

(1) 创建一个接收器类:

public class MyPhoneReceiver extends BroadcastReceiver {
@Override
public void onReceive(final Context context, final Intent intent) {
    Bundle extras = intent.getExtras();
    if (extras != null) 
    {
        String state = extras.getString(TelephonyManager.EXTRA_STATE);
        if (state.equals(TelephonyManager.EXTRA_STATE_RINGING)) 
            String phoneNumber = extras.getString(TelephonyManager.EXTRA_INCOMING_NUMBER);

        Intent i = new Intent(context, IncomingCallActivity.class);
        i.putExtras(intent);
        i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        context.startActivity(i);
    }
    }
}

(2) 您的活动xml应该是这样的:

RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:layout_gravity="top"
android:gravity="top"
android:orientation="vertical"
android:windowAnimationStyle="@android:style/Animation.Translucent"
android:windowBackground="@android:color/transparent"
android:windowIsTranslucent="true"

(3) 将您的活动布局设置为透明(在调用屏幕之上),请在清单中编写以下代码

<activity android:name=".IncomingCallActivity" 
         android:theme="@android:style/Theme.Translucent">
</activity>

(4)在清单文件中添加你的广播接收器

<receiver android:name="MyPhoneReceiver" >
        <intent-filter>
            <action android:name="android.intent.action.PHONE_STATE" >
            </action>
        </intent-filter>
</receiver>

(5) 在IncomingCallActivity的oncreate()方法中添加以下代码

getWindow().addFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE);
getWindow().addFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL);
this.requestWindowFeature(Window.FEATURE_NO_TITLE);

干杯!

如果你遇到任何问题,请让我知道!


1
你的意思是说你成功地使用了AIDL技术,并且使用了上述逻辑来允许用户使用屏幕按钮(在你的activity XML中定义)来接电话吗?如果真的可以这样工作,那就太好了。 - omkar.ghaisas
我已经在onStop()中完成了该应用程序的所有活动。 - Dhrupal
@Dhrupal:在这种情况下,我们将触摸事件传递到下面的屏幕(即本机传入屏幕),那么如果我想访问我的屏幕上的视图(比如某个按钮),是否可能呢? - Vikram Singh
你好,很好的解决方案!我正在尝试实现它,但是我遇到了一个异常; Caused by: android.util.AndroidRuntimeException: requestFeature() 必须在添加内容之前调用 请帮忙。 - SamAko
2
在调用 setContentView(id) 之前,请使用 @elimence 调用窗口请求功能。 - Vikrant_Dev
显示剩余6条评论

-5
如果您的Gingerbread应用程序在平板电脑上运行且没有电话,则这是预期行为。您需要使清单中与电话相关的权限变为非强制性,以便在平板电脑上运行。
请在您的清单中尝试以下内容:
<uses-feature android:name="android.hardware.telephony" 
    android:required="false" />

当然,我对这个平板电脑做了一个很大的假设。你也可以在这里看到Android的参考资料。


3
谢谢。我的应用确实使用了与电话相关的功能。在Gingerbread/Android 2.3上出现了问题,但在Android 2.2上一直正常工作。 - Bao Le

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接