Как предоставить разрешение MODIFY_PHONE_STATE для приложений, запущенных на Gingerbread - PullRequest
37 голосов
/ 17 января 2011

Я пишу приложение, которое пытается изменить состояние телефонного звонка. Он хорошо работает на 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)

Ответы [ 5 ]

55 голосов
/ 23 февраля 2011

Проблема, с которой вы столкнулись, была представлена ​​в Android 2.3 (Gingerbread). Любой имеющийся у вас код, требующий MODIFY_PHONE_STATE, будет работать вплоть до (включая) версии Android 2.2, но не работает для Android 2.3 +.

Изменение было проверено в Дэвидом Брауном, который ограничивает использование разрешения MODIFY_PHONE_STATE для системных приложений. Системные приложения либо

  1. Предварительно установлено в системную папку на ПЗУ
  2. Составлено производителем с использованием сертификата безопасности

Я подозреваю, что вы пытаетесь использовать скрытый API, такой как ITelephony. Я был - и я был сожжен этим изменением. Android-команда оправдывает себя тем, что это был скрытый API, который не должен был использоваться в первую очередь.

Тем не менее, запрос на улучшение открыт для создания надлежащего общедоступного API телефонии, но Google убил заявку. Похоже, их позиция заключается в том, что они не собираются менять направление, и эти API не предназначены для общественного потребления.

21 голосов
/ 18 января 2011

MODIFY_PHONE_STATE - это разрешение только для системы, поэтому приложениям не разрешено его получать.

Возможно, оно изменилось по сравнению с предыдущими версиями платформы, но это нормально, поскольку оно защищает только частные API, поэтомуесли вы делаете что-то, что требует этого, вы используете частные API, которые не поддерживаются и приведут к тому, что ваше приложение сломается на разных сборках платформы.

Сканирование стека, которое вы включаете, не завершено, поэтомунет никакого способа сказать, что ты на самом деле делаешь.

6 голосов
/ 14 марта 2012

Попробуйте это.

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 голосов
/ 27 февраля 2012

Я получил решение.

Вместо того, чтобы переопределить экран входящего вызова, сделайте ниже две вещи.которая позволит вам получить доступ к кнопке принять и отклонить, а также отобразить экран над экраном входящего вызова.

(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) Сделайте макет вашей активности прозрачным (что будет выше экрана вызова), напишите следующий код в menifest

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

(4) В menifestдобавьте получателя широкого диапазона

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

(5) добавьте приведенный ниже код в oncreate () IncomingCallActivity

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

Cheers!

Дайте мне знать, если вы сталкиваетесьпроблема!

0 голосов
/ 17 января 2011

Если ваше приложение для Gingerbread запущено на планшете, а телефона нет, то это ожидаемое поведение.Вам нужно будет сделать разрешения, связанные с телефонией в манифесте, необязательными для запуска на планшетах.

Попробуйте это в своем манифесте:

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

Конечно, я делаюБольшое предположение о планшете.Вы также можете увидеть ссылку на Android здесь .

...