Flutter vs 前端 杂谈:SliverAppBar、手动实现Appbar、前端Html+JS怎么实现滚动变化型Appbar - 比较

news2024/11/26 4:52:35
Flutter vs 前端 杂谈
SliverAppBar的弹性背景的显隐效果使用Html+JS怎么实现

作者李俊才 (jcLee95):https://blog.csdn.net/qq_28550263
邮箱 :291148484@163.com
本文地址:https://blog.csdn.net/qq_28550263/article/details/134149018



1. 一些AppBar效果

Flutter 中,最简单的 appbar 就是 Appbar 组件,它没有任何难点,任何刚刚入门的开发着在 Flutter 脚手架创建的计数器应用中就使用了它。但是现实的开发场景中,Appbar 组件往往难以适应复杂的需求场景。

比如以下是 “王者营地” APP (即王者荣耀官方的社区应用) 的 Appbar,这个据说也是 Flutter 实现的:
在这里插入图片描述
这种向下滚动时,AppBar出现,向上滚到顶AppBar逐渐隐藏的效果还是比较简单,可以直接使用SliverAppBar。

与之相比,下面这个高德地图滚动方向与王者营地是相反的,并且还带有一个相遇于下面内容部分似乎在向下跑的图片:
在这里插入图片描述

这些效果当然不是使用 Appbar 组件做的。

在 Flutter 中,最简单的随着滚动带有显影效果的appbar可以使用 SliverAppBar 组件实现。

但是实际上appbar仅仅是一个应用顶部导航的效果不仅仅局限于 Flutter 原生的 AppbarSliverAppBar 。实际上,为了实现更加灵活的 appbar,还可以考虑基于 Sliver 协议 实现外观类似的组件,将它放在页面的顶部,着很好理解,因为在 写 Web 的时候就可以这样干(事实上我就是这样干过)。因此先从一个类似的 Web 中手写的例子看起。

2. 一个Web移动端上的复杂AppBar例子

先看效果吧(其实就是模仿上面的高德地图的大概效果):
在这里插入图片描述

(附:感谢图片来源地址,我在网络随便拿的,仅仅用于此示例,祝愿贵App、贵店铺生意红火。)

这个Appbar以及相关其它动画效果,本质上都是与滚动相关的。总结起来,我们要实现的效果如下:

  1. 页面上方有一个固定的Appbar,背景颜色为蓝色,内部包含一个输入框;

  2. 页面的上半部分有一个背景图像,通过#bg-item元素实现,并且这个元素在最下层;

  3. 页面的下半部分分为两部分:

    • #scroll-item:一个滚动元素,包含一张图片,它会随着页面的滚动而滚动,但在背景元素之上。
    • #content:一个内容区域,包含一个标题,它也会随着页面的滚动而滚动,但在滚动元素之上。
  4. 使用JavaScript监听页面的滚动事件,根据滚动距离动态改变以下效果:

    • Appbar的背景颜色透明度,使其在页面滚动时逐渐变为透明。
    • Appbar内文字的颜色透明度,同样逐渐变为透明。
    • Appbar内输入框的透明度,使其在页面滚动时逐渐变为透明。
    • 滚动元素的位置,随着页面滚动而向上移动,实现视差效果。
    • 内容区域的位置,随着页面滚动而向上移动,也实现视差效果。
  5. 使用CSS的变量--my-height定义了滚动元素的初始高度,以便在JavaScript中使用。

Web 中,要实现总体思路是通过 JavaScript 监听页面滚动事件,根据滚动距离动态改变页面元素的样式,从而实现Appbar背景颜色和文字颜色的渐变效果,以及滚动元素和内容区域的视差滚动效果。这种交互设计可以提升页面的视觉吸引力和用户体验。

Web代码如下:

<!DOCTYPE html>
<html>
<head>
    <!-- 作者信息 -->
    <!-- Author: 李俊才 -->
    <!-- Email: 291149494@163.com -->

    <!-- 许可证信息 -->
    <!-- LICENSE: MIT -->
    <style>
        body {
            margin: 0;
            padding: 0;
            --my-height: 460px;  // 定义一个CSS变量,表示滚动元素的初始高度
        }

        #appbar {
            position: fixed;
            top: 0;
            width: 100%;
            height: 50px;
            background: rgba(0, 123, 255, 1);
            color:white;
            font-size: large;
            transition: background 0.3s;
            display: flex;
            align-items: center;
            justify-content: space-between;
            padding: 0 10px;
            padding-right: 20px;
            z-index: 3;  // 设置appbar在最上层
        }

        #appbar-input {
            padding: 5px;
            border-radius: 5px;
            border: 1px solid white;
            margin-right: 20px;
        }

        #appbar-input::placeholder {
            color: white;
        }

        #bg-item {
            position: fixed;
            top: 0;
            width: 100%;
            height: var(--my-height);
            z-index: 0;  // 设置背景元素在最下层
        }

        #bg-item img {
            width: 100%;
            height: auto;
            object-fit: cover;
        }

        #scroll-item {
            position: absolute;
            top: var(--my-height);
            width: 100%;
            z-index: 1;  // 设置滚动元素在背景元素之上
        }

        #scroll-item img {
            width: 100%;
            height: auto;
            object-fit: cover;
        }

        #content {
            position: absolute;
            top: var(--my-height);
            height: 610px;
            width: 100%;
            background-color: #ececec;
            z-index: 2;  // 设置内容元素在滚动元素之上
        }
    </style>
</head>

从CSS部分就可以看出,实际上归纳起来,我把页面拆分为了 appbar、背景图层、滚动图层、内容层,通过 z-index 属性来控制层级关系(可以结合下面html部分)。代码接上:

    <!-- 作者信息 -->
    <!-- Author: 李俊才 -->
    <!-- Email: 291149494@163.com -->

    <!-- 许可证信息 -->
    <!-- LICENSE: MIT -->
<body>
    <div id="appbar">
        <div>我是appbar</div>
        <input id="appbar-input" type="text" placeholder="我是输入框">
    </div>
    <div id="bg-item">
        <img src="https://gw.alicdn.com/imgextra/i4/2212013333132/O1CN01DetIjE1Z0VMx4155t_!!2212013333132.jpg_Q75.jpg_.webp" alt="Image">
    </div>

    <div id="scroll-item">
        <img src="https://gitee.com/jacklee1995/example-pictures/raw/master/piano/jonathanvasquez8950_piano_795a8e31-a910-48aa-9eae-45b1602f7cba.png" alt="Image"/>
    </div>
    
    <div id="content">
        <h1>我是内容区域</h1>
    </div>

    <script>
        // 获取 appbar 以及appbar内的输入框元素节点
        const appbar = document.getElementById('appbar');
        const appbarInput = document.getElementById('appbar-input');
        // 获取滚动项节点,这是一个与内容节点差速滚动的元素
        const scrollItem = document.getElementById('scroll-item');
        // 获取内容节点
        const content = document.getElementById('content');
        // 定义页面滚动的最大距离,在这个距离内appbar的背景颜色和文字颜色会发生变化
        const maxScroll = 280;

        window.addEventListener("scroll", function() {
            let scrollTop = window.pageYOffset || document.documentElement.scrollTop;
            let opacity =  (scrollTop / maxScroll);
            opacity = opacity < 0 ? 0 : opacity;
            // 根据滚动距离动态改变appbar的背景颜色透明度
            appbar.style.background = `rgba(0, 123, 255, ${opacity})`;
            // 根据滚动距离动态改变appbar内文字的颜色透明度
            appbar.style.color = `rgba(255, 255, 255, ${opacity})`;
            appbarInput.style.opacity = `${opacity}`;
            // 根据滚动距离动态改变滚动项的位置
            scrollItem.style.top = `calc(var(--my-height) - ${scrollTop}px)`;
            // 根据滚动距离动态改变内容的位置
            content.style.top = `calc(var(--my-height) - ${scrollTop / 2}px)`;
        });
    </script>
</body>
</html>

所有的控制逻辑我在 scroll 监听中完成的,实际上就是对各个层的控制。

3. Flutter小试:我就不用SliverAppBar了

其实我的意思就是想自定义与滚动效果相关的appbar。既然需要滚动控制,而且使用 SliverAppBar 套参数也不是那么方便,即使这样的效果无非是控制一个盒子的透明度变化以及其它的位置移动。说来说去, SliverAppBar 也可以通过其它的组件实现,最后转为 Sliver 协议放入CustomScrollView不就可以了。

整体思路和 Web 中差不多,由于是分层,需要使用 StackPositioned 组件(相比于上一小节在Web中我们使用的是html+CSS的z-index,然后在 JS 代码中动态调整opacity)。

整体思路完全一样。于是,可以将上一个小节的 Web 代码 改用Flutter实现一下:

import 'package:flutter/material.dart';

// Author: 李俊才
// Email: 291149494@163.com
// https://blog.csdn.net/qq_28550263/article/details/134149018

class WebAppBarScaffold extends StatefulWidget {
  const WebAppBarScaffold({Key? key}) : super(key: key);

  
  State<WebAppBarScaffold> createState() => _WebAppBarScaffoldState();
}

class _WebAppBarScaffoldState extends State<WebAppBarScaffold> {
  double _opacity = 0.0;
  double _offsetImage = 0.0;
  double _offsetContent = 0.0;

  
  Widget build(BuildContext context) {
    return Scaffold(
      body: NotificationListener<ScrollNotification>(
        onNotification: (ScrollNotification scrollInfo) {
          if (scrollInfo is ScrollUpdateNotification) {
            setState(() {
              _opacity = scrollInfo.metrics.pixels / 280;
              _opacity = _opacity.clamp(0.0, 1.0);
              _offsetImage = scrollInfo.metrics.pixels * 1.5; // 修改这里
              _offsetContent = scrollInfo.metrics.pixels / 2; // 修改这里
            });
          }
          return true;
        },
        child: Stack(
          children: <Widget>[
            Positioned(
              top: 0,
              child: Image.network(
                'https://gw.alicdn.com/imgextra/i4/2212013333132/O1CN01DetIjE1Z0VMx4155t_!!2212013333132.jpg_Q75.jpg_.webp',
                width: MediaQuery.of(context).size.width,
                fit: BoxFit.cover,
              ),
            ),
            Positioned(
              top: 460 - _offsetImage,
              child: Image.network(
                'https://gitee.com/jacklee1995/example-pictures/raw/master/piano/jonathanvasquez8950_piano_795a8e31-a910-48aa-9eae-45b1602f7cba.png',
                width: MediaQuery.of(context).size.width,
                fit: BoxFit.cover,
              ),
            ),
            Positioned(
              top: 460 - _offsetContent,
              child: Container(
                color: Colors.grey[200],
                width: MediaQuery.of(context).size.width,
                height: 610,
                child: const Center(child: Text('我是内容区域')),
              ),
            ),
            Positioned(
              top: 0,
              child: Container(
                width: MediaQuery.of(context).size.width,
                height: 50,
                color: Colors.blue.withOpacity(_opacity),
                child: Row(
                  mainAxisAlignment: MainAxisAlignment.spaceBetween,
                  children: <Widget>[
                    Padding(
                      padding: const EdgeInsets.only(left: 10),
                      child: Text(
                        '我是appbar',
                        style: TextStyle(
                            color: Colors.white.withOpacity(_opacity)),
                      ),
                    ),
                    Padding(
                      padding: const EdgeInsets.only(right: 20),
                      child: Opacity(
                        opacity: _opacity,
                        child: const SizedBox(
                          width: 200, // 限定宽度
                          child: TextField(
                            decoration: InputDecoration(
                              hintText: '我是输入框',
                              hintStyle: TextStyle(color: Colors.white),
                            ),
                          ),
                        ),
                      ),
                    ),
                  ],
                ),
              ),
            ),
            ListView.builder(
              itemCount: 1,
              itemBuilder: (context, index) {
                return Container(
                    height: MediaQuery.of(context).size.height * 2);
              },
            ),
          ],
        ),
      ),
    );
  }
}

在这里插入图片描述
这里其实有一个小缺陷,就是滚动过头我没去做处理了。这里是一点小数学问题,就是计算中间层的图片相对于内容层图片滚动的位移值恰好为图片的高度时,让中间滚动图片层和内容层一起滚动,就可以避免看到中间滚动图层相比于内容层越来越远。读者可以尝试修改一下代码。

4. 结论

其实本文主要目的还是比较。可以看到,使用Web事件监听处理滚动事件,其实和Flutter中使用滚动控制差不多。对于一些复杂的效果,没有必要拘束于现有的组件,可以基于一些更加基础的部件构成复杂的效果。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1169667.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL数据库中不同数据类型字段关联后结果居然有这么大差异?

点击上方蓝字关注我 在数据库的世界里&#xff0c;数据的连接操作是至关重要的。但在处理关联表的字段的数据类型不同时&#xff0c;得到的结果经常会出乎预料。 1. 案例 1.1 数据库中先创建表及数据 -- 创建tb1 CREATE TABLE tb1 (id BIGINT NOT NULL PRIMARY KEY, NAME VARC…

掌握Maven和SpringBoot的灵活性:定制化lib目录和依赖范围

前言 在开发基于Maven和SpringBoot的项目时&#xff0c;我们经常会使用第三方库来满足需求。然而&#xff0c;有时候我们需要更灵活地控制这些库的依赖范围和加载方式。本文将介绍如何使用Maven和SpringBoot实现定制化的lib目录和依赖范围。经过如下定制化后&#xff0c;打包执…

【C语言】备战校赛Day3

日期:11.3 星期五 L1-007 念数字 题目描述 输入一个整数&#xff0c;输出每个数字对应的拼音。当整数为负数时&#xff0c;先输出fu字。十个数字对应的拼音如下&#xff1a; 0: ling 1: yi 2: er 3: san 4: si 5: wu 6: liu 7: qi 8: ba 9: jiu 输入描述 输入在一行中给出一个…

Spring Data Redis + RabbitMQ - 基于 string + hash 实现缓存,计数(高内聚)

目录 一、Spring Data Redis 1.1、缓存功能(分析) 1.2、案例实现 一、Spring Data Redis 1.1、缓存功能(分析) hash 类型存储缓存相比于 string 类型就有更多的更合适的使用场景. 例如,我有以下这样一个 UserInfo 信息 假设这样一个场景就是:万一只想获取其中某一个…

MySQL第三讲·SQL boy的CRUD操作

你好&#xff0c;我是安然无虞。 文章目录 增删查改&#xff1a;如何操作表中的数据&#xff1f;添加数据插入数据记录插入查询结果 删除数据修改数据查询数据select&#xff5c;where&#xff5c;group by&#xff5c;havingfromorder bylimit 增删查改&#xff1a;如何操作表…

C++标准模板(STL)- 类型支持 (类型属性,is_literal_type,is_polymorphic,is_empty)

类型特性 类型特性定义一个编译时基于模板的结构&#xff0c;以查询或修改类型的属性。 试图特化定义于 <type_traits> 头文件的模板导致未定义行为&#xff0c;除了 std::common_type 可依照其所描述特化。 定义于<type_traits>头文件的模板可以用不完整类型实例…

uniapp小程序刮刮乐抽奖

使用canvas画布画出刮刮乐要被刮的图片&#xff0c;使用移动清除画布。 当前代码封装为刮刮乐的组件&#xff1b; vue代码&#xff1a; <template><view class"page" v-if"merchantInfo.cdn_static"><image class"bg" :src&q…

强化学习的动态规划二

一、典型示例 考虑如下所示的44网格。 图1 非终端状态为S {1, 2, . . . , 14}。在每个状态下有四种可能的行为&#xff0c;A {up, down, right, left}&#xff0c;这些行为除了会将代理从网格上移走外&#xff0c;其他都会确定性地引起相应的状态转换。因此&#xff0c;例如&…

VMware产品收集日志方法汇总

概述 vCenter日志是一个用于存储与vSphere环境相关的各种活动、事件和警告的日志系统。通过收集并分析vCenter日志&#xff0c;管理员可以获得有关其虚拟化环境的重要洞察和故障排除信息。 vCenter日志由多个组件组成&#xff0c;包括vCenter Server、ESXi主机和其他vSphere组…

软件测试面试题:Web 端测试和 App 端测试有何不同

Web 端测试和 App 端测试是针对不同平台的上的应用进行测试&#xff0c;Web应用和App端的应用实现方式不同&#xff0c;测试时的侧重点也不一样。 今天这篇文章就来介绍下两者的不同之处以及测试时的侧重点。 Web 端应用和 App 端应用的区别 平台兼容性 Web 端应用可以在任何…

C#开源的一个能利用Windows通知栏背单词的软件 - ToastFish

前言 今天给大家推荐一个C#开源且免费的能利用Windows通知栏背单词的软件&#xff0c;可以让你在上班、上课等恶劣环境下安全隐蔽地背单词&#xff08;利用摸鱼时间背单词的软件&#xff09;&#xff1a;ToastFish。 操作系统要求 目前该软件只支持Windows10及以上系统&…

【C语言】C语言⻘蛙跳台阶问题--递归问题

&#x1f308;write in front :&#x1f50d;个人主页 &#xff1a; 啊森要自信的主页 本期专栏&#x1f525;&#xff1a;本期将分享一些猜数字小游戏怎么一步一步实现的 &#x1f308;作者寄语 &#x1f308;&#xff1a; 小菜鸟的力量不在于它的体型&#xff0c;而在于它内心…

Fourier分析导论——第3章——Fourier级数的收敛性(E.M. Stein R. Shakarchi)

第 3 章 Fourier级数的收敛性(Convergence of Fourier Series) The sine and cosine series, by which one can represent an arbitrary function in a given interval, enjoy among other remarkable properties that of being convergent. This property did not escape…

nodejs+vue网上商城系统系统-毕业设计

网上商城系统的架构设计通常分为三层&#xff1a;客户端层、应用层和数据层。 客户端层&#xff1a;客户端层是用户与系统交互的界面&#xff0c;包括Web页面、移动App等。用户可以通过客户端层进行商品浏览、下单、支付等操作。应用层&#xff1a;应用层是业务逻辑处理的中心&…

一个 不用氪金 也能让你变强的 VSCode 插件 Ai

哈喽,大家好 我是 彩色之外&#x1f468;&#x1f3fb;‍&#x1f4bb;。今天给大家推荐一款不用充钱也能让你变强的 vscode 插件 通义灵码&#xff08;TONGYI Lingma&#xff09;&#xff0c;可以称之为 Copilot 的替代甜品 &#x1f4aa; &#x1f440; 前期回顾 NPM- 滚动进…

BIOS开发笔记 – 显示

UEFI启动流程跑完前三阶段,UEFI环境的准备基本完成,到BDS阶段的任务就是准备引导OS。在此之前还需要使一些必要的硬件工作起来,比如键盘设备,屏幕等,怎么让屏幕工作呢?简单的说就是执行其相关的UEFI驱动。要注意一下的是,这里所说的驱动并不是屏幕的驱动,而是GPU的驱动…

Proteus仿真--1602LCD随机模拟显示乘法口诀(仿真文件+程序)

本文主要介绍基于51单片机的1602LCD随机模拟显示乘法口诀实验&#xff08;完整仿真源文件及代码见文末链接&#xff09; 仿真图如下 其中上方主要是1602LCD液晶显示&#xff0c;使用方法点击左下方的按键开关实现不同的乘数切换&#xff0c;按照乘法表进行 仿真运行视频 Prot…

Android Studio的Java项目种运行main()的方法

首先随便在项目里面建一个类 public class TestSocket {public static void main(String[] args) {System.out.println("hahah");}}直接运行时会直接报错的 在项目的 .idea 添加代码 <option name"delegatedBuild" value"false"/> 再…

Centos7安装Elasticsearch和Kibana 记录(无坑版)

说明&#xff1a; 本文命令全部标红 elasticsearch使用版本是7.17.5 ik分词器也是7.17.5 虚拟机配置&#xff1a;2核4g centos版本&#xff1a;7 1、获取elasticsearch的安装包 下载&#xff1a;wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-7.17…

《网络协议》01. 基本概念

title: 《网络协议》01. 基本概念 date: 2022-08-30 09:50:52 updated: 2023-11-04 07:28:52 categories: 学习记录&#xff1a;网络协议 excerpt: 互联网、网络互连模型&#xff08;OSI&#xff0c;TCP/IP&#xff09;、计算机通信基础。 comments: false tags: top_image: /i…