我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Flutter中使用Overlay传入context提示:Null check operator used on a null value(对空值使用空检查运算符)

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Flutter中使用Overlay传入context提示:Null check operator used on a null value(对空值使用空检查运算符)

首先此时使用的是GetX框架,框架截图如下:

View中代码如下:

class AddTaskPage extends StatelessWidget {  const AddTaskPage({Key? key}) : super(key: key);  @override  Widget build(BuildContext context) {    final logic = Get.put(AddTaskLogic());    final state = Get.find().state;    return Scaffold(      appBar: AppBar(        backgroundColor: Tcolor.barBackgroudColor,        elevation: 0,      ),      body: _addTaskContent(state),    );  }  _addTaskContent(AddTaskState addTaskState) {    return GetBuilder(builder: (logic) {      return Container(        child: Column(          children: [            //任务类型 0-工作 1-学习 2-生活            Container(              key: logic.typeKey,              child: InputFied(                  fieldwidth: 300,                  hintText: "请选择任务类型",                  controller: addTaskState.contentController,                  iconWidget: InkWell(                    child: Icon(addTaskState.typeExpand?Icons.expand_less_outlined:Icons.expand_more,color: Colors.grey,size: 18,),                    onTap: (){                      print("弹出任务类型选择框");                      logic.selectTypePop();                    },                  )              ),            ),          ],        ),      );    });  }}

logic代码

class AddTaskLogic extends GetxController {  final AddTaskState state = AddTaskState();  GlobalKey typeKey = GlobalKey();  //弹出任务类型选择框  selectTypePop(){    state.typeExpand=!state.typeExpand;    update();    RenderBox typeBox = typeKey.currentContext!.findRenderObject() as RenderBox;    print("要开始构建了Pop了");        PopToastManager().buildUI(      context: Get.context!,      isClickDiss: true,      X: typeBox.localToGlobal(Offset.zero).dx + 28 + 76,      Y: typeBox.localToGlobal(Offset.zero).dy + 26,      offx: 0,      offy: 0,      width: 260,      height: 90,      childWidget: ListView.builder(        itemCount: state.taskType.length,        itemBuilder: (context, index) {          print("构建UI");          return Container(            child: Row(              children: [                ClipRRect(                  borderRadius: BorderRadius.circular(60),                  child: Container(                    width: 13,                    height: 13,                    decoration: BoxDecoration(                      color: state.typeColor[index],                    ),                  ),                ),                SizedBox(width: 20,),                Text(state.taskType[index]),              ],            ),          );        },      ),    );    update();  }}

对Overlay的使用封装了一个单独的类PopToastManager(),PopToastManager()代码如下:

import 'package:flutter/cupertino.dart';import 'package:flutter/material.dart';//封装一个构造弹窗类class PopToastManager{    OverlayEntry? _overlayEntry;    static PopToastManager? _manager;    PopToastManager._();    factory PopToastManager(){    if(_manager==null){      _manager=PopToastManager._();    }    return _manager!;  }    void buildUI(      {        required BuildContext context,        required bool isClickDiss,        required double X,        required double Y,        required double offx,        required double offy,        required double width,        required double height,        required Widget childWidget,        Function? dismissCallBack      }){    //创建 overlayEntry    OverlayEntry overlayEntry=OverlayEntry(        builder: (context){          print("开始构建overlayEntry");          return GestureDetector(            behavior:HitTestBehavior.opaque ,                        onTap: (){                            if(isClickDiss){                                if (dismissCallBack != null) {                  dismissCallBack();                }                                dissmiss();              }            },                        child: Container(              width: MediaQuery.of(context).size.width,              height: MediaQuery.of(context).size.height,              color: Colors.transparent,              child: Column(                crossAxisAlignment: CrossAxisAlignment.start,                children: [                  // Container(height: Y,),                  SizedBox(height: Y,),                  Container(                    margin: EdgeInsets.fromLTRB(offx + X,offy, 0, 0),                    width: width,                    height: height,                    child: childWidget,                  ),                ],              ),            ),          );        });    this._overlayEntry = overlayEntry;    //插入到 Overlay中显示 OverlayEntry    Overlay.of(context)!.insert(overlayEntry);      }  void dissmiss(){    if(this._overlayEntry != null){      print("蒙层消失");      this._overlayEntry!.remove();      this._overlayEntry = null;    }  }}

运行结果如下:

 根据打印的语句可以得知,代码根本没有开始构建OverlayEntry(builder:(context){})

查看错误,发现问题出现在对Overlay.of(context)!.insert(overlayEntry);中context的空值判断

于是我首先在PopToastManager()类buildUI()方法中添加对context是否为空的判断

 

此时运行结果

说明:此时传入的context不为空但是Overlay.of(context)为空

为什么会这样?

首先:需要知道Overlay.of(context)意味着什么?

据搜索Overlay 是一个可以在应用程序中显示浮动元素的特殊 Widget。Overlay.of(context)是Flutter框架中的一个方法,用于获取指定上下文中最近的Overlay,Overlay.of(context) 方法返回的对象可以用于向 Overlay 中添加浮动元素。

如果 Overlay.of(context) 返回 null,则表示指定上下文中没有找到 Overlay。这通常是由于没有在该上下文的 Widget 树中添加 Overlay 导致的。在使用 Overlay.of(context) 之前,需要确保在应用程序的 Widget 树中添加了 Overlay。

此时可推测:

在上下文中没有添加Overlay,所以不能向Overlay 中添加浮动元素x

上下文出现了问题

 试解决问题(1):

 

 出错的原因:

1、在 Flutter 中,context 是一个非常重要的概念,它代表了 widget 在 widget 树中的位置。context 的不同,可能会影响到你能够访问到的 widget 或者 state。

2、在代码中传入context时使用了 Get.context。Get 是一个用于状态管理和依赖注入的库,它的 context 是全局的,可能并不在你当前 widget 的 widget 树中。因此,当你使用 Get.context 时,你可能得到的是一个全局的、不在当前 widget 树中的 context。


3、但是 Overlay.of(context) 需要一个在当前 widget 树中的 context 才能正常工作。如果你传入的 context 不在当前 widget 树中,Overlay.of(context) 就会返回 null。

4、所以,即使 Get.context 不为 null,Overlay.of(Get.context) 也可能为 null,因为 Get.context 可能不在当前 widget 树中。

要解决这个问题,你需要确保你传入 Overlay.of(context) 的 context 是在当前 widget 树中的。

将context作为参数传入进selectTypePop()

 

来源地址:https://blog.csdn.net/weixin_43244083/article/details/131471321

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Flutter中使用Overlay传入context提示:Null check operator used on a null value(对空值使用空检查运算符)

下载Word文档到电脑,方便收藏和打印~

下载Word文档

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录