Всплывающее окно на экране входящего вызова, как truecaller

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

  public void onReceive(Context context, Intent intent) {

    try {
        // TELEPHONY MANAGER class object to register one listner
        TelephonyManager tmgr = (TelephonyManager) context
                .getSystemService(Context.TELEPHONY_SERVICE);


            }
    } catch (Exception e) {
        Log.e("Phone Receive Error", " " + e);
    }

}

2 ответа

Я использовал приведенный ниже подход для своего приложения, в котором я хотел показать представление поверх приложения Dialer при его запуске (что-то похожее на Truecaller). Для этого создайте Broadcast Receiver, который помогает принимать различные события устройства, как описано ниже.

Приемник вещания:

        private WindowManager wm;
        private static LinearLayout ly1;
        private WindowManager.LayoutParams params1;

        // onReceive function of the Broadcast Receiver
        public void onReceive(Context arg0, Intent arg1) {
                String state = arg1.getStringExtra(TelephonyManager.EXTRA_STATE);

                // Adds a view on top of the dialer app when it launches.
                if(state.equals(TelephonyManager.EXTRA_STATE_OFFHOOK)){
                    wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
                    params1 = new WindowManager.LayoutParams(
                            LayoutParams.MATCH_PARENT,
                            LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.TYPE_SYSTEM_ALERT |
                            WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
                            WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |
                            WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,
                            PixelFormat.TRANSPARENT);

                    params1.height = 75;
                    params1.width = 512;
                    params1.x = 265; 
                    params1.y = 400;
                    params1.format = PixelFormat.TRANSLUCENT;

                    ly1 = new LinearLayout(context);
                    ly1.setBackgroundColor(Color.BLACK);
                    ly1.setOrientation(LinearLayout.VERTICAL);

                    wm.addView(ly1, params1);
                }

                // To remove the view once the dialer app is closed.
                if(arg1.getAction().equals("android.intent.action.PHONE_STATE")){
                    String state = arg1.getStringExtra(TelephonyManager.EXTRA_STATE);
                    if(state.equals(TelephonyManager.EXTRA_STATE_IDLE)){
                        WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
                        if(ly1!=null)
                        {
                            wm.removeView(ly1);
                            ly1 = null;
                        }
                    }
                }
            }

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

        ly1 = new LinearLayout(getApplicationContext());
        ly1.setOrientation(LinearLayout.HORIZONTAL);


        View hiddenInfo = getLayoutInflater().inflate(R.layout.layout1, ly1, false);
        ly1.addView(hiddenInfo);

        wm.addView(ly1, params1);

PS: Layout1 - это макет, который необходимо создать в папке макета и сослаться на него здесь.

Кроме того, в манифесте необходимо указать следующие разрешения.

<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"></uses-permission>
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.MODIFY_PHONE_STATE" />
<action android:name="android.intent.action.PHONE_STATE" /> (within intent filter of Broadcast Receiver)

Попробуй это

AlertDialog.Builder builder = new AlertDialog.Builder(context.getApplicationContext());
            LayoutInflater inflater = LayoutInflater.from(context);
            View dialogView = inflater.inflate(R.layout.caller_dialog, null);

            ImageView button = dialogView.findViewById(R.id.close_btn);

            builder.setView(dialogView);
            final AlertDialog alert = builder.create();
            alert.getWindow().requestFeature(Window.FEATURE_NO_TITLE);
            alert.getWindow().setType(WindowManager.LayoutParams.TYPE_PHONE);
            alert.setCanceledOnTouchOutside(true);
            alert.show();
            WindowManager.LayoutParams lp = new WindowManager.LayoutParams();
            Window window = alert.getWindow();
            window.addFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE);
            window.clearFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND);
            window.setGravity(Gravity.TOP);
            lp.copyFrom(window.getAttributes());
            //This makes the dialog take up the full width
            lp.width = WindowManager.LayoutParams.MATCH_PARENT;
            lp.height = WindowManager.LayoutParams.WRAP_CONTENT;
            window.setAttributes(lp);
            button.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View view) {
                    //close the service and remove the from from the window
                    alert.dismiss();
                }
            });

И добавьте проверку прав доступа в вашем onCreate()

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M && !Settings.canDrawOverlays(getActivity())) {
        //If the draw over permission is not available open the settings screen
        //to grant the permission.
        Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
                Uri.parse("package:" + getActivity().getPackageName()));
        startActivityForResult(intent, CODE_DRAW_OVER_OTHER_APP_PERMISSION);
    }

И onActivityResult()

private static final int CODE_DRAW_OVER_OTHER_APP_PERMISSION = 2084; 
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (requestCode == CODE_DRAW_OVER_OTHER_APP_PERMISSION) {

        //Check if the permission is granted or not.
        if (resultCode == RESULT_OK) {

        }
        else
            {
                // Permission is not available
            }

    } else {
        super.onActivityResult(requestCode, resultCode, data);
    }
}

И не забудьте добавить разрешение в манифест

<uses-permission android:name="android.permission.ACTION_MANAGE_OVERLAY_PERMISSION" />
Другие вопросы по тегам