当前位置:首页 > 冰箱 > 文章正文

3月Flutter小报|读小报涨常识

编辑:[db:作者] 时间:2024-08-25 04:15:46

2. Flutter如何获取键盘的完全高度

3月Flutter小报|读小报涨常识

3. Flutter快速实现新手勾引气泡

Flutter如何Mock MethodChannel进行单元测试

在做Flutter单元测试的时候,有时候我们会碰着Flutter Widget的某个方法调用了Platform的方法,这时候就须要Mock这个MethodChannel来肃清依赖,否则测试用例实行到Channel的方法就会抛出非常。

1. 获取TestDefaultBinaryMessenger

在测试环境下,Flutter给我们供应了TestDefaultBinaryMessenger来拦截MethodChannel,以是我们须要先获取到它。

/// 依赖WidgetTester,须要在测试用例中获取testWidgets('one test case', (widgetTester) async {final TestDefaultBinaryMessenger messenger = widgetTester.binding.defaultBinaryMessenger;});/// 通过单例获取,写在setUp中可以在所有测试用例实行前运行setUp(() {final TestDefaultBinaryMessenger messenger = TestDefaultBinaryMessengerBinding.instance!.defaultBinaryMessenger;}2. Mock Flutter与Platform间的相互调用

• Flutter调用Platform方法:

TestDefaultBinaryMessenger#setMockMethodCallHandler,第一个参数是须要拦截的MethodChannel,第二个参数是Function表示Mock调用。

TestDefaultBinaryMessenger#allMessagesHandler,和上面类似,但这里是拦截所有的MethodChannel,并且,此项设置后,setMockMethodCallHandler将不生效。

• Platform调用Flutter方法:

TestDefaultBinaryMessenger#handlePlatformMessage,第一个参数是MethodChannel名字,第二个参数是传给Flutter编码后的MethodCall,第三个参数是Flutter处理后结果的回调。

3. 示例

allMessagesHandler利用举例,如果我们有这样一个MethodChannel

class PipeLinePlugin { PipeLinePlugin({this.pipeLineId, this.onTextureRegistered}) { _channel = MethodChannel('method_channel/pipeline_$pipeLineId');/// 调用start后,Platform会回调registerTextureId _channel.setMethodCallHandler((call) {if (call.method == 'registerTextureId' && call.arguments is Map) {int textureId = (call.arguments as Map)?['textureId']; onTextureRegistered?.call(textureId); } }); }final String pipeLineId;final MethodChannel _channel;final Function(int textureId)? onTextureRegistered; Future<bool?> start() async {final bool? result = await _channel.invokeMethod('start', <String, dynamic>{'id': pipeLineId}) as bool?;return result; } Future<bool?> stop() async {final bool? result = await _channel.invokeMethod('stop', <String, dynamic>{'id': pipeLineId}) as bool?;return result; }}

我们可以这样Mock它,然后我们的测试用例就能正常实行了。

const StandardMethodCodec methodCodec = StandardMethodCodec();/// 如果channel名字是按规则天生的,可以拦截所有的MethodChannel,再从中找到你须要Mock的MethodChannelmessenger.allMessagesHandler = (String channel, MessageHandler? handler, ByteData? message) async {final MethodCall call = methodCodec.decodeMethodCall(message);if (channel.startWith('method_channel/pipeline')) {if (call.method == 'start') {/// Platform收到start后,须要回调registerTextureIdfinal platformResultCall = MethodCall('registerTextureId', {'textureId': 0}); messenger.handlePlatformMessage(channel, methodCodec.encodeMethodCall(platformResultCall), ); } }/// Flutter的MethodCall统一返回truereturn methodCodec.encodeSuccessEnvelope(true);}Flutter如何获取键盘的完全高度

我们知道在Flutter中获取键盘高度可以通过向WidgetsBinding注册监听,当键盘弹出或消逝时会回调didChangeMetrics方法。
须要把稳的是,在Flutter3新增IOS键盘动画后,IOS会和Android一样回调didChangeMetrics多次,并且每次回调中MediaQueryData.fromWindow(window).viewInsets.bottom的值都是此时键盘冒出来的高度,即键盘实时高度。
如果我们想获取键盘的完全高度,只须要一贯取和上次比较的最大值,然后保存下来就可以了。

/// Flutter3 /// 获取键盘高度@overridevoid didChangeMetrics() {final bottom = MediaQueryData.fromWindow(window).viewInsets.bottom;// 键盘存在中间态,回调是键盘冒出来的高度 keyboardHeight = max(keyboardHeight, bottom);if (bottom == 0) { isKeyboardShow = false; } else if (bottom == keyboardHeight || keyboardHeight == 0) { isKeyboardShow = true; } else { isKeyboardShow = ; }// 键盘完备收起或展开再刷新页面if (isKeyboardShow != && _preKeyboardShow != isKeyboardShow) { _keyboardStateNotifier.notifyListeners(); }if (bottom < keyboardHeight) { _sp?.setDouble(KEYBOARD_MAX_HEIGHT, keyboardHeight); }}/// Flutter3之前,获取键盘高度@overridevoid didChangeMetrics() {final bottom = MediaQueryData.fromWindow(window).viewInsets.bottom;/// ios点击键盘表情时,键盘高度会增加,后面点拼音后也回不到这个高度了if (Platform.isIOS) {/// ios键盘有两种高度,但不存在中间态,回调便是键盘高度 isKeyboardShow = bottom > 0;if (isKeyboardShow) { keyboardHeight = bottom; } } else {/// Android键盘存在中间态,回调是键盘冒出来的高度 keyboardHeight = max(keyboardHeight, bottom);if (bottom == 0) { isKeyboardShow = false; } else if (bottom == keyboardHeight || keyboardHeight == 0) { isKeyboardShow = true; } else { isKeyboardShow = ; } }// 键盘完备收起或展开再刷新页面if (isKeyboardShow != && _preKeyboardShow != isKeyboardShow) { _keyboardStateNotifier.notifyListeners(); }if (bottom < keyboardHeight) { _sp?.setDouble(KEYBOARD_MAX_HEIGHT, keyboardHeight); }}

例如我们想在键盘上做一个输入框,只须要给输入框底部添加一个和键盘实时高度相同的Padding,就能做到像Scaffold#bottomSheet一样跟随键盘动画的效果了

Flutter快速实现新手勾引气泡

当我们上线一个新功能时常常须要做个气泡勾引用户利用,但如何在不改变布局的情形下给Widget加上气泡呢? 我们可以通过OverlayEntry在更高的页面层级上插入气泡,同时根据相对的Widget位置来打算气泡的位置,详细代码实现如下。
须要把稳的是,虽然我们在Widget的dispose中会销毁气泡,但如果Widget是页面并且页面有弹窗,气泡会涌如今弹窗上,以是利用的时候须要在弹窗前主动销毁气泡。

/// 以相对widget为中央扩展刚好容纳bubble大小的RectRect _expandRelativeRect(BuildContext relativeContext, Size bubbleSize, bool allowOutsideScreen) {if ((relativeContext as Element)?.isElementActive != true) {return Rect.zero; }// 找到相对widget的位置和大小final RenderBox renderBox = relativeContext.findRenderObject();final Offset offset = renderBox.localToGlobal(Offset.zero);final Size size = renderBox.size;// 以相对widget为中央扩展刚好容纳bubble大小的Rectfinal rect = Rect.fromLTWH(offset.dx - bubbleSize.width, offset.dy - bubbleSize.height, bubbleSize.width 2 + size.width, bubbleSize.height 2 + size.height);if (allowOutsideScreen) {return rect; } else {final screenSize = MediaQueryData.fromWindow(ui.window).size;final screenRect = Rect.fromLTWH(0, 0, screenSize.width, screenSize.height);return rect.intersect(screenRect); }}

/// 构建气泡EntryOverlayEntry bubbleEntry = OverlayEntry(builder: (bubbleContext) { Rect rect = _expandRelativeRect(relativeContext, bubbleSize, allowOutsideScreen);return Positioned.fromRect( rect: rect, child: Align( alignment: alignment, child: SizedBox( child: bubble.call(bubbleContext), width: bubbleSize.width, height: bubbleSize.height, )));});Overlay.of(context).insert(bubbleEntry);/// 关闭气泡VoidCallback hideBubbleCallback = () {if (!bubbleEntry.mounted) {return; } bubbleEntry.remove();};

本站所发布的文字与图片素材为非商业目的改编或整理,版权归原作者所有,如侵权或涉及违法,请联系我们删除,如需转载请保留原文地址:http://www.baanla.com/bx/116021.html

XML地图 | 自定链接

Copyright 2005-20203 www.baidu.com 版权所有 | 琼ICP备2023011765号-4 | 统计代码

声明:本站所有内容均只可用于学习参考,信息与图片素材来源于互联网,如内容侵权与违规,请与本站联系,将在三个工作日内处理,联系邮箱:123456789@qq.com