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)
4

5 回答 5

55

您遇到的问题是在 Android 2.3 (Gingerbread) 中引入的。您拥有的任何需要 MODIFY_PHONE_STATE 的代码都可以一直运行到(包括)Android 2.2,但会在 Android 2.3+ 上中断。

David Brown签入了一项更改,该更改将 MODIFY_PHONE_STATE 权限的使用限制在系统应用程序中。系统应用程序要么

  1. 预安装到 ROM 上的系统文件夹中
  2. 由制造商使用其安全证书编译

我怀疑您正在尝试使用隐藏的 API,例如 ITelephony。我是 - 我被这种变化烧伤了。Android 团队的理由是它是一个隐藏的 API,你一开始就不应该使用它。

也就是说,打开了一个增强请求来创建一个适当的公共电话 API,但谷歌取消了票。他们的立场似乎是他们不打算改变方向,并且这些 API 不是供公众使用的。

于 2011-02-23T19:24:26.293 回答
20

MODIFY_PHONE_STATE 是系统独有的权限,因此不允许应用获取它。

这可能与以前版本的平台有所不同,但这没关系,因为它只保护私有 API,所以如果你正在做一些需要它的事情,你正在使用不受支持的私有 API,并且会导致像你的应用程序这样的事情打破平台的不同版本。

您包含的堆栈爬网不完整,因此无法判断您实际在做什么。

于 2011-01-18T02:51:47.920 回答
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");
}
于 2012-03-14T12:03:29.177 回答
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的布局透明(会出现在调用屏幕的上方),在menifest中写下下面的代码

<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);

干杯!

如果您遇到任何问题,请告诉我!

于 2012-02-27T08:40:00.243 回答
-5

如果您的 Gingerbread 应用程序在平板电脑上运行并且没有手机,那么这是预期的行为。您需要将清单中的电话相关权限设为非强制性才能在平板电脑上运行。

在你的清单中试试这个:

<uses-feature android:name="android.hardware.telephony" 
    android:required="false" />

当然,我对平板电脑做了一个很大的假设。您还可以在此处查看 Android 参考。

于 2011-01-17T16:26:21.143 回答