在 Android 原生来电屏幕上弹出窗口,如真正的来电者 Android 应用程序

Posted

技术标签:

【中文标题】在 Android 原生来电屏幕上弹出窗口,如真正的来电者 Android 应用程序【英文标题】:Pop up window over Android native incoming call screen like true caller Android app 【发布时间】:2013-03-19 00:22:43 【问题描述】:

我正在为 android 中的来电开发一个广播接收器,在收到来电时,我想在原生来电屏幕上弹出一个弹出窗口。

我完成了那个代码。但现在的问题是,在Android 4.1 (Jelly Bean) API level 17 中,当电话响起时,PHONE_STATEOFF HOOK 的形式出现,如果我正在调用一个活动,它会被调用,但它下面的代码不会被执行。我列出了代码:

我的广播接收器

package com.example.popwindowonincomingcallscreen;

import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.telephony.TelephonyManager;
import android.util.Log;

public class IncomingBroadcastReceiver extends BroadcastReceiver 

    @Override
    public void onReceive(Context context, Intent intent) 

        Log.d("IncomingBroadcastReceiver: onReceive: ", "flag1");

        String state = intent.getStringExtra(TelephonyManager.EXTRA_STATE);
        Log.d("IncomingBroadcastReceiver: onReceive: ", state);
        if (state.equals(TelephonyManager.EXTRA_STATE_RINGING)
                || state.equals(TelephonyManager.EXTRA_STATE_OFFHOOK)) 

            Log.d("Ringing", "Phone is ringing");

            Intent i = new Intent(context, IncomingCallActivity.class);
            i.putExtras(intent);
            i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            i.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
            Wait.oneSec();
            context.startActivity(i);
        
    

我正在调用的活动:

import android.app.Activity;
import android.os.Bundle;
import android.telephony.TelephonyManager;
import android.util.Log;
import android.view.View.MeasureSpec;
import android.view.Window;
import android.view.WindowManager;
import android.widget.TextView;

public class IncomingCallActivity extends Activity 

    @Override
    protected void onCreate(Bundle savedInstanceState) 

        try 
            Log.d("IncomingCallActivity: onCreate: ", "flag2");

            */ After this line, the code is not executed in Android 4.1 (Jelly Bean) only/*

            // TODO Auto-generated method stub
            super.onCreate(savedInstanceState);

            getWindow().addFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE);
            getWindow().addFlags(
                    WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL);

            Log.d("IncomingCallActivity: onCreate: ", "flagy");

            setContentView(R.layout.main);

            Log.d("IncomingCallActivity: onCreate: ", "flagz");

            String number = getIntent().getStringExtra(
                    TelephonyManager.EXTRA_INCOMING_NUMBER);
            TextView text = (TextView) findViewById(R.id.text);
            text.setText("Incoming call from " + number);
         
        catch (Exception e) 
            Log.d("Exception", e.toString());
            // TODO Auto-generated catch block
            e.printStackTrace();
        
    

之后

try 
    Log.d("IncomingCallActivity: onCreate: ", "flag2");

该代码未在 Android 4.1 (Jelly Bean) 中执行,但在其他版本中运行。

我已经尝试了几乎所有我能做到的方法。此代码在本机呼叫屏幕上显示半透明活动,并且不会阻止背景控件,例如拿起电话。但我希望它像真正的来电者一样。我附上了关于真正来电者如何在来电屏幕上显示窗口的快照。

如何为 Android 应用实现此功能?

这就是真正的调用者的工作方式:

我现在的输出:

更新 1

在赏金之后,我也没有得到我想要的东西,但我会回复所有的;我正在努力。无论如何,此代码适用于大多数 Android 手机。如果有人要使用并抓住它的解决方案,请写在这里,以便每个人都能受益。

更新 2

我尝试在广播接收器的 onReceive 方法中实现 Toast,因为 toast 是 Android 的原生组件,但它也不会在 Android 4.1 (Jelly Bean) 中显示。

我的想法是在广播接收器的 onReceive 方法中实现 Toast,然后根据我们的需要更改其设计并调整其显示持续时间。但是还有一个问题是 findViewById 在广播接收器中不起作用,所以我认为我们必须以编程方式制作一个 LinearLayout 来自定义 toast。

【问题讨论】:

它阻止后台控制,如接听/拒绝呼叫。我也做过类似的事情。 你是怎么解决这个问题的???? 该线程的解决方案有效,但上层视图隐藏了某些设备上的呼叫控制按钮。有人对此有任何解决方案吗??? 我在使用你的代码,但接收器没有调用.. 为什么会这样 你能定义清单吗..我也在做同样的事情 【参考方案1】:

我不确定您的自定义 GUI 是否始终位于默认 GUI 之上,因为系统广播接收器和您的接收器都试图在屏幕顶部显示其 GUI。我们不确定首先调用哪一个,但是让您的 GUI 显示在屏幕顶部的一项棘手工作是当电话响起时,在使用处理程序的 1-2 秒后调用您的活动。

new Handler().postDelayed(new Runnable() 

     @Override
     public void run() 
         // TODO Auto-generated method stub
         Intent intent = new Intent(context, AcceptReject.class);
         intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
         context.startActivity(intent);
     
 , 2000);

希望对你有帮助。

【讨论】:

我试图在 3 4 5 秒后调用 mu 活动,但它也不工作。 我将这个赏金奖励给你,因为你对这个问题的思考正朝着正确的方向发展。它需要一些延迟,但这不适用于果冻豆请帮助我。 我试图在通话断开后关闭弹出窗口,但无法成功。【参考方案2】:

尝试super.onCreate 方法之前的代码。我认为在调用 super 之后代码会被跳过。有时这种技巧对我有用。

【讨论】:

不,我之前在创建时没有尝试过。我会尝试的。但是在 super.oncreate() 之前写它的原因是什么。 我会尝试但背后的原因。为什么你认为它会起作用。 没有理由.. 我知道这听起来很疯狂,但以前对我有用(但不是这个问题) 我会在星期六尝试,因为我现在正在休假。 一切顺利。我认为您在 try 块中调用 super 方法的另一种可能性对吗?如果超级导致异常并且您的块正在捕获它怎么办。那么剩下的代码就不会执行了吧?所以尝试重新排列顺序,只需尝试调用 super.onCreate(savedInstanceState);在您尝试阻止并检查您的日志是否正在打印之前。【参考方案3】:

我刚刚在Android 4.2 (Jelly Bean) 模拟器上进行了测试,它可以完美地阻止整个来电屏幕,就像 truecaller 一样:

public void onReceive(Context context, Intent intent) 

    WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);

    WindowManager.LayoutParams params = 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);

    params.height = LayoutParams.MATCH_PARENT;
    params.width = LayoutParams.MATCH_PARENT;
    params.format = PixelFormat.TRANSLUCENT;

    params.gravity = Gravity.TOP;

    LinearLayout ly = new LinearLayout(context);
    ly.setBackgroundColor(Color.RED);
    ly.setOrientation(LinearLayout.VERTICAL);

    wm.addView(ly, params);

在清单中:

<receiver android:name=""  android:enabled="true" >
    <intent-filter android:priority="-1">
        <action android:name="android.intent.action.PHONE_STATE" />
    </intent-filter>
</receiver>

【讨论】:

你能放一个屏幕截图吗,因为在我的果冻豆里它不起作用。 你可以在 google play 上查看 Callurgency 的截图。我就是这样做的,它适用于果冻豆 4.2 我已遵循您的代码。你能告诉我在哪里以及如何从 windowManager 中删除此布局吗? 考虑添加权限“android.permission.SYSTEM_ALERT_WINDOW”,以防遇到“无法添加窗口 android.view.ViewRoot$W@6292365 — 此窗口类型的权限被拒绝”错误 很好的解决方案伙伴,只有一个问题,如何通过通话挂断事件删除它?【参考方案4】:

我也在努力(我在​​这里理解你可能是错误的)。您想要实现的是在 Android 4.2 (Jelly Bean) 中显示该活动。我只是延迟显示活动。我在不同的班级中使用过 PhoneStateListener。我能够在呼叫者屏幕上显示新活动。这是我的完整代码:

文件 MyBroadcastReceiver.java

public class MyBroadcastReceiver extends BroadcastReceiver 
    static CustomPhoneStateListener phoneStateListener;
    Context context;
    Intent intent;

    @Override
    public void onReceive(Context context, Intent intent) 
        this.context = context;
        this.intent = intent;
        // TODO Auto-generated method stub

            TelephonyManager telephonyManager = (TelephonyManager) context
                    .getSystemService(Context.TELEPHONY_SERVICE);           
            phoneStateListener = new CustomPhoneStateListener(context);
            telephonyManager.listen(phoneStateListener,
                    PhoneStateListener.LISTEN_CALL_STATE);
    

文件 CustomPhoneStateListener.java

public class CustomPhoneStateListener extends PhoneStateListener 

    // private static final String TAG = "PhoneStateChanged";
    Context context; // Context to make Toast if required
    private AudioManager amanager;
    Intent i1;

    public CustomPhoneStateListener(Context context) 
        super();
        this.context = context;
        i1 = new Intent(context, YourActivity.class);       
        i1.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        i1.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);

    

    @Override
    public void onCallStateChanged(int state, String incomingNumber) 
        super.onCallStateChanged(state, incomingNumber);

        switch (state) 
        case TelephonyManager.CALL_STATE_IDLE:
            Toast.makeText(context, "Phone state Idle", Toast.LENGTH_LONG)
                    .show();

            break;
        case TelephonyManager.CALL_STATE_OFFHOOK:

            Toast.makeText(context, "Phone state Off hook", Toast.LENGTH_LONG)
                    .show();

            break;
        case TelephonyManager.CALL_STATE_RINGING:           
            try 
                Thread.sleep(3000);
                context.startActivity(i1);              
             catch (Exception e) 
                e.getLocalizedMessage();
            

        default:
            break;
        
    

YourActivity 将保持您创建的状态... 注意:我在这段代码中也遇到了一些问题。

    当呼叫关闭时(未接来电或被拒绝),活动不会被关闭。 我无法点击 Activity(我想为我的应用添加一个按钮) 它只在第一次工作。当我第二次拨打电话时,我的应用程序停止了(我认为这是因为呼叫被解除时 Activity 没有被关闭)

(接受这些问题的帮助。谢谢。可能会帮助一些人)

更新

这里是SMALL DEMO 的链接如何实现这一点。

    当呼叫关闭时(未接来电或被拒绝),活动不会被关闭。 - 解决了 我无法点击 Activity(我想为我的应用添加一个按钮) - 已解决 它只在第一次工作。当我第二次拨打电话时,我的应用程序停止了(我认为这是因为呼叫被解除时 Activity 没有被关闭) - 已解决

【讨论】:

谢谢..我无法在最新的 android 4.4.X 上运行这个..请帮助 什么问题??对话框不显示或监听器不工作?。 嗨 Dharmik,我知道线程已经很老了。但是你想分享代码吗.. 因为我可以在呼叫屏幕上祝酒,但不能像你这样的文本框.. 请分享代码。当接到或拨打电话时,我正在尝试在屏幕上制作弹出屏幕。和 TrueCaller 一样请帮忙 @shivam.kotwalia 我已经添加了 git hub 链接。如果有任何问题,请告诉我, 嗨 Dharmik,谢谢 :) 如果我遇到问题,我一定会打扰你。【参考方案5】:

我认为你不应该开始活动来达到所描述的结果。您需要在其布局参数中设置 LayoutParams.TYPE_SYSTEM_OVERLAY 的单独视图。

您可以将此视图放置在屏幕上的任何位置,或者只是覆盖整个屏幕。

下面是几行代码:

 _av = new ActivatorView(this);
 _avLayoutParams = new WindowManager.LayoutParams(0, 0, 0, 0,
     WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
     WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE,
     PixelFormat.OPAQUE);
 _avLayoutParams.screenBrightness = _fScreenBrightness = 20f;

 WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
 wm.addView(_av, _avLayoutParams);

https://bitbucket.org/gyrussolutions/yaab/src/f01cc8aff690cae1b1107287cb17835b8a3c1643/src/biz/gyrus/yaab/LightMonitorService.java?at=default#cl-338 - 完整的源代码,将其视为示例。

【讨论】:

@Nikhil Agrawal 这是正确的方向。对于 2.x 到 4.2.x 版本,我有类似的工作没有失败。 (在我的应用程序 4W Job Manager 中查看屏幕截图)。您需要在您的 cmets 中更加具体:“不工作”不鼓励我们提供任何进一步的帮助。 LayoutParams.TYPE_SYSTEM_OVERLAY 不接受窗口上的输入事件。【参考方案6】:

我正在尝试类似的方法,在来电屏幕上添加一个额外的按钮。

Sam Adams 发布的答案对我有用,尽管我是从 PhoneStateListener 调用代码。除此之外,他的代码唯一真正的区别是我正在夸大布局:

overlay = (RelativeLayout) inflater.inflate(R.layout.overlay, null);
wm.addView(overlay, params);

它可以在模拟器和 HTC One S(运行 Android 4.1.1)上运行。

您需要记住的是保留对您正在添加的覆盖视图的引用,并在电话回到空闲状态时(当侦听器获取 TelephonyManager.CALL_STATE_IDLE 时再次删除它(在 windowmanager 实例上调用 removeView()) ),否则您的叠加层将保留在屏幕上。

        WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
    if(overlay!=null)
    
        wm.removeView(overlay);
        overlay = null;
    

【讨论】:

【参考方案7】:

我们也遇到了类似的问题,即覆盖没有显示在具有 pin 锁的设备上。对我们有用的解决方案如下:

mWindowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
    mParams = new LayoutParams(
        LayoutParams.MATCH_PARENT,
        LayoutParams.WRAP_CONTENT,
        LayoutParams.TYPE_SYSTEM_ERROR,
        LayoutParams.FLAG_NOT_FOCUSABLE,
        PixelFormat.TRANSLUCENT);

LayoutParams.TYPE_SYSTEM_ERROR 造成了不同。

【讨论】:

【参考方案8】:

试试这个

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();
                
            );

【讨论】:

它适用于所有版本吗?在这种情况下需要什么权限?【参考方案9】:

我的方法:

    使用接收器接收电话事件

    使用服务制作叠加层

    ps:wmParams.type = WindowManager.LayoutParams.TYPE_PHONE; 
    

【讨论】:

【参考方案10】:
new Handler().postDelayed(new Runnable() 
    @Override
    public void run() 
        // TODO Auto-generated method stub
        Intent i = new Intent(context, CallingIncoming.class);
        i.putExtras(intent);
        i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK /*| Intent.FLAG_ACTIVITY_CLEAR_TASK*/);
        context.startActivity(i);
    
, 450);//It will help you to delay your screen and after it your screen will be top of default app screen

【讨论】:

【参考方案11】:

使用简单的广播接收器并将这段代码放入广播接收器中:

public void onReceive(final Context context, final Intent intent) 

    Log.v(LOG_TAG, "Receieved notification about network status");
    WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);

    WindowManager.LayoutParams params = new WindowManager.LayoutParams(
            WindowManager.LayoutParams.MATCH_PARENT,
            WindowManager.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);

    params.height = WindowManager.LayoutParams.MATCH_PARENT;
    params.width = WindowManager.LayoutParams.MATCH_PARENT;
    params.format = PixelFormat.TRANSLUCENT;

    params.gravity = Gravity.TOP;

    LinearLayout ly = new LinearLayout(context);
    ly.setBackgroundColor(Color.RED);
    ly.setOrientation(LinearLayout.VERTICAL);

    wm.addView(ly, params);


【讨论】:

如何在 TYPE_SYSTEM_OVERLAY 上获得点击事件,覆盖上的按钮无法点击..请帮助

以上是关于在 Android 原生来电屏幕上弹出窗口,如真正的来电者 Android 应用程序的主要内容,如果未能解决你的问题,请参考以下文章

当我打开android studio时,如何摆脱我的MAC上弹出的xcodebuild弹出窗口?

将反应原生弹出窗口定位在屏幕底部

ASP.NET 如何在网页上弹出提示窗口?

推送通知不会在 Android 上弹出

在移动设备上弹出不工作 - wordpress

我的Qt学习笔记 4 如何在 QWidget 窗口上弹出右键菜单