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

  • Post author:
  • Post category:其他




参考资料

链接:

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

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



版权声明:本文为shop_and_sleep原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。