Android Studio App开发之捕获屏幕的变更事件实战(包括竖屏与横屏切换,回到桌面与切换到任务列表)

简介: Android Studio App开发之捕获屏幕的变更事件实战(包括竖屏与横屏切换,回到桌面与切换到任务列表)

需要源码请点赞关注收藏后评论区留言~~~

一、竖屏与横屏切换

除了系统广播之外,App所处的环境也会影响运行,比如手机有竖屏与横批两种模式,竖屏时水平方向较短且垂直方向较长,横屏时水平方向较长而垂直方向较短,两种屏幕方向不但造成App界面的展示差异,而且竖屏和横屏切换之时,甚至会打乱App的生命周期

由下图可见横屏与竖屏的日志时间不一样,说明App从竖屏变为横屏的时候,整个活动页面又重头创建了一遍,这样不停的重新加载是非常消耗系统资源的。所以Android设计了一种配置变更机制,无须重启活动页面只需执行特定的变更行为即可,分为两步

1:修改AndroidManifest.xml文件

2:修改活动页面的java代码

此处要连接真机比较好进行测试 模拟机没法旋转~~~

代码如下

java类代码

package com.example.chapter09;
import androidx.appcompat.app.AppCompatActivity;
import android.content.res.Configuration;
import android.os.Bundle;
import android.widget.TextView;
import com.example.chapter09.util.DateUtil;
public class ChangeDirectionActivity extends AppCompatActivity {
    private TextView tv_monitor; // 声明一个文本视图对象
    private String mDesc = ""; // 屏幕变更的描述说明
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_change_direction);
        tv_monitor = findViewById(R.id.tv_monitor);
    }
    // 在配置项变更时触发。比如屏幕方向发生变更等等
    @Override
    public void onConfigurationChanged(Configuration newConfig) {
        super.onConfigurationChanged(newConfig);
        switch (newConfig.orientation) { // 判断当前的屏幕方向
            case Configuration.ORIENTATION_PORTRAIT: // 切换到竖屏
                mDesc = String.format("%s%s %s\n", mDesc,
                        DateUtil.getNowTime(), "当前屏幕为竖屏方向");
                tv_monitor.setText(mDesc);
                break;
            case Configuration.ORIENTATION_LANDSCAPE: // 切换到横屏
                mDesc = String.format("%s%s %s\n", mDesc,
                        DateUtil.getNowTime(), "当前屏幕为横屏方向");
                tv_monitor.setText(mDesc);
                break;
            default:
                break;
        }
    }
}

XML文件

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="5dp">
    <TextView
        android:id="@+id/tv_monitor"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="请旋转手机使屏幕在竖屏与横屏之间切换"
        android:textColor="@color/black"
        android:textSize="17sp" />
</LinearLayout>

二、回到桌面与切换到任务列表

App不但能检测手机屏幕的方向变更,还能获知回到桌面的事件,连打开任务列表的事件也能实时得知,回到桌面与打开任务列表都由按键触发。因此,若想知晓是否回到桌面,以及是否打开任务列表,都需要收听系统广播 效果如下

可是监听回到桌面的广播能用来干什么呢?一种用处是开启App的画中画模式,比如原先应用正在播放视频,回到桌面时势必要暂停播放,有了画中画模式之后,可将播放界面缩小为屏幕上的一个小方块,这样即使回到桌面也能继续观看视频。

代码如下

Java类代码

package com.example.chapter09;
import androidx.appcompat.app.AppCompatActivity;
import android.app.PictureInPictureParams;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.res.Configuration;
import android.os.Build;
import android.os.Bundle;
import android.text.TextUtils;
import android.util.Log;
import android.util.Rational;
import android.widget.TextView;
import com.example.chapter09.util.DateUtil;
public class ReturnDesktopActivity extends AppCompatActivity {
    private final static String TAG = "ReturnDesktopActivity";
    private TextView tv_monitor;
    private String mDesc = "";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_return_desktop);
        tv_monitor = findViewById(R.id.tv_monitor);
        initDesktopRecevier(); // 初始化桌面广播
    }
    // 显示变更的状态
    private void showChangeStatus(String reason) {
        mDesc = String.format("%s%s 按下了%s键\n", mDesc, DateUtil.getNowTime(), reason);
        tv_monitor.setText(mDesc);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O
                && !isInPictureInPictureMode()) { // 当前未开启画中画,则开启画中画模式
            // 创建画中画模式的参数构建器
            PictureInPictureParams.Builder builder = new PictureInPictureParams.Builder();
            // 设置宽高比例值,第一个参数表示分子,第二个参数表示分母
            // 下面的10/5=2,表示画中画窗口的宽度是高度的两倍
            Rational aspectRatio = new Rational(10,5);
            builder.setAspectRatio(aspectRatio); // 设置画中画窗口的宽高比例
            // 进入画中画模式,注意enterPictureInPictureMode是Android8.0之后新增的方法
            enterPictureInPictureMode(builder.build());
        }
    }
    // 在进入画中画模式或退出画中画模式时触发
    @Override
    public void onPictureInPictureModeChanged(boolean isInPicInPicMode, Configuration newConfig) {
        Log.d(TAG, "onPictureInPictureModeChanged isInPicInPicMode="+isInPicInPicMode);
        super.onPictureInPictureModeChanged(isInPicInPicMode, newConfig);
        if (isInPicInPicMode) { // 进入画中画模式
        } else { // 退出画中画模式
        }
    }
    // 初始化桌面广播
    private void initDesktopRecevier() {
        desktopRecevier = new DesktopRecevier(); // 创建一个返回桌面的广播接收器
        // 创建一个意图过滤器,只接收关闭系统对话框(即返回桌面)的广播
        IntentFilter intentFilter = new IntentFilter(Intent.ACTION_CLOSE_SYSTEM_DIALOGS);
        registerReceiver(desktopRecevier, intentFilter); // 注册接收器,注册之后才能正常接收广播
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        unregisterReceiver(desktopRecevier); // 注销接收器,注销之后就不再接收广播
    }
    private DesktopRecevier desktopRecevier; // 声明一个返回桌面的广播接收器对象
    // 定义一个返回到桌面的广播接收器
    private class DesktopRecevier extends BroadcastReceiver {
        // 在收到返回桌面广播时触发
        @Override
        public void onReceive(Context context, Intent intent) {
            if (intent.getAction().equals(Intent.ACTION_CLOSE_SYSTEM_DIALOGS)) {
                String reason = intent.getStringExtra("reason"); // 获取变更原因
                // 按下了主页键或者任务键
                if (!TextUtils.isEmpty(reason) && (reason.equals("homekey")
                        || reason.equals("recentapps"))) {
                    showChangeStatus(reason); // 显示变更的状态
                }
            }
        }
    }
}

XML文件

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    android:padding="5dp">
    <TextView
        android:id="@+id/tv_monitor"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="请按下主页键或者任务键,然后回到该页面观察结果"
        android:textColor="@color/black"
        android:textSize="17sp" />
</LinearLayout>

创作不易 觉得有帮助请点赞关注收藏~~~

相关文章
|
6月前
|
缓存 移动开发 JavaScript
如何优化UniApp开发的App的启动速度?
如何优化UniApp开发的App的启动速度?
1165 139
|
6月前
|
移动开发 JavaScript weex
UniApp开发的App在启动速度方面有哪些优势和劣势?
UniApp开发的App在启动速度方面有哪些优势和劣势?
545 137
|
6月前
|
数据采集 JavaScript 前端开发
开发比分App?你缺的不是程序员
开发体育比分App,关键不在代码,而在懂体育、懂数据、懂用户。明确定位、理清需求、选好数据源,再找专业的产品、数据与技术人才协同,才能少走弯路。程序员最后入场,效率最高。
337 154
|
7月前
|
移动开发 小程序 Android开发
基于 uni-app 开发的废品回收类多端应用功能与界面说明
本文将对一款基于 uni-app 开发的废品回收类多端应用,从多端支持范围、核心功能模块及部分界面展示进行客观说明,相关资源信息也将一并呈现。
245 0
|
6月前
|
移动开发 前端开发 Android开发
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
1125 12
【02】建立各项目录和页面标准化产品-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
|
6月前
|
移动开发 JavaScript 应用服务中间件
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
885 5
【06】优化完善落地页样式内容-精度优化-vue加vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
|
6月前
|
移动开发 Rust JavaScript
【01】首页建立-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
【01】首页建立-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
1024 4
【01】首页建立-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓伊凡
|
6月前
|
人工智能 前端开发 JavaScript
最佳实践3:用通义灵码开发一款 App
本示例演示使用通义灵码,基于React Native与Node.js开发跨平台类通义App,重点展示iOS端实现。涵盖前端页面生成、后端代码库自动生成、RTK Query通信集成及Qwen API调用全过程,体现灵码在全栈开发中的高效能力。(238字)
761 11
|
6月前
|
人工智能 小程序 开发者
【一步步开发AI运动APP】十二、自定义扩展新运动项目03
继【一步步开发AI运动小程序】后,我们推出新系列【一步步开发AI运动APP】,助开发者打造高性能、优体验的AI运动应用。本文详解自定义扩展运动分析器的统一管理实现,提升代码复用性与可维护性,涵盖APP与小程序插件差异及完整代码示例,助力AI运动场景深度拓展。