Android使用Jetpack WindowManager来开发可折叠设备的探索

本文主要是介绍Android使用Jetpack WindowManager来开发可折叠设备的探索,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、背景

我们在Google开发者大会上,看到Jetpack WindowManager和WindowSizeClass这些技术,如下图。

那这里不得不说折叠屏手机了

  • 在其中一个显示区域中运行一个应用。

  • 同时运行两个应用,各位于一个显示区域中(在 multi-window 模式下)。

  • 可折叠设备还支持不同的折叠状态。折叠状态可用来以不同的方式显示内容

折叠屏手机有哪些优势呢?

  • 更大的屏幕空间

  • 更高的灵活性

  • 更好的移动性

  • 更高的科技含量

随着人们对生活品质的越来越高,单屏幕手机已经难以满足部分需求,折叠屏手机也越来越受欢迎,能给用户带来与众不同的体验,Jetpack WindowManager和WindowSizeClass就是折叠屏手机开发所需要的技术,有了这些技术,可以帮我们更好的开发折叠屏。

二、Jetpack WindowManager的简单了解

WindowManager的功能,包括对响应式UI的支持、检测屏幕变化的回调适配器以及窗口测试API。但Jetpack WindowManager还提供了对新型设备的支持,如可折叠设备和Chrome OS等窗口环境,是一个现代的、以Kotlin为首的库,它支持新的设备形态因素,并提供 "类似AppCompat "的功能,以构建具有响应式用户界面的应用程序。

主要的API

新的WindowManager APIs包括以下内容。

WindowLayoutInfo:包含了一个窗口的显示特征,例如窗口是否包含了折叠或铰链

FoldingFeature:使你能够监测可折叠设备的折叠状态,以确定设备的姿势

WindowMetrics:提供当前窗口的指标或整体显示的指标

Jetpack WindowManager与安卓系统没有捆绑,允许更快地迭代API,以支持快速发展的设备市场,并使应用程序开发人员能够采用库的更新,而不必等待最新的安卓版本。

三、Jetpack WindowManager的简单使用

首先我们需要到官网先下载好Android studio,把开发环境搞好。

1)、创建一个新项目

加入部分依赖

dependencies {ext.windowmanager_version = "1.0.0"implementation "androidx.window:window:$windowmanager_version"androidTestImplementation "androidx.window:window-testing:$windowmanager_version"// Needed to use lifecycleScope to collect the WindowLayoutInfo flowimplementation "androidx.lifecycle:lifecycle-runtime-ktx:2.4.0"
}

2)、使用 WindowManager

通过 WindowManager 的 WindowInfoTracker 接口访问窗口功能。

打开 MainActivity.kt 源文件并调用 WindowInfoTracker.getOrCreate(this@MainActivity),以初始化与当前 activity 相关联的 WindowInfoTracker 实例:

import androidx.window.layout.WindowInfoTrackerprivate lateinit var windowInfoTracker: WindowInfoTrackeroverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)windowInfoTracker = WindowInfoTracker.getOrCreate(this@MainActivity)
}

3)、设置应用界面

通过 Jetpack WindowManager,我们能够获取窗口指标、布局和显示配置的相关信息。让我们在主 activity 布局中显示这些信息(针对每项信息分别使用一个 TextView)。

为此,我们需要一个 ConstraintLayout,其中包含三个 TextView,并在屏幕上居中。

打开 activity_main.xml 文件,然后粘贴以下内容:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"xmlns:tools="http://schemas.android.com/tools"android:id="@+id/constraint_layout"android:layout_width="match_parent"android:layout_height="match_parent"tools:context=".MainActivity"><TextViewandroid:id="@+id/window_metrics"android:layout_width="wrap_content"android:layout_height="wrap_content"android:padding="20dp"tools:text="Window metrics"android:textSize="20sp"app:layout_constraintBottom_toTopOf="@+id/layout_change"app:layout_constraintStart_toStartOf="parent"app:layout_constraintTop_toTopOf="parent"app:layout_constraintVertical_chainStyle="packed" /><TextViewandroid:id="@+id/layout_change"android:layout_width="wrap_content"android:layout_height="wrap_content"android:padding="20dp"tools:text="Layout change"android:textSize="20sp"app:layout_constrainedWidth="true"app:layout_constraintBottom_toTopOf="@+id/configuration_changed"app:layout_constraintStart_toStartOf="parent"app:layout_constraintTop_toBottomOf="@+id/window_metrics" /><TextViewandroid:id="@+id/configuration_changed"android:layout_width="wrap_content"android:layout_height="wrap_content"android:padding="20dp"tools:text="Using one logic/physical display - unspanned"android:textSize="20sp"app:layout_constraintBottom_toBottomOf="parent"app:layout_constraintStart_toStartOf="parent"app:layout_constraintTop_toBottomOf="@+id/layout_change" /></androidx.constraintlayout.widget.ConstraintLayout>

4)、直观呈现 WindowMetrics 信息

在 MainActivity 的 onCreate 方法中,我们将调用一个将在后续步骤中实现的函数。该函数将用于获取和显示 WindowMetrics 信息。首先,在 onCreate 方法中添加 obtainWindowMetrics() 调用:

MainActivity.kt

override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)binding = ActivityMainBinding.inflate(layoutInflater)setContentView(binding.root)windowInfoTracker = WindowInfoTracker.getOrCreate(this@MainActivity)obtainWindowMetrics()
}

obtainWindowMetrics 方法:

import androidx.window.layout.WindowMetricsCalculatorprivate fun obtainWindowMetrics() {val wmc = WindowMetricsCalculator.getOrCreate()val currentWM = wmc.computeCurrentWindowMetrics(this).bounds.flattenToString()val maximumWM = wmc.computeMaximumWindowMetrics(this).bounds.flattenToString()binding.windowMetrics.text ="CurrentWindowMetrics: ${currentWM}\nMaximumWindowMetrics: ${maximumWM}"
}

运行该应用。根据您使用的可折叠设备,将得到不同的结果。例如,在双屏设备模拟器中(如下图所示),您会得到与模拟器所镜像的设备尺寸对应的 CurrentWindowMetrics。还可以查看应用在单屏模式下运行时的指标:

当应用跨显示屏显示时,窗口指标会发生变化(如下图所示),因此它们现在反映的应用所用窗口区域比之前大:

由于该应用在单屏和双屏设备上始终运行并占满整个显示区域,因此当前窗口指标和最大窗口指标的值相同。

在有水平折叠边的可折叠设备模拟器中,这些值在应用跨整个物理显示屏运行时和在多窗口模式下运行时会有所不同:

5)、使用 Jetpack WindowManager 进行测试

在任何模拟器或设备上测试可折叠设备的折叠状态,对于测试如何将界面元素放置在 FoldingFeature 周围是非常有用的。WindowManger 附带了一个非常实用的工件,以用于进行插桩测试。我们来看看该工件的使用方式。应用 build.gradle 文件中,我们不仅添加了主 WindowManager 依赖项,还添加了测试工件:androidx.window:window-testing

window-testing 工件附带有一个名为 WindowLayoutInfoPublisherRule 的实用的新 TestRule,它可以帮助您使用一系列 WindowLayoutInfo 值进行测试。借助 WindowLayoutInfoPublisherRule,您可以按需推送不同的 WindowLayoutInfo 值。

为了使用它,并在此基础上创建一个示例以帮助您使用此新工件来测试界面,我们将更新由 Android Studio 模板创建的测试类。将 ExampleInstrumentedTest 类中的所有代码替换为以下代码:

import androidx.test.ext.junit.rules.ActivityScenarioRule
import androidx.test.ext.junit.runners.AndroidJUnit4
import androidx.window.testing.layout.WindowLayoutInfoPublisherRule
import org.junit.Rule
import org.junit.rules.RuleChain
import org.junit.rules.TestRule
import org.junit.runner.RunWith@RunWith(AndroidJUnit4::class)
class MainActivityTest {private val activityRule = ActivityScenarioRule(MainActivity::class.java)private val publisherRule = WindowLayoutInfoPublisherRule()@get:Ruleval testRule: TestRuleinit {testRule = RuleChain.outerRule(publisherRule).around(activityRule)}
}

我们来创建一项测试,用于检查相应实现是否正确无误。创建一项名为 testText_is_left_of_Vertical_FoldingFeature 的测试:

@Test
fun testText_is_left_of_Vertical_FoldingFeature() {activityRule.scenario.onActivity { activity ->val hinge = FoldingFeature(activity = activity,state = FoldingFeature.State.FLAT,orientation = FoldingFeature.Orientation.VERTICAL,size = 2)val expected = TestWindowLayoutInfo(listOf(hinge))publisherRule.overrideWindowLayoutInfo(expected)}onView(withId(R.id.layout_change)).check(PositionAssertions.isCompletelyLeftOf(withId(R.id.folding_feature)))
}

我们可以在设备或模拟器上运行测试,以检查应用的行为是否符合预期。这个测试无需使用可折叠设备或模拟器即可运行。

四、总结

可折叠和双屏设备不再是实验性的或未来主义的--大的显示区域和额外的姿势具有被证实的用户价值,而且现在有更多的设备可以供你的用户使用。可折叠设备和双屏设备代表了智能手机的自然进化。对于安卓开发者来说,他们提供了进入一个正在增长的高端市场的机会,这也得益于设备制造商的重新关注,而我们使用Jetpack WindowManager和WindowSizeClass来促进可折叠手机的发展,Google带来的开发工具真的越来越高效便捷。

想要进一步了解谷歌最新开发产品和工具的小伙伴,我给大家推荐CSDN专题页上的专题回放视频。https://marketing.csdn.net/p/8b1b4b3f5f0fe4c3cdf1c2d5e42a05c3

此外还可以前往 Google 开发者在线课程进一步巩固知识点,提高自己的开发技能。https://developers.google.cn/learn/pathways?hl=zh-cn&utm_source=csdn

这篇关于Android使用Jetpack WindowManager来开发可折叠设备的探索的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/531598

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Python使用国内镜像加速pip安装的方法讲解

《Python使用国内镜像加速pip安装的方法讲解》在Python开发中,pip是一个非常重要的工具,用于安装和管理Python的第三方库,然而,在国内使用pip安装依赖时,往往会因为网络问题而导致速... 目录一、pip 工具简介1. 什么是 pip?2. 什么是 -i 参数?二、国内镜像源的选择三、如何

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Linux使用nload监控网络流量的方法

《Linux使用nload监控网络流量的方法》Linux中的nload命令是一个用于实时监控网络流量的工具,它提供了传入和传出流量的可视化表示,帮助用户一目了然地了解网络活动,本文给大家介绍了Linu... 目录简介安装示例用法基础用法指定网络接口限制显示特定流量类型指定刷新率设置流量速率的显示单位监控多个

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没