Как сделать телефонный звонок, используя интент в Android?

329

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

 posted_by = "111-333-222-4";

 String uri = "tel:" + posted_by.trim() ;
 Intent intent = new Intent(Intent.ACTION_CALL);
 intent.setData(Uri.parse(uri));
 startActivity(intent);

разрешений

 <uses-permission android:name="android.permission.CALL_PHONE" />

исключение

11-25 14:47:01.661: ERROR/AndroidRuntime(302): Uncaught handler: thread main exiting due to uncaught exception
11-25 14:47:01.681: ERROR/AndroidRuntime(302): java.lang.SecurityException: Permission Denial: starting Intent { act=android.intent.action.CALL dat=tel:111-333-222-4 cmp=com.android.phone/.OutgoingCallBroadcaster } from ProcessRecord{43d32508 302:com.Finditnear/10026} (pid=302, uid=10026) requires android.permission.CALL_PHONE
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.os.Parcel.readException(Parcel.java:1218)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.os.Parcel.readException(Parcel.java:1206)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.app.ActivityManagerProxy.startActivity(ActivityManagerNative.java:1214)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.app.Instrumentation.execStartActivity(Instrumentation.java:1373)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.app.Activity.startActivityForResult(Activity.java:2749)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.app.Activity.startActivity(Activity.java:2855)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at com.Finditnear.PostDetail$2$1$1$1.onClick(PostDetail.java:604)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at com.android.internal.app.AlertController$AlertParams$3.onItemClick(AlertController.java:884)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.widget.AdapterView.performItemClick(AdapterView.java:284)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.widget.ListView.performItemClick(ListView.java:3285)
11-25 14:47:01.681: ERROR/AndroidRuntime(302):     at android.widget.AbsListView$PerformClick.run(AbsListView.java:1640)
UMAR
источник

Ответы:

383

Вы можете использовать Intent.ACTION_DIALвместо Intent.ACTION_CALL. Это показывает номеронабиратель с уже введенным номером, но позволяет пользователю решать, действительно ли он совершает вызов или нет. ACTION_DIALне требует CALL_PHONEразрешения.

Чудакулли
источник
ACTION_DIAL не требует разрешения, это важное отличие по сравнению с намерением ACTION_CALL :)
maxwellnewage
237

Это демо будет полезно для вас ...

При нажатии кнопки вызова:

Intent intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + "Your Phone_number"));
startActivity(intent);

Разрешение в Манифесте:

 <uses-permission android:name="android.permission.CALL_PHONE" />
Денни Шарма
источник
1
+1 за "тел:". Вместо этого у меня был звонок, и у меня не было никаких намерений. Tnx
Тина
Это просто не работает в моем Nexus 4. Он открывает номеронабиратель, показывающий номер телефона, но не запускает вызов без взаимодействия с пользователем. Любое предложение?
MatheusJardimB
5
В чем разница между кодом в вопросе и в этом ответе? Как это решает проблему?
Гавриил
Это открывает скайп для меня.
RJB
Пожалуйста, добавьте Разрешение вызова телефона Время выполнения тоже
Raghu Krishnan R
181

Более элегантный вариант:

String phone = "+34666777888";
Intent intent = new Intent(Intent.ACTION_DIAL, Uri.fromParts("tel", phone, null));
startActivity(intent);
cprcrack
источник
5
Это работает без добавления чего-либо в Манифест, как предыдущий ответ
Павлос
4
Разрешения во время выполнения не требуются. +1
кик
82

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

Intent intent = new Intent(Intent.ACTION_DIAL);
intent.setData(Uri.parse("tel:0987654321"));
startActivity(intent);
wheeliez
источник
2
Это лучше в случае, когда вам не нужно спрашивать разрешения.
Закигаурав
1
Во второй строке кода пропущена точка с запятой. идеальный ответ!
ahmed_khan_89
73

Все хорошо.

я просто поместил тег разрешения вызова перед тегом приложения в файле манифеста

и теперь все работает нормально.

UMAR
источник
2
Посмотрите мой ответ ниже о том, как добиться почти того же самого без необходимости в разрешении и с возможностью для пользователя фактически не совершать вызов.
Чудесно
3
Кроме того, следует помнить об устройствах без телефонии: stackoverflow.com/a/9300036/693752
Snicolas
дальнейшая информация, предложенная Сниколасом londatiga.net/it/programming/android/…
Lorensius WL T
31

ВАЖНАЯ ЗАМЕТКА:

Если вы используете, Intent.ACTION_CALLвы должны добавить CALL_PHONEразрешение.

Это хорошо, только если вы не хотите, чтобы ваше приложение отображалось в Google Play для планшетов, которые не используют SIM-карту или не имеют GSM.

В ВАШЕЙ ДЕЯТЕЛЬНОСТИ:

            Intent callIntent = new Intent(Intent.ACTION_CALL);
            callIntent.setData(Uri.parse("tel:" + Constants.CALL_CENTER_NUMBER));
            startActivity(callIntent);

МАНИФЕСТ:

<uses-permission android:name="android.permission.CALL_PHONE" />

Поэтому, если это не критичная функция для вашего приложения, постарайтесь не добавлять CALL_PHONEразрешения.

ДРУГОЕ РЕШЕНИЕ:

Показывает приложение Phone с номером, написанным на экране, поэтому пользователю нужно будет всего лишь нажать кнопку вызова:

            Intent dialIntent = new Intent(Intent.ACTION_DIAL);
            dialIntent.setData(Uri.parse("tel:" + Constants.CALL_CENTER_NUMBER));
            startActivity(dialIntent);

Для этого не нужно разрешение.

МБХ
источник
24

Просто простой oneliner без каких-либо дополнительных разрешений:

private void dialContactPhone(final String phoneNumber) {
    startActivity(new Intent(Intent.ACTION_DIAL, Uri.fromParts("tel", phoneNumber, null)));
}
Артемий
источник
18

используйте этот полный код

          Intent callIntent = new Intent(Intent.ACTION_DIAL);
          callIntent.setData(Uri.parse("tel:"+Uri.encode(PhoneNum.trim())));
          callIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
          startActivity(callIntent);     
Усама Ибрагим
источник
1
Этот метод работал для меня, хотя мне пришлось изменить Intent.ACTION_DIAL на Intent.Anction_CALL.
Наим А. Малик
12

Запрос разрешения в манифесте

<uses-permission android:name="android.permission.CALL_PHONE" />

Для звонков используйте этот код

Intent in = new Intent(Intent.ACTION_CALL, Uri.parse("tel:99xxxxxxxx"));
try {
    startActivity(in);
} catch (android.content.ActivityNotFoundException ex) {
    Toast.makeText(mcontext, "Could not find an activity to place the call.", Toast.LENGTH_SHORT).show();
}
Абхай Ананд
источник
3
Я думаю, что вам не нужно разрешение, так как ваше приложение не вызывает себя, а запрашивает выделенное приложение (у которого есть разрешение), чтобы сделать это.
Мострапотски
9

Разрешение в AndroidManifest.xml

<uses-permission android:name="android.permission.CALL_PHONE" />

Полный код:

private void onCallBtnClick(){
    if (Build.VERSION.SDK_INT < 23) {
        phoneCall();
    }else {

        if (ActivityCompat.checkSelfPermission(mActivity,
                Manifest.permission.CALL_PHONE) == PackageManager.PERMISSION_GRANTED) {

            phoneCall();
        }else {
            final String[] PERMISSIONS_STORAGE = {Manifest.permission.CALL_PHONE};
            //Asking request Permissions
            ActivityCompat.requestPermissions(mActivity, PERMISSIONS_STORAGE, 9);
        }
    }
}

@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
    boolean permissionGranted = false;
    switch(requestCode){
        case 9:
            permissionGranted = grantResults[0]== PackageManager.PERMISSION_GRANTED;
            break;
    }
    if(permissionGranted){
        phoneCall();
    }else {
        Toast.makeText(mActivity, "You don't assign permission.", Toast.LENGTH_SHORT).show();
    }
}

private void phoneCall(){
    if (ActivityCompat.checkSelfPermission(mActivity,
            Manifest.permission.CALL_PHONE) == PackageManager.PERMISSION_GRANTED) {
        Intent callIntent = new Intent(Intent.ACTION_CALL);
        callIntent.setData(Uri.parse("tel:12345678900"));
        mActivity.startActivity(callIntent);
    }else{
        Toast.makeText(mActivity, "You don't assign permission.", Toast.LENGTH_SHORT).show();
    }
}
Атиф Махмуд
источник
8

Разрешения:

<uses-permission android:name="android.permission.CALL_PHONE" />

Намерение:

Intent callIntent = new Intent(Intent.ACTION_CALL);
callIntent.setData(Uri.parse("tel:0377778888"));
startActivity(callIntent);

источник
6

Вы также можете использовать это:

String uri = "tel:" + posted_by.replaceAll("[^0-9|\\+]", "");
Арон Павел
источник
5

Для выполнения действия вызова с использованием намерений, вы должны запросить соответствующие разрешения.

Для этого вы используете права доступа в файле AndroidManifest.xml.

<uses-permission android:name="android.permission.CALL_PHONE" />

Затем включите следующий код в вашу деятельность

if (ActivityCompat.checkSelfPermission(mActivity,
        Manifest.permission.CALL_PHONE) == PackageManager.PERMISSION_GRANTED) {
    //Creating intents for making a call
    Intent callIntent = new Intent(Intent.ACTION_CALL);
    callIntent.setData(Uri.parse("tel:123456789"));
    mActivity.startActivity(callIntent);

}else{
    Toast.makeText(mActivity, "You don't assign permission.", Toast.LENGTH_SHORT).show();
}
Codemaker
источник
4

Для звонка с номеронабирателя (разрешение не требуется):

   fun callFromDailer(mContext: Context, number: String) {
        try {
            val callIntent = Intent(Intent.ACTION_DIAL)
            callIntent.data = Uri.parse("tel:$number")
            mContext.startActivity(callIntent)
        } catch (e: Exception) {
            e.printStackTrace()
            Toast.makeText(mContext, "No SIM Found", Toast.LENGTH_LONG).show()
        }
    }

Для прямого звонка из приложения (требуется разрешение):

  fun callDirect(mContext: Context, number: String) {
        try {
            val callIntent = Intent(Intent.ACTION_CALL)
            callIntent.data = Uri.parse("tel:$number")
            mContext.startActivity(callIntent)
        } catch (e: SecurityException) {
            Toast.makeText(mContext, "Need call permission", Toast.LENGTH_LONG).show()
        } catch (e: Exception) {
            e.printStackTrace()
            Toast.makeText(mContext, "No SIM Found", Toast.LENGTH_LONG).show()
        }
    }

Разрешение:

<uses-permission android:name="android.permission.CALL_PHONE"></uses-permission>
Шохан Ахмед Сиян
источник
3

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

Сакет
источник
2
// Java
String mobileNumber = "99XXXXXXXX";
Intent intent = new Intent();
intent.setAction(Intent.ACTION_DIAL); // Action for what intent called for
intent.setData(Uri.parse("tel: " + mobileNumber)); // Data with intent respective action on intent
startActivity(intent);

// Kotlin
val mobileNumber = "99XXXXXXXX"
val intent = Intent()
intent.action = Intent.ACTION_DIAL // Action for what intent called for
intent.data = Uri.parse("tel: $mobileNumber") // Data with intent respective action on intent
startActivity(intent)
Сумит Джайн
источник
1

В Android для определенных функций вам необходимо добавить разрешение в файл манифеста.

  1. Перейти к проектам AndroidManifest.xml
  2. Нажмите на вкладку Разрешения
  3. Нажмите на Добавить
  4. Выберите Использование разрешений
  5. Смотрите снимок нижевведите описание изображения здесь

6. Сохраните файл манифеста и запустите ваш проект. Ваш проект теперь должен работать так, как ожидается.

MindBrain
источник
1
Какой IDE вы используете?
SHA2NK
1
11-25 14:47:01.681: ERROR/AndroidRuntime(302): blah blah...requires android.permission.CALL_PHONE

^ Ответ лежит в выводе исключения " requires android.permission.CALL_PHONE" :)

Чарли Скотт-Скиннер
источник
1
    final public void Call(View view){

    try {

        EditText editt=(EditText)findViewById(R.id.ed1);
        String str=editt.getText().toString();
        Intent intent = new Intent(Intent.ACTION_DIAL, Uri.parse("tel:"+str));
        startActivity(intent);
    }
    catch (android.content.ActivityNotFoundException e){

        Toast.makeText(getApplicationContext(),"App failed",Toast.LENGTH_LONG).show();
    }
Дипак Сингх
источник
1
 if(ContextCompat.checkSelfPermission(
        mContext,android.Manifest.permission.CALL_PHONE) != 
              PackageManager.PERMISSION_GRANTED) {
                    ActivityCompat.requestPermissions((Activity) mContext, new 
                  String[]{android.Manifest.permission.CALL_PHONE}, 0);
                } else {
                    startActivity(new Intent(Intent.ACTION_CALL, Uri.parse("tel:" + "Your Number")));
                }
Mohsinali
источник