Как предоставить разрешение MODIFY_PHONE_STATE для приложений, запущенных на Gingerbread

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

Решение

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

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

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

Проблема, с которой вы столкнулись, была представлена ​​в 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 не предназначены для общественного потребления.

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

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

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

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

(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) добавить приведенный ниже код в oncreate () IncomingCallActivity

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

Ура!

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

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

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

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

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

Другие вопросы по тегам