Blazor实现菜单动画
想到动画,你可能会去安装Blazor的动画组件BlazorAnimate,然后使用它。本人初学,暂时我也不知道原理,先不用组件,自己实现吧。虽然项目中我用了AntDesignBlazor,但是我忘了使用它的菜单组件,我用的菜单组件还是VS2022自动生成的,后来我把这个菜单改造了一下,支持多级菜单,加了展示收缩箭头,那就在这基础上做吧。
1. 引用jQuery
这里使用jquery的animate方法实现动画
在wwwroot/js目录放一个jquery-1.9.1.js文件,然后在html(或_Layout.cshtml文件)中引入该js
<script src="js/jquery-1.9.1.js"></script>

2. 为左侧菜单组件NavMenu.razor添加一个js文件:NavMenu.razor.js

内容如下:
export function animate(index) { //index是菜单编码 2位数是一级菜单,4位数是二级菜单,以此类推
let time = 200;
let content = $(".content" + index);
let h = content.height() + "px";
content.css("overflow", "hidden");
if (index <= 99 || content.hasClass("collapse")) { //index<=99表示一级菜单,一级菜单只有展开动画,没有收缩动画;content.hasClass("collapse")表示当前是收缩状态。
content.css("height", "0");
//展开动画
content.animate({
height: h
}, time, "linear", () => {
content.css("height", "auto");
});
} else { //不是一级菜单并且当前是展开状态,将执行收缩动画
content.css("height", "auto");
//收缩动画
content.animate({
height: 0
}, time, "linear", () => {
setTimeout(function () { //延迟执行,否则会导致闪烁
content.css("height", "auto");
}, 100);
});
return [time]; //收缩时,需要等待收缩动画展示完成,再隐藏菜单容器div
}
return [0];
}
3. 在Blazor组件NavMenu.razor文件中引入该js
@inject IJSRuntime _js;
@code {
IJSObjectReference _module;
protected override async void OnAfterRender(bool firstRender)
{
if (firstRender)
{
_module = await _js.InvokeAsync<IJSObjectReference>("import", "./Shared/NavMenu.razor.js");
}
}
}
4. 调用js方法实现菜单动画
@code{
...省略此处的代码
private async Task ToggleMenuClick(int index)
{
object[] args = new object[] { index };
object[] objs = await _module.InvokeAsync<object[]>("animate", args);
int time = ((JsonElement)objs[0]).GetInt32();
if (time > 0) { await Task.Delay(time); } //time大于0,表示需要等待收缩动画展示完成,再隐藏菜单容器div
if (index > 99)
{
if (_dict.ContainsKey(index))
{
_dict[index] = !_dict[index]; //记录非一级菜单的展开状态
}
else
{
_dict[index] = false; //记录非一级菜单的展开状态
}
}
else
{
_index = index; //_index是当前展示的菜单编码
}
}
...省略此处的代码
}
效果图

NavMenu.razor页面完整代码如下
@using System.Text.Json;
@inject IJSRuntime _js;
<!-- 菜单组 -->
<div class="top-row ps-3 navbar navbar-dark" @onclick="@(async ()=>await ToggleMenuClick(01))">
<div class="container-fluid">
<a class="navbar-brand" href="javascript:void(0);">任务管理</a>
<button title="Navigation menu" class="navbar-toggler">
<span class="navbar-toggler-icon"></span>
</button>
<div class="@ArrowClass(01)"></div>
</div>
</div>
<!-- 菜单组内容 -->
<div class="@MenuGroupContentClass(01)">
<nav class="flex-column">
<div class="nav-item px-3">
<NavLink class="nav-link" href="kpTask/taskList">
<span class="oi oi-list" aria-hidden="true"></span> 任务列表
</NavLink>
</div>
<div class="nav-item px-3">
<NavLink class="nav-link" href="kpTask/taskRunList">
<span class="oi oi-list" aria-hidden="true"></span> 任务执行记录
</NavLink>
<!-- 二级菜单组 -->
<NavLink class="nav-link" href="javascript:void(0);" @onclick="@(()=> ToggleMenuClick(0101))" style="display:;">
<span class="oi oi-list" aria-hidden="true"></span> 测试二级菜单组
<div class="@ArrowClass(0101)"></div>
</NavLink>
<!-- 二级菜单组内容 -->
<div class="@MenuGroupContentClass(0101)" style="display:;">
<nav class="flex-column">
<div class="nav-item px-3">
<NavLink class="nav-link" href="counter">
<span class="oi oi-list" aria-hidden="true"></span> 测试二级菜单
</NavLink>
<!-- 三级菜单组 -->
<NavLink class="nav-link" href="javascript:void(0);" @onclick="@(()=> ToggleMenuClick(010101))" style="display:;">
<span class="oi oi-list" aria-hidden="true"></span> 测试三级菜单组
<div class="@ArrowClass(010101)"></div>
</NavLink>
<!-- 三级菜单组内容 -->
<div class="@MenuGroupContentClass(010101)" style="display:;">
<nav class="flex-column">
<div class="nav-item px-3">
<NavLink class="nav-link" href="counter">
<span class="oi oi-list" aria-hidden="true"></span> 测试三级菜单
</NavLink>
</div>
</nav>
</div>
</div>
<div class="nav-item px-3">
<NavLink class="nav-link" href="counter">
<span class="oi oi-list" aria-hidden="true"></span> 测试二级菜单2
</NavLink>
</div>
</nav>
</div>
</div>
</nav>
</div>
<!-- 菜单组 -->
<div class="top-row ps-3 navbar navbar-dark" @onclick="@(async ()=>await ToggleMenuClick(02))">
<div class="container-fluid">
<a class="navbar-brand" href="javascript:void(0);">任务管理</a>
<button title="Navigation menu" class="navbar-toggler">
<span class="navbar-toggler-icon"></span>
</button>
<div class="@ArrowClass(01)"></div>
</div>
</div>
<!-- 菜单组内容 -->
<div class="@MenuGroupContentClass(02)">
<nav class="flex-column">
<div class="nav-item px-3">
<NavLink class="nav-link" href="kpTask/taskList">
<span class="oi oi-list" aria-hidden="true"></span> 任务列表
</NavLink>
</div>
<div class="nav-item px-3">
<NavLink class="nav-link" href="kpTask/taskRunList">
<span class="oi oi-list" aria-hidden="true"></span> 任务执行记录
</NavLink>
<!-- 二级菜单组 -->
<NavLink class="nav-link" href="javascript:void(0);" @onclick="@(()=> ToggleMenuClick(0201))" style="display:none;">
<span class="oi oi-list" aria-hidden="true"></span> 测试二级菜单组
<div class="@ArrowClass(0201)"></div>
</NavLink>
<!-- 二级菜单组内容 -->
<div class="@MenuGroupContentClass(0201)" style="display:none;">
<nav class="flex-column">
<div class="nav-item px-3">
<NavLink class="nav-link" href="counter">
<span class="oi oi-list" aria-hidden="true"></span> 测试二级菜单
</NavLink>
</div>
</nav>
</div>
</div>
</nav>
</div>
<!-- 菜单组 -->
<div class="top-row ps-3 navbar navbar-dark" @onclick="@(async ()=>await ToggleMenuClick(03))">
<div class="container-fluid">
<a class="navbar-brand" href="javascript:void(0);">任务管理</a>
<button title="Navigation menu" class="navbar-toggler">
<span class="navbar-toggler-icon"></span>
</button>
<div class="@ArrowClass(01)"></div>
</div>
</div>
<!-- 菜单组内容 -->
<div class="@MenuGroupContentClass(03)">
<nav class="flex-column">
<div class="nav-item px-3">
<NavLink class="nav-link" href="kpTask/taskList">
<span class="oi oi-list" aria-hidden="true"></span> 任务列表
</NavLink>
</div>
<div class="nav-item px-3">
<NavLink class="nav-link" href="kpTask/taskRunList">
<span class="oi oi-list" aria-hidden="true"></span> 任务执行记录
</NavLink>
<!-- 二级菜单组 -->
<NavLink class="nav-link" href="javascript:void(0);" @onclick="@(()=> ToggleMenuClick(0301))" style="display:none;">
<span class="oi oi-list" aria-hidden="true"></span> 测试二级菜单组
<div class="@ArrowClass(0301)"></div>
</NavLink>
<!-- 二级菜单组内容 -->
<div class="@MenuGroupContentClass(0301)" style="display:none;">
<nav class="flex-column">
<div class="nav-item px-3">
<NavLink class="nav-link" href="counter">
<span class="oi oi-list" aria-hidden="true"></span> 测试二级菜单
</NavLink>
</div>
</nav>
</div>
</div>
</nav>
</div>
@code {
int _index = 01;
Dictionary<int, bool> _dict = new Dictionary<int, bool>();
IJSObjectReference _module;
protected override async void OnAfterRender(bool firstRender)
{
if (firstRender)
{
_module = await _js.InvokeAsync<IJSObjectReference>("import", "./Shared/NavMenu.razor.js");
}
}
private async Task ToggleMenuClick(int index)
{
object[] args = new object[] { index };
object[] objs = await _module.InvokeAsync<object[]>("animate", args);
int time = ((JsonElement)objs[0]).GetInt32();
if (time > 0) { await Task.Delay(time); }
if (index > 99)
{
if (_dict.ContainsKey(index))
{
_dict[index] = !_dict[index];
}
else
{
_dict[index] = false;
}
}
else
{
_index = index;
}
}
private string MenuGroupContentClass(int index)
{
if (index > 99)
{
if (!_dict.ContainsKey(index) || _dict[index])
{
return $"collapse content{index}";
}
else
{
return $"content{index}";
}
}
else
{
if (index == _index)
{
return $"content{index}";
}
else
{
return $"collapse content{index}";
}
}
}
private string ArrowClass(int index)
{
if (index > 99)
{
if (!_dict.ContainsKey(index) || _dict[index])
{
return "arrow-collapse arrow-float";
}
else
{
return "arrow arrow-float";
}
}
else
{
if (index == _index)
{
return "arrow";
}
else
{
return "arrow-collapse";
}
}
}
}
Blazor实现菜单动画的更多相关文章
- Android菜单(动画菜单、360波纹菜单)
Android菜单(动画菜单.360波纹菜单) 前言:Android菜单常用集合:FragmentTabHost系统菜单.上移式菜单.360波纹菜单.展开式菜单.详解注释,可直接拿来用! 效果: ...
- CSS3--点击弹出菜单动画
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <meta name ...
- 【译】仿Taasky的3D翻转菜单动画实现
最终效果 最终效果 开始 首先下载并打开一个事先搭好架子的Demo,然后来分析一下.这个Demo包含一个主页和详情页,其中MenuViewController继承自UITableViewControl ...
- jquery css3 手机菜单动画综合版
html <header> <a id="go-back" href="javascript:window.location.back(-1)" ...
- iOS-一个弹出菜单动画视图开源项目分享
相似于Tumblr公布button的弹出视图 使用非常easy: 初始化: @property (nonatomic, strong) XWMenuPopView *myMenuPopView; - ...
- No.1 - 制作一个简单的菜单动画效果---百度IFE
最近比较闲,在家做点训练 http://ife.baidu.com/course/detail/id/18?t=1527144851578#learn CSS3新特性,兼容性,兼容方法总结 https ...
- 基于HTML5和SVG的手机菜单动画
在线演示 本地下载
- vue基于 element-ui 实现菜单动画效果,任意添加 li 个数均匀撑满 ul 宽度
%)%)%%%))) .) .) .) .) .) .) .) .) .) .) .) .) .) .) .) .% %% %deg);}
- Flutter Demo: 径向菜单动画
video import 'dart:math'; import 'package:flutter/material.dart'; import 'package:vector_math/vector ...
- IOS中的动画菜单
SvpplyTable(可折叠可张开的菜单动画) 允许你简单地创建可折叠可张开的菜单动画效果,灵感来自于Svpply app.不同表格项目使用JSON定义,你可以定义每个菜单项和任何子菜单,为每个项目 ...
随机推荐
- 【教程】cpp转python Nanobind 实践 加速轻量版 pythonbind11
主要是尝试一下把c++这边的函数封装打包给python用,选择nanobind的原因是:1. 优化速度快,2. 生成二进制包小,不过pythonbind11是更为广泛知道的,nanobind也是pyt ...
- 微信小程序敏感内容检测
获取access_token access_token是公众号的全局唯一接口调用凭据,公众号调用各接口时都需使用access_token.开发者需要进行妥善保存.access_token的存储至少要保 ...
- 【python】【报错:pip中第三库下载成功,但是pycharm却没有显示】一步解决
解决方案: 直接在这个目录下安装第三方库
- 如何用excel制作图表?
Excel是一个非常强大的电子表格软件,其中包含了很多绘制图表的功能.下面是一些基本步骤,可以帮助你用Excel制作图表: 打开Excel并输入数据.在Excel中,首先需要输入需要制作图表的数据.可 ...
- Centos 7在线安装JDK1.8+Tomcat+MySQL8.0+Nginx
一.安装JDK 注: 以下命令环境在Xshell中进行. 温馨提示: 在线安装方式需要有网速的前提,有的软件下载需要大量时间. 1.查询出系统自带的OpenJDK及版本 rpm -qa | grep ...
- Head First Java学习:第十章-数字很重要
1.Math 方法:最接近全局的方法 一种方法的行为不依靠实例变量值,方法对参数执行操作,但是操作不受实例变量状态影响,那么为了执行该方法去堆上建立对象实例比较浪费. 举例: Math mathOb ...
- matlab 2018b 下载链接
matlab 2018b 功能强大下载地址为 https://pan.baidu.com/s/1QZO35BtzcIkh_yPYRIGVWg
- 传统GIS与数字孪生结合带来的改变
传统的地理信息系统(GIS)在地理数据的收集.存储和分析方面发挥着重要作用,而数字孪生技术则通过虚拟模型的构建与真实世界进行交互和模拟.将传统GIS与数字孪生技术相结合,不仅增强了地理数据的可视化和分 ...
- STM32CubeMX教程4 EXTI 按键外部中断
1.准备材料 开发板(STM32F407G-DISC1) ST-LINK/V2驱动 STM32CubeMX软件(Version 6.10.0) keil µVision5 IDE(MDK-Arm) 2 ...
- Gh0st木马
https://www.secrss.com/articles/50209 Gh0st是一种远程控制软件,它可以在被攻击的计算机上运行并允许攻击者远程控制该计算机.为了查找Gh0st的进程.文件.注册 ...