安卓:初识Presentation(实现双屏异显,特殊的权限添加)️️

参考资料

链接: Android实现双屏异显.
链接: Android 6.0:Unable to add window android.view.ViewRootImpl$W@5e2d85a – permission denied.

公司项目后期需要设备支持双屏异显,家境贫寒的我表示压根没听过,赶紧码起来,自己建了一个小demo。

初识Presentation💻🖥️

我发现了Presentation,安卓官方对于Presentation是这样描述的

在这里插入图片描述

大致意思就是,是一种特殊的对话框,是为了在辅助显示器上演示内容。

快速上手

想要使用Presentation,首先得找到自己的辅助屏幕,这里我们使用了安卓官方推荐的第二种方式
创建一个activity,然后在里面写一个内部类继承于Presentation,当然我只是图方便,实际上肯定是要新建一个的

    private class DifferentDislay extends Presentation {
        public DifferentDislay(Context outerContext, Display display) {
            super(outerContext, display);
        }

        public DifferentDislay(Context outerContext, Display display, int theme) {
            super(outerContext, display, theme);
        }

        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            
            setContentView(R.layout.diffrentdisplay);
        }
    }

上例代码中
setContentView(R.layout.diffrentdisplay); 的布局就是你副屏的布局拉!布局我就不贴了
然后开始简单快乐的找副屏,找到过后显示出来

public class MainActivity extends AppCompatActivity {

    DisplayManager mDisplayManager;

    Display[] displays;

    public static int OVERLAY_PERMISSION_REQ_CODE = 99;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initPresentation();
    }

    private void initPresentation() {
        mDisplayManager = (DisplayManager) this.getSystemService(Context.DISPLAY_SERVICE);

        displays = mDisplayManager.getDisplays();

        DifferentDislay mPresentation = new DifferentDislay(this
                , displays[1]);

        mPresentation.getWindow().setType(

                WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);

        mPresentation.show();
    }

上例代码中

  • mDisplayManager 为屏幕管理类,帮助我们管理多个屏幕
  • displays为屏幕数组,就是我们找到的所有屏幕,displays[1] 就是副屏啦

然后就开始血泪填坑史,运行过后果断的闪退了,一看报错,哦,没有加权限嘛,小事,在AndroidManifest.xml中加上权限

    <!-- 显示系统窗口权限 -->
    <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>

ok,再次运行,果断又闪退💔,等等,我的设备是安卓7.0的,要动态申请,那就申请一个呗。可是我发现我的动态申请权限代码不起作用,最后发现这个权限要跳到单独的页面申请,完全代码如下:

懒人直达

public class MainActivity extends AppCompatActivity {

    DisplayManager mDisplayManager;//屏幕管理类

    Display[] displays;//屏幕数组

    public static int OVERLAY_PERMISSION_REQ_CODE = 99;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        requestDrawOverLays();
    }

    private void initPresentation() {
        mDisplayManager = (DisplayManager) this.getSystemService(Context.DISPLAY_SERVICE);

        displays = mDisplayManager.getDisplays();

        DifferentDislay mPresentation = new DifferentDislay(this
                , displays[1]);//displays[1]是副屏  如果只有一个屏幕使用displays[0]

        mPresentation.getWindow().setType(

                WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);

        mPresentation.show();
    }


    private class DifferentDislay extends Presentation {
        public DifferentDislay(Context outerContext, Display display) {
            super(outerContext, display);
        }

        public DifferentDislay(Context outerContext, Display display, int theme) {
            super(outerContext, display, theme);
        }

        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.diffrentdisplay);
        }
    }


    @TargetApi(Build.VERSION_CODES.M)
    public void requestDrawOverLays() {
        if (!Settings.canDrawOverlays(MainActivity.this)) {
            Toast.makeText(this, "您还没有打开悬浮窗权限", Toast.LENGTH_SHORT).show();
            //跳转到相应软件的设置页面
            Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:" + MainActivity.this.getPackageName()));
            startActivityForResult(intent, OVERLAY_PERMISSION_REQ_CODE);
        } else {
            // 授权成功之后执行的方法
            initPresentation();
        }
    }


    @TargetApi(Build.VERSION_CODES.M)
    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (requestCode == OVERLAY_PERMISSION_REQ_CODE) {
            if (!Settings.canDrawOverlays(this)) {
                Toast.makeText(this, "授权失败", Toast.LENGTH_SHORT).show();
            } else {
                Toast.makeText(this, "授权成功", Toast.LENGTH_SHORT).show();
                initPresentation();
            }
        }
    }


}

这样,一个简单的双屏异显demo就完成啦,共勉,别忘记加权限哦!

补充(TYPE_APPLICATION_OVERLAY)

后来的后来,我有一个同事要做类似的功能,我想起来有记录这部分的知识,就将这篇文章分享给她了,她使用了我的源码但是还是报了类似的没有权限的问题,这让我十分的不解,后面经过她的一番努力,发现8.0以上版本的安卓机需要将懒人直达代码中的

        mPresentation.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);

替换为

        mPresentation.getWindow().setType(WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY);

所以我们最终将代码更改为如下

 if (Build.VERSION.SDK_INT >= 26) {
                getWindow().setType(WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY);
            } else {
                getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
            }

算是一个版本的适配问题吧

评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我怀里的猫

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值