Flutter 105: 图解自定义 ACEPageMenu 滑动菜单 (一)

简介: 0 基础学习 Flutter,第一百零五步:自定义从屏幕四周滑动 ACEPageMenu!

      小菜尝试做一个类似 BottomSheet 的滑动 Menu,不局限于底部,可以从屏幕四周滑出;因涉及内容较多,小菜计划拆分开来总结和完善,先介绍大体结构,之后再详细学习;

      小菜自定义的 ACEPageMenu 滑动菜单在绘制及动画主要涉及两方面,小菜简单介绍;

AnimatedBuilder

      小菜需要 Menu 从屏幕四周滑动出来,此时一定需要 Animation 动画,而对于动画,小菜尝试用 AnimatedBuilder 来处理,虽然需要设置 AnimatedController 等,但对于动画的处理相对灵活;

1. AnimationController

      首先需要设置一个 Animation 控制器,在指定的 Duration 时长内,屏幕绘制过程中,会线性的生成 0.0-1.0 的数值用来控制动画的开始与结束以及设置动画的监听;通过 vsync 防止在屏幕外的 Animation 消耗不必要资源;

      使用 AnimationController 时需要注意在 initState() 生命周期中进行初始化和在 dispose() 结束生命周期时进行销毁;同时可以通过 addStatusListener() 对动画过程进行监听;
      a. AnimationStatus.forward 为动画开始时的回调监听,与 AnimationController.forward() 对应;
      b. AnimationStatus.completed 为动画执行结束时的回调监听;
      c. AnimationStatus.reverse 为动画反向执行时的回调监听,与 AnimationController.reverse() 对应;
      d. AnimationStatus.dismissed 为动画反向执行结束时的回调监听;

@override
void initState() {
  super.initState();
  _controller = AnimationController(
      duration: const Duration(milliseconds: 600), vsync: this);
  _controller.addStatusListener((status) {
    switch(status){
      case AnimationStatus.dismissed:
        print("Current status is dismissed !");
        break;
      case AnimationStatus.forward:
        print("Current status is forward !");
        break;
      case AnimationStatus.reverse:
        print("Current status is reverse !");
        break;
      case AnimationStatus.completed:
        print("Current status is completed !");
        break;
    }
  });
}

@override
void dispose() {
  _controller.dispose();
  super.dispose();
}

2. AnimatedBuilder

      AnimationController 之后需要设置具体 Menu Widget 所在的 AnimatedBuilder 动画构造器;在其中设置平移动画,并与 AnimationController 控制器进行关联;具体的动画相关的会在之后的博客中继续详细学习;

return AnimatedBuilder(
    animation: _controller,
    child: Container(
        color: Color(0xF3242424),
        height: 200.0,
        width: ScreenUtils.getScreenWidth()),
    builder: (BuildContext context, Widget child) {
      return Transform.translate(offset: Offset(0, _controller.value * 50), child: child);
    });

SingleChildLayoutDelegate

      动画的处理基本搞定,重要的是如何让 Widget 从屏幕四周外部开始平移,此时小菜尝试用 SingleChildLayoutDelegate 来处理;

      SingleChildLayoutDelegate 是用于计算带有单个子对象的渲染对象的布局的委托,其本身是一个抽象类,需要自己实现对应的 Delegate 委托;小菜自定义一个 ACEMenuDelegate,主要实现两个方法,分别为:确定要应用于子项的约束的 getConstraintsForChild() 和确定子项位置的 getPositionForChild()

      当提供对应的实例时,应调用 shouldRelayout(),判断实例是否实际代表其他信息;具体的应用小菜会在之后的博客中进一步学习;

class ACEMenuDelegate extends SingleChildLayoutDelegate {
  final MenuType _menuType;
  final double _controllerValue;

  ACEMenuDelegate(this._menuType, this._controllerValue);

  @override
  BoxConstraints getConstraintsForChild(BoxConstraints constraints) {
    return BoxConstraints(
        minWidth: (_menuType == MenuType.MENU_LEFT ||
                _menuType == MenuType.MENU_RIGHT)
            ? 0
            : constraints.maxWidth,
        maxWidth: (_menuType == MenuType.MENU_LEFT ||
                _menuType == MenuType.MENU_RIGHT)
            ? ScreenUtils.getScreenWidth() * 0.75
            : constraints.maxWidth,
        minHeight: 0.0,
        maxHeight: (_menuType == MenuType.MENU_LEFT ||
                _menuType == MenuType.MENU_RIGHT)
            ? constraints.maxHeight
            : constraints.maxHeight * 0.45);
  }

  @override
  Offset getPositionForChild(Size size, Size childSize) {
    double _offsetX = Offset.zero.dx, _offsetY = Offset.zero.dy;
    switch (_menuType) {
      case MenuType.MENU_TOP:
        _offsetY = -childSize.height * (1 - _controllerValue);
        break;
      case MenuType.MENU_BOTTOM:
        _offsetY = size.height - childSize.height * _controllerValue;
        break;
      case MenuType.MENU_LEFT:
        _offsetX = -childSize.width * (1 - _controllerValue);
        break;
      case MenuType.MENU_RIGHT:
        _offsetX = size.width - childSize.width * _controllerValue;
        break;
    }
    return Offset(_offsetX, _offsetY);
  }

  @override
  bool shouldRelayout(ACEMenuDelegate oldDelegate) {
    return _controllerValue != oldDelegate._controllerValue;
  }
}

      ACEPageMenu 源码


      小菜今天只是大概介绍一下功能实现,对于细节部分以及手势操作正在进一步完善,对于动画和委托的学习会在之后进一步学习;如有错误,请多多指导!

来源: 阿策小和尚

目录
相关文章
|
定位技术 开发工具 开发者
为了让外卖小哥在地图里开上火箭🚀我用FLutter自定义了地图
花了五天时间,用Flutter自定义地图是什么体验?外卖小哥都开上火箭了?什么?我被女朋友赶出家门啦?欢迎观看被女友赶出家门之开火箭送外卖篇~
|
4天前
|
前端开发 搜索推荐 开发者
【Flutter前端技术开发专栏】Flutter中的自定义主题与暗黑模式
【4月更文挑战第30天】本文介绍了如何在Flutter中自定义主题和实现暗黑模式。通过`ThemeData`类定义应用的外观,包括颜色、字体和样式。示例展示了如何设置主色、强调色及文本、按钮主题。暗黑模式可通过`darkTheme`属性启用,结合`ThemeData.dark()`方法定制。利用`MediaQuery`监听系统亮度变化,动态调整暗黑模式状态。Flutter的这些特性有助于开发者创建独特且用户友好的界面。
【Flutter前端技术开发专栏】Flutter中的自定义主题与暗黑模式
|
4天前
|
缓存 前端开发 搜索推荐
【Flutter前端技术开发专栏】Flutter中的自定义绘制与Canvas API
【4月更文挑战第30天】Flutter允许开发者通过`CustomPaint`和`CustomPainter`进行自定义绘制,以实现丰富视觉效果。`CustomPaint` widget将`CustomPainter`应用到画布,而`CustomPainter`需实现`paint`和`shouldRepaint`方法。`paint`用于绘制图形,如示例中创建的`MyCirclePainter`绘制蓝色圆圈。Canvas API提供绘制形状、路径、文本和图片等功能。注意性能优化,避免不必要的重绘和利用缓存提升效率。自定义绘制让Flutter UI更具灵活性和个性化,但也需要图形学知识和性能意识。
【Flutter前端技术开发专栏】Flutter中的自定义绘制与Canvas API
|
4天前
|
开发框架 前端开发 搜索推荐
【Flutter前端技术开发专栏】Flutter中的自定义Widget与渲染流程
【4月更文挑战第30天】探索Flutter的自定义Widget与渲染流程。自定义Widget是实现复杂UI设计的关键,优点在于个性化设计、功能扩展和代码复用,但也面临性能优化和复杂性管理的挑战。创建步骤包括设计结构、定义Widget类、实现构建逻辑和处理交互。Flutter渲染流程涉及渲染对象树、布局、绘制和合成阶段。实践案例展示如何创建带渐变背景和阴影的自定义按钮。了解这些知识能提升应用体验并应对开发挑战。查阅官方文档以深入学习。
【Flutter前端技术开发专栏】Flutter中的自定义Widget与渲染流程
|
4天前
|
前端开发 开发者 UED
Flutter的自定义Painter:深入探索自定义绘制Widget的技术实现
【4月更文挑战第26天】Flutter的自定义Painter允许开发者根据需求绘制独特UI,通过继承`CustomPaint`类和重写`paint`方法实现。在`paint`中使用`Canvas`绘制图形、路径等。创建自定义Painter类后,将其作为`CustomPaint` Widget的`painter`属性使用。此技术可实现自定义形状、渐变、动画等复杂效果,提升应用视觉体验。随着Flutter的进化,自定义Painter将提供更丰富的功能。
|
4天前
|
运维 监控 定位技术
应用研发平台EMAS常见问题之flutter插件不支持自定义图标如何解决
应用研发平台EMAS(Enterprise Mobile Application Service)是阿里云提供的一个全栈移动应用开发平台,集成了应用开发、测试、部署、监控和运营服务;本合集旨在总结EMAS产品在应用开发和运维过程中的常见问题及解决方案,助力开发者和企业高效解决技术难题,加速移动应用的上线和稳定运行。
81 0
|
4天前
Flutter 自定义ICON库
Flutter 自定义ICON库 Flutter提供了一些内置的ICON库,但在实际开发中,可能需要一些自定义的ICON图标。Flutter允许我们使用自定义图标,本文将介绍如何创建和使用自定义ICON库。
|
4天前
|
UED
Flutter之自定义路由切换动画
Flutter之自定义路由切换动画 在Flutter中,我们可以通过Navigator来实现路由管理,包括路由的跳转和返回等。默认情况下,Flutter提供了一些简单的路由切换动画,但是有时候我们需要自定义一些特殊的动画效果来提高用户体验。本文将介绍如何在Flutter中实现自定义的路由切换动画。
|
4天前
|
开发框架 Dart 容器
Flutter 自定义渐变按钮 GradientButton
Flutter 自定义渐变按钮 GradientButton Flutter 是一种流行的跨平台移动应用开发框架。Flutter 提供了许多内置的小部件,但有时您可能需要创建自己的小部件以满足特定的需求。这个文档将介绍如何创建一个自定义渐变按钮小部件 GradientButton。
|
4天前
|
容器
Flutter 自定义实现时间轴、侧边进度条
时间轴和侧边进度条是非常常见的 UI 控件,它们可以增强应用的视觉效果和交互体验。在这篇文章中,我们将详细介绍如何使用 Flutter 自定义实现这两个控件。
111 1