2011-01-17 2 views
37

Я пишу приложение, которое пытается изменить состояние телефонного звонка. Он отлично работает на Android 2.2 или меньше, но выдает исключение на Android 2.3 из-за отсутствия разрешения на разрешение android.permission.MODIFY_PHONE_STATE (я объявил это разрешение на AndroidManifest.xml). Есть идеи? Ниже журнал исключение:Как предоставить разрешение MODIFY_PHONE_STATE для приложений, запущенных на Gingerbread

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) 
+0

Обратитесь к [Shamrflow в пост здесь] (http://stackoverflow.com/questions/10054348) –

ответ

21

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

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

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

+1

Мое приложение пытается заблокировать входящий вызов, основанный на внутреннем API. Это отлично работает на Android 2.2 и более ранних версиях, но Android 2.3. Есть ли у вас какие-либо идеи –

+7

Да, вы используете частный API, и API больше не работает. Это то, что происходит, когда вы полагаетесь на API. Вот почему мы стараемся сделать так, чтобы ясность, когда вы не используете публичные API, и рискуете столкнуться с проблемой. – hackbod

+4

Android плохо нуждается в полноценной общественной телефоне API Dianne! Пожалуйста!! :) –

-3

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

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

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

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

+3

Спасибо. В моем приложении используются функции, связанные с телефонией. Это превратилось в проблему на Gingerbread/Android 2.3, но всегда работает отлично на Android 2.2 –

54

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

A change was checked in от David Brown, который ограничивает использование разрешения MODIFY_PHONE_STATE для системных приложений. Системные приложения являются либо

  1. Предварительно установленный в системную папку на диске
  2. Составитель производителя, используя их сертификат безопасности

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

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

+0

В вашем ответе ссылка не работает, возможно, вы знаете, где ее можно найти сейчас? – donfuxx

4

У меня есть решение.

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

(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

Ура!

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

+1

Вы имеете в виду, что вы успешно использовали технику aidl и использовали вышеуказанную логику для активности, чтобы позволить пользователю отвечать на вызов с помощью экранных кнопок (которые определены в вашей деятельности xml)? Это было бы здорово, если бы это действительно так. –

+0

Нет, этот код позволит вам поставить ваши вещи выше исходного экрана входящих вызовов и позволить вам одновременно получать доступ к кнопке входящего и исходящего вызова по умолчанию. – Dhrupal

+0

@Dhrupal, не могли бы вы объяснить, как вы это сделали. Что вы на самом деле делаете в своей деятельности. У меня есть аналогичная requiremnet, я хочу показать дополнительную информацию для входящих звонков – png

5

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

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

Это работает со мной android 4.1.2 Xperia –

+0

Работает на 2.3.3, но не работает 4.1.2 –

Смежные вопросы