Android多屏幕支持-Android12

Android多屏幕支持-Android12

  • 1、概览及相关文章
  • 2、屏幕窗口配置
    • 2.1 配置xml文件
    • 2.2 DisplayInfo#uniqueId 屏幕标识
    • 2.3 adb查看信息
  • 3、配置文件解析
    • 3.1 xml字段读取
    • 3.2 简要时序图
  • 4、每屏幕焦点

android12-release


1、概览及相关文章

AOSP > 文档 > 心主题 > 多屏幕概览

术语
在这些文章中,主屏幕和辅助屏幕的定义如下:

主(默认)屏幕的屏幕 IDDEFAULT_DISPLAY
辅助屏幕的屏幕 ID 不是 DEFAULT_DISPLAY

主题区域文章
开发和测试推荐做法
测试和开发环境
常见问题解答
相关文章集显示
系统装饰支持
输入法支持
单篇文章多项恢复
Activity 启动政策
锁定屏幕
输入路由
多区音频

2、屏幕窗口配置

2.1 配置xml文件

/data/system/display_settings.xml 配置:

  • 模拟屏幕uniqueId 用于在名称属性中标识屏幕,对于模拟屏幕,此 ID 为 overlay:1
<?xml version='1.0' encoding='utf-8' standalone='yes' ?>
<display-settings>
<config identifier="0" />
<display
  name="overlay:1"
  shouldShowSystemDecors="true"
  shouldShowIme="true" />
</display-settings>
  • 内置屏幕uniqueId 对于内置屏幕,示例值可以是 “local:45354385242535243453”。另一种方式是使用硬件端口信息,并设置 identifier=“1” 以与 DisplayWindowSettingsProvider#IDENTIFIER_PORT 对应,然后更新 name 以使用 "port:<port_id>" 格式
<?xmlversion='1.0' encoding='utf-8' standalone='yes' ?>
<display-settings>
<config identifier="1" />
<display
  name="port:12345"
  shouldShowSystemDecors="true"
  shouldShowIme="true" />
</display-settings>

2.2 DisplayInfo#uniqueId 屏幕标识

DisplayInfo#uniqueId,以添加稳定的标识符并区分本地、网络和虚拟屏幕

屏幕类型格式
本地local:<stable-id>
网络network:<mac-address>
虚拟virtual:<package-name-and-name>

2.3 adb查看信息

$ dumpsys SurfaceFlinger --display-id
# Example output.
Display 21691504607621632 (HWC display 0): port=0 pnpId=SHP displayName="LQ123P1JX32"
Display 9834494747159041 (HWC display 2): port=1 pnpId=HWP displayName="HP Z24i"
Display 1886279400700944 (HWC display 1): port=2 pnpId=AUS displayName="ASUS MB16AP"

frameworks/native/services/surfaceflinger/SurfaceFlinger.cpp#SurfaceFlinger::dumpDisplayIdentificationData

void SurfaceFlinger::dumpDisplayIdentificationData(std::string& result) const {
    for (const auto& [token, display] : mDisplays) {
        const auto displayId = PhysicalDisplayId::tryCast(display->getId());
        if (!displayId) {
            continue;
        }
        const auto hwcDisplayId = getHwComposer().fromPhysicalDisplayId(*displayId);
        if (!hwcDisplayId) {
            continue;
        }

        StringAppendF(&result,
                      "Display %s (HWC display %" PRIu64 "): ", to_string(*displayId).c_str(),
                      *hwcDisplayId);
        uint8_t port;
        DisplayIdentificationData data;
        if (!getHwComposer().getDisplayIdentificationData(*hwcDisplayId, &port, &data)) {
            result.append("no identification data\n");
            continue;
        }

        if (!isEdid(data)) {
            result.append("unknown identification data\n");
            continue;
        }

        const auto edid = parseEdid(data);
        if (!edid) {
            result.append("invalid EDID\n");
            continue;
        }

        StringAppendF(&result, "port=%u pnpId=%s displayName=\"", port, edid->pnpId.data());
        result.append(edid->displayName.data(), edid->displayName.length());
        result.append("\"\n");
    }
}

在这里插入图片描述

3、配置文件解析

3.1 xml字段读取

  • 文件路径:DATA_DISPLAY_SETTINGS_FILE_PATH = "system/display_settings.xml"VENDOR_DISPLAY_SETTINGS_FILE_PATH = "etc/display_settings.xml"Settings.Global.getString(resolver,DEVELOPMENT_WM_DISPLAY_SETTINGS_PATH)DEVELOPMENT_WM_DISPLAY_SETTINGS_PATH = "wm_display_settings_path"
  • FileData对象:fileData.mIdentifierType = getIntAttribute(parser, "identifier", IDENTIFIER_UNIQUE_ID)name = parser.getAttributeValue(null, "name")shouldShowIme = getBooleanAttribute(parser, "shouldShowIme", null /* defaultValue */)settingsEntry.mShouldShowSystemDecors = getBooleanAttribute(parser, "shouldShowSystemDecors", null /* defaultValue */)等等
private static final class FileData {
  int mIdentifierType;
  final Map<String, SettingsEntry> mSettings = new HashMap<>();

  @Override
  public String toString() {
      return "FileData{"
              + "mIdentifierType=" + mIdentifierType
              + ", mSettings=" + mSettings
              + '}';
  }
}
  • DisplayWindowSettings.java有关显示器的当前持久设置。提供显示设置的策略,并将设置值的持久性和查找委派给提供的{@link SettingsProvider}
@Nullable
private static FileData readSettings(ReadableSettingsStorage storage) {
    InputStream stream;
    try {
        stream = storage.openRead();
    } catch (IOException e) {
        Slog.i(TAG, "No existing display settings, starting empty");
        return null;
    }
    FileData fileData = new FileData();
    boolean success = false;
    try {
        TypedXmlPullParser parser = Xml.resolvePullParser(stream);
        int type;
        while ((type = parser.next()) != XmlPullParser.START_TAG
                && type != XmlPullParser.END_DOCUMENT) {
            // Do nothing.
        }

        if (type != XmlPullParser.START_TAG) {
            throw new IllegalStateException("no start tag found");
        }

        int outerDepth = parser.getDepth();
        while ((type = parser.next()) != XmlPullParser.END_DOCUMENT
                && (type != XmlPullParser.END_TAG || parser.getDepth() > outerDepth)) {
            if (type == XmlPullParser.END_TAG || type == XmlPullParser.TEXT) {
                continue;
            }

            String tagName = parser.getName();
            if (tagName.equals("display")) {
                readDisplay(parser, fileData);
            } else if (tagName.equals("config")) {
                readConfig(parser, fileData);
            } else {
                Slog.w(TAG, "Unknown element under <display-settings>: "
                        + parser.getName());
                XmlUtils.skipCurrentTag(parser);
            }
        }
        success = true;
    } catch (IllegalStateException e) {
        Slog.w(TAG, "Failed parsing " + e);
    } catch (NullPointerException e) {
        Slog.w(TAG, "Failed parsing " + e);
    } catch (NumberFormatException e) {
        Slog.w(TAG, "Failed parsing " + e);
    } catch (XmlPullParserException e) {
        Slog.w(TAG, "Failed parsing " + e);
    } catch (IOException e) {
        Slog.w(TAG, "Failed parsing " + e);
    } catch (IndexOutOfBoundsException e) {
        Slog.w(TAG, "Failed parsing " + e);
    } finally {
        try {
            stream.close();
        } catch (IOException ignored) {
        }
    }
    if (!success) {
        fileData.mSettings.clear();
    }
    return fileData;
}

private static int getIntAttribute(TypedXmlPullParser parser, String name, int defaultValue) {
    return parser.getAttributeInt(null, name, defaultValue);
}

@Nullable
private static Integer getIntegerAttribute(TypedXmlPullParser parser, String name,
        @Nullable Integer defaultValue) {
    try {
        return parser.getAttributeInt(null, name);
    } catch (Exception ignored) {
        return defaultValue;
    }
}

@Nullable
private static Boolean getBooleanAttribute(TypedXmlPullParser parser, String name,
        @Nullable Boolean defaultValue) {
    try {
        return parser.getAttributeBoolean(null, name);
    } catch (Exception ignored) {
        return defaultValue;
    }
}

private static void readDisplay(TypedXmlPullParser parser, FileData fileData)
        throws NumberFormatException, XmlPullParserException, IOException {
    String name = parser.getAttributeValue(null, "name");
    if (name != null) {
        SettingsEntry settingsEntry = new SettingsEntry();
        settingsEntry.mWindowingMode = getIntAttribute(parser, "windowingMode",
                WindowConfiguration.WINDOWING_MODE_UNDEFINED /* defaultValue */);
        settingsEntry.mUserRotationMode = getIntegerAttribute(parser, "userRotationMode",
                null /* defaultValue */);
        settingsEntry.mUserRotation = getIntegerAttribute(parser, "userRotation",
                null /* defaultValue */);
        settingsEntry.mForcedWidth = getIntAttribute(parser, "forcedWidth",
                0 /* defaultValue */);
        settingsEntry.mForcedHeight = getIntAttribute(parser, "forcedHeight",
                0 /* defaultValue */);
        settingsEntry.mForcedDensity = getIntAttribute(parser, "forcedDensity",
                0 /* defaultValue */);
        settingsEntry.mForcedScalingMode = getIntegerAttribute(parser, "forcedScalingMode",
                null /* defaultValue */);
        settingsEntry.mRemoveContentMode = getIntAttribute(parser, "removeContentMode",
                REMOVE_CONTENT_MODE_UNDEFINED /* defaultValue */);
        settingsEntry.mShouldShowWithInsecureKeyguard = getBooleanAttribute(parser,
                "shouldShowWithInsecureKeyguard", null /* defaultValue */);
        settingsEntry.mShouldShowSystemDecors = getBooleanAttribute(parser,
                "shouldShowSystemDecors", null /* defaultValue */);
        final Boolean shouldShowIme = getBooleanAttribute(parser, "shouldShowIme",
                null /* defaultValue */);
        if (shouldShowIme != null) {
            settingsEntry.mImePolicy = shouldShowIme ? DISPLAY_IME_POLICY_LOCAL
                    : DISPLAY_IME_POLICY_FALLBACK_DISPLAY;
        } else {
            settingsEntry.mImePolicy = getIntegerAttribute(parser, "imePolicy",
                    null /* defaultValue */);
        }
        settingsEntry.mFixedToUserRotation = getIntegerAttribute(parser, "fixedToUserRotation",
                null /* defaultValue */);
        settingsEntry.mIgnoreOrientationRequest = getBooleanAttribute(parser,
                "ignoreOrientationRequest", null /* defaultValue */);
        settingsEntry.mIgnoreDisplayCutout = getBooleanAttribute(parser,
                "ignoreDisplayCutout", null /* defaultValue */);
        settingsEntry.mDontMoveToTop = getBooleanAttribute(parser,
                "dontMoveToTop", null /* defaultValue */);

        fileData.mSettings.put(name, settingsEntry);
    }
    XmlUtils.skipCurrentTag(parser);
}

private static void readConfig(TypedXmlPullParser parser, FileData fileData)
        throws NumberFormatException,
        XmlPullParserException, IOException {
    fileData.mIdentifierType = getIntAttribute(parser, "identifier",
            IDENTIFIER_UNIQUE_ID);
    XmlUtils.skipCurrentTag(parser);
}

3.2 简要时序图

在这里插入图片描述

4、每屏幕焦点

每个屏幕焦点

为了同时支持多个以单个屏幕为目标的输入源,可以将 Android 10 配置为支持多个聚焦窗口,每个屏幕最多支持一个。当多个用户同时与同一设备交互并使用不同的输入方法或设备(例如 Android Automotive)时,此功能仅适用于特殊类型的设备。

强烈建议不要为常规设备启用此功能,包括跨屏设备或用于类似桌面设备体验的设备。这主要是出于安全方面的考虑,因为这样做可能会导致用户不确定哪个窗口具有输入焦点

想象一下,用户在文本输入字段中输入安全信息,也许是登录某个银行应用或者输入包含敏感信息的文本。恶意应用可以创建一个虚拟的屏幕外屏幕用于执行 activity,也可以使用文本输入字段执行 activity。合法 activity 和恶意 activity 均具有焦点,并且都显示一个有效的输入指示符(闪烁光标)。

不过,键盘(硬件或软件)的输入只能进入最顶层的 activity(最近启动的应用)。通过创建隐藏的虚拟屏幕,即使在主设备屏幕上使用软件键盘,恶意应用也可以获取用户输入。

使用 com.android.internal.R.bool.config_perDisplayFocusEnabled 设置每屏幕焦点。

兼容性
**问题:**在 Android 9 及更低版本中,系统中一次最多只有一个窗口具有焦点。

**解决方案:**在极少数情况下,来自同一进程的两个窗口都处于聚焦状态,则系统仅向在 Z 轴顺序中较高的窗口提供焦点。对于以 Android 10 为目标平台的应用,目前已取消这一限制,此时预计这些应用可以支持同时聚焦多个窗口。

实现
WindowManagerService#mPerDisplayFocusEnabled 用于控制此功能的可用性。在 ActivityManager 中,系统现在使用的是 ActivityDisplay#getFocusedStack(),而不是利用变量进行全局跟踪。ActivityDisplay#getFocusedStack() 根据 Z 轴顺序确定焦点,而不是通过缓存值来确定。这样一来,只有一个来源 WindowManager 需要跟踪 activity 的 Z 轴顺序。

如果必须要确定系统中最顶层的聚焦堆栈,ActivityStackSupervisor#getTopDisplayFocusedStack() 会采用类似的方法处理这些情况。系统将从上到下遍历这些堆栈,搜索第一个符合条件的堆栈。

InputDispatcher 现在可以有多个聚焦窗口(每个屏幕一个)。如果某个输入事件特定于屏幕,则该事件会被分派到相应屏幕中的聚焦窗口。否则,它会被分派到聚焦屏幕(即用户最近与之交互的屏幕)中的聚焦窗口。

请参阅 InputDispatcher::mFocusedWindowHandlesByDisplay 和 InputDispatcher::setFocusedDisplay()。聚焦应用也会通过 NativeInputManager::setFocusedApplication()InputManagerService 中分别更新。

WindowManager 中,系统还会单独跟踪聚焦窗口。请参阅 DisplayContent#mCurrentFocusDisplayContent#mFocusedApp 以及各自的用途。相关的焦点跟踪和更新方法已从 WindowManagerService 移至 DisplayContent

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/72879.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Mac M2 Pro安装使用Cocoapods

Mac Pro M2安装使用Cocoapods 在新公司要做iOS开发&#xff0c;所以在新电脑上安装Cocoapods 在升级gem&#xff0c;sudo gem update --system&#xff0c;和安装cocoapods时都遇到如下的提示&#xff1a; ERROR: While executing gem ... (Errno::EPERM)Operation not per…

关于使用pycharm遇到只能使用unittest方式运行,无法直接选择Run

相信大家可能都遇到过这个问题&#xff0c;使用pycharm直接运行脚本的时候&#xff0c;只能选择unittest的方式&#xff0c;能愁死个人 经过几次各种尝试无果之后&#xff0c;博主就放弃死磕了&#xff0c;原谅博主是个菜鸟 后来遇到这样的问题&#xff0c;往往也就直接使用cm…

最新Anaconda安装-保姆级教程

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 随着人工智能的不断发展&#xff0c;机器学习这门技术也越来越重要&#xff0c;很多…

Linux源码剖析匿名共享内存shmem原理

如下问题如果都清楚了就不用看本文了&#xff1a; 1. shmem ram文件系统的初始化流程是怎样的 2. shmem思想上想复用基于文件的操作流程&#xff0c;实现上shmem也引入了一个文件&#xff0c;那么类似文件open会生成struct file&#xff0c;shmem的struct file怎么生成的 3.…

5款鲜为人知的国产办公软件,可私有化部署,保障内部数据安全

在企业内部办公环境中&#xff0c;如何保障内部数据安全是一项至关重要的事项。为了应对这个挑战&#xff0c;越来越多的企业选择使用私有化部署的办公软件来确保数据的自主性和安全性。下面我将分享5款鲜为人知的国产办公软件&#xff0c;均支持私有化部署&#xff0c;保障企业…

代理设计模式——静态代理和动态代理

代理模式 在代理模式&#xff08;Proxy Pattern&#xff09;中&#xff0c;一个类代表另一个类的功能。这种类型的设计模式属于结构型模式&#xff0c;在代理模式中&#xff0c;我们创建具有现有对象的对象&#xff0c;以便向外界提供功能接口。 意图&#xff1a;为其他对象提…

limereport报表使用

在这里我使用报表是以报表的形式显示数据库的信息。所以首先需要准备的资料有&#xff1a;limereport源码&#xff0c;还有数据库&#xff0c;我这里使用的是qsqlite数据库。 1、下载limereport报表源码 2、运行自带的案例&#xff1a;demo_r1 3、点击 “Run Report Designer”…

PDF文件限制编辑怎么取消?

PDF文件设置了限制编辑&#xff0c;想要取消PDF文件的限制编辑&#xff0c;很简单&#xff0c;打开PDF编辑器&#xff0c;点击工具栏中的文件&#xff0c;选择属性按钮&#xff0c;进入到熟悉感界面之后&#xff0c;点击安全&#xff0c;然后我们点击权限下拉框&#xff0c;选择…

如何使用Spark/Flink等分布式计算引擎做网络入侵检测

如何使用Spark/Flink等分布式计算引擎做网络入侵检测 引言16 Distributed Abnormal Behavior Detection Approach Based on Deep Belief Network and Ensemble SVM Using Spark17 Spark configurations to optimize decision tree classification on UNSW-NB1518 A dynamic spa…

【Pytroch】基于支持向量机算法的数据分类预测(Excel可直接替换数据)

【Pytroch】基于支持向量机算法的数据分类预测&#xff08;Excel可直接替换数据&#xff09; 1.模型原理2.数学公式3.文件结构4.Excel数据5.下载地址6.完整代码7.运行结果 1.模型原理 支持向量机&#xff08;Support Vector Machine&#xff0c;SVM&#xff09;是一种强大的监…

换架 3D 飞机,继续飞呀飞

相信大多数图扑 HT 用户都曾见过这个飞机的 Demo&#xff0c;在图扑发展的这十年&#xff0c;这个 Demo 是许多学习 HT 用户一定会参考的经典 Demo 之一。 这个 Demo 用简洁的代码生动地展示了 OBJ 模型加载、数据绑定、动画和漫游等功能的实现。许多用户参考这个简单的 Demo 后…

Redis进阶(4)——结合redis.conf配置文件深入理解 Redis两种数据持久化方案:RDB和AOF

目录 引出持久化方案RDBAOF Redis的持久化方案RDB如果采用docker stop关闭如果采用强制关闭 AOF参数设置混编方式的加载让aof进行重写 两种持久化方案的优缺点AOF优缺点RDB优势和劣势 总结 引出 1.Redis数据持久化的两种方式&#xff0c;RDB和AOF; 2.RDB采用二进制存储&#xf…

QT报表Limereport v1.5.35编译及使用

1、编译说明 下载后QT CREATER中打开limereport.pro然后直接编译就可以了。编译后结果如下图&#xff1a; 一次编译可以得到库文件和DEMO执行程序。 2、使用说明 拷贝如下图编译后的lib目录到自己的工程目录中。 release版本的重新命名为librelease. PRO文件中配置 QT …

抖音小程序实现less语言编译样式

1.在抖音开发工具中搜索扩展less 2. 然后点击小齿轮选择扩展设置 3. 然后在扩展设置中选择在settings.json中编辑# 4. 在settings.json中加入以下这段代码即可 // Easy LESS配置"less.compile": {"compress": false,//是否压缩"sourceMap": fal…

virtualBox桥接模式下openEuler镜像修改IP地址、openEule修改IP地址、openEule设置IP地址

安装好openEuler后,设置远程登入前,必不可少的一步,主机与虚拟机之间的通信要解决,下面给出详细步骤: 第一步:检查虚拟机适配器模式:桥接模式 第二步:登入虚拟机修改IP cd /etc/sysconfig/network-scripts vim ifcfg-enpgs3 没有vim的安装或者用vi代替:sudo dnf …

优化开发体验:掌握VSCode配置Vue模板的技巧与方法

前言 当你使用了 VSCode 配置 vue 模版时&#xff0c;你会发现它就像是一位贴心的助手&#xff0c;本文将为大家介绍如何使用 VSCode 配置 vue 模板&#xff0c;让你的代码更加高效、美观。下面让我们一起来看看吧。 一、打开 VSCode 控制台 文件 --> 首选项 --> 用户片段…

HOT92-最小路径和

leetcode原题链接&#xff1a;最小路径和 题目描述 给定一个包含非负整数的 m x n 网格 grid &#xff0c;请找出一条从左上角到右下角的路径&#xff0c;使得路径上的数字总和为最小。 说明&#xff1a;每次只能向下或者向右移动一步。 示例 1&#xff1a; 输入&#xff1a;…

运维工程师常见面试题

1、http常见返回码 2、mysql的同步方式 1&#xff09;异步复制 MySQL默认的复制即是异步的&#xff0c;主库在执行完客户端提交的事务后会立即将结果返给给客户端&#xff0c;并不关心从库是否已经接收并处理&#xff0c;这样就会有一个问题&#xff0c;主如果crash掉了&a…

c++11 标准模板(STL)(std::basic_stringbuf)(二)

定义于头文件 <sstream> template< class CharT, class Traits std::char_traits<CharT>, class Allocator std::allocator<CharT> > class basic_stringbuf : public std::basic_streambuf<CharT, Traits> std::basic_stringbuf…

拒绝摆烂!C语言练习打卡第一天

&#x1f525;博客主页&#xff1a;小王又困了 &#x1f4da;系列专栏&#xff1a;每日一练 &#x1f31f;人之为学&#xff0c;不日近则日退 ❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ &#x1f5d2;️前言&#xff1a; 在前面我们学习完C语言的所以知识&#xff0c;当…