Flutter全局悬浮按钮

width: 56,

height: 56,

top: size.height - 72,

left: size.width - 72,

child: Material(

color: Colors.transparent,

child: GestureDetector(

onTap: () => print(‘ON TAP OVERLAY!’),

child: Container(

decoration: BoxDecoration(

shape: BoxShape.circle, color: Colors.redAccent),

),

),

),

);

}),

);

}

方法三

===

1.场景 现在需要做一个Test按钮,悬浮在所有页面之上,并且可以拖拽。

2.思路 1)悬浮按钮可以使用flutter提供的Overlay + OverlayEntry 组合实现

2)拖拽功能可以使用GestureDetector手势按钮或者Draggable实现(PS:我做了一版Draggable实现的,但是发现它会有原本的widget浮在原地,显然不是我要的效果)

3)点击的时候我是让它弹出一个底部弹框,这里你们可以自由发挥,本篇文章不做多余赘述

PubScaffold(

child: MaterialApp(

theme: CustomTheme.lightTheme,

darkTheme: CustomTheme.darkTheme,

themeMode: currentTheme.currentTheme,

home: Scaffold(

body: Stack(

children: [

_pageList[_currentIndex],

// Positioned(

// left: _offset.dx,

// top: _offset.dy,

// child: GestureDetector(

// onPanUpdate: (d) =>

// setState(() => _offset += Offset(d.delta.dx, d.delta.dy)),

// child: FloatingActionButton(

// onPressed: () {},

// backgroundColor: Colors.orange,

// child: Icon(Icons.add),

// ),

// ),

// ),

],

),

bottomNavigationBar: CurvedNavigationBar(

// key: _bottomNavigationKey,

index: 0,

height: 60.0,

items: [

Icon(Icons.home, size: 30),

Icon(Icons.list, size: 30),

Icon(Icons.compare_arrows, size: 30),

// Icon(Icons.call_split, size: 30),

],

color: Colors.white,

buttonBackgroundColor: Colors.white,

backgroundColor: Colors.blueAccent,

animationCurve: Curves.easeInOut,

animationDuration: Duration(milliseconds: 600),

onTap: (index) {

setState(() {

_currentIndex = index;

});

},

// letIndexChange: (index) => true,

),

),

),

);

这里的PubScaffold就是我封装的一个悬浮按钮组件,把它包裹在MaterialApp外面,就可以实现悬浮在所有的组件之上的一个按钮啦(当然也可以不是按钮,具体样式可以自己定义)。下面我们来看一下PubScaffold中的代码吧~

import ‘dart:math’;

import ‘package:flutter/material.dart’;

class PubScaffold extends StatefulWidget {

final Widget child;

PubScaffold({this.child});

@override

_PubScaffoldState createState() => _PubScaffoldState();

}

class _PubScaffoldState extends State {

bool draggable = false;

//静止状态下的offset

Offset idleOffset = Offset(0, 0);

//本次移动的offset

Offset moveOffset = Offset(0, 0);

//最后一次down事件的offset

Offset lastStartOffset = Offset(0, 0);

int count = 0;

final List testWidgetList = [

‘测试1’,

‘测试2’,

];

testAppFun(e) {

// TODO: 你的代码逻辑

}

// 显示一个底部弹窗,这里是一个测试列表

showTestList() {

showModalBottomSheet(

context: context,

enableDrag: false,

shape: RoundedRectangleBorder(

borderRadius: BorderRadius.only(

topLeft: Radius.circular(20.0),

topRight: Radius.circular(20.0),

),

),

builder: (BuildContext context) {

return ListView(

children: testWidgetList

.map(

(e) => Container(

decoration: BoxDecoration(

border: Border(

bottom: BorderSide(color: Color(0xFFe3e3e3)),

),

),

child: ListTile(

onTap: () => testAppFun(e),

title: Text(e),

),

),

)

.toList(),

);

},

);

}

@override

Widget build(BuildContext context) {

return LayoutBuilder(

builder: (context, constraints) {

// 显示悬浮按钮

WidgetsBinding.instance

.addPostFrameCallback((_) => _insertOverlay(context));

return widget.child;

},

);

}

// 悬浮按钮,可以拖拽(可自定义样式)

void _insertOverlay(BuildContext context) {

return Overlay.of(context).insert(

OverlayEntry(builder: (context) {

final size = MediaQuery.of(context).size;

print(size.width);

return Positioned(

top: draggable ? moveOffset.dy : size.height - 102,

left: draggable ? moveOffset.dx : size.width - 72,

child: GestureDetector(

// 移动开始

onPanStart: (DragStartDetails details) {

setState(() {

lastStartOffset = details.globalPosition;

draggable = true;

});

if (count <= 1) {

count++;

}

},

// 移动中

onPanUpdate: (DragUpdateDetails details) {

setState(() {

moveOffset =

details.globalPosition - lastStartOffset + idleOffset;

if (count > 1) {

moveOffset = Offset(max(0, moveOffset.dx), moveOffset.dy);

} else {

moveOffset = Offset(max(0, moveOffset.dx + (size.width - 72)),

moveOffset.dy + (size.height - 102));

}

});

},

// 移动结束

onPanEnd: (DragEndDetails detail) {

setState(() {

idleOffset = moveOffset * 1;

});

},

child: TestContainer(

onPress: () => showTestList(),

),

),

);

}),

);

}

}

// 悬浮按钮的样式

class TestContainer extends StatelessWidget {

final Function onPress;

TestContainer({this.onPress});

@override

Widget build(BuildContext context) {

return Material(

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

总结

学习技术是一条慢长而艰苦的道路,不能靠一时激情,也不是熬几天几夜就能学好的,必须养成平时努力学习的习惯。所以:贵在坚持!

最后如何才能让我们在面试中对答如流呢?

答案当然是平时在工作或者学习中多提升自身实力的啦,那如何才能正确的学习,有方向的学习呢?有没有免费资料可以借鉴?为此我整理了一份Android学习资料路线:

这里是一部分我工作以来以及参与过的大大小小的面试收集总结出来的一套BAT大厂面试资料专题包,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家。

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划。来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。

最后,祝愿即将跳槽和已经开始求职的大家都能找到一份好的工作!

这些只是整理出来的部分面试题,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢再关注一下

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!

T大厂面试资料专题包**,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家。

[外链图片转存中…(img-FL8ItbsR-1712265135191)]

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划。来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。

最后,祝愿即将跳槽和已经开始求职的大家都能找到一份好的工作!

这些只是整理出来的部分面试题,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢再关注一下

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值