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定义,你可以定义每个菜单项和任何子菜单,为每个项目 ...
随机推荐
- 比较Spring Security6.X 和 Spring Security 5.X的不同
项目使用了SpringBoot3 ,因此 SpringSecurity也相应进行了升级 版本由5.4.5升级到了6.1.5 写法上发生了很大的变化,最显著的变化之一就是对 WebSecurityCon ...
- 发现AI自我意识:从理解到思维
广义"理解"已经实现 在最新的人工智能系统中,我们经常可以观察到一种类似"理解"的能力.这种广义的"理解"能力,主要建立在两个基础之上:海量 ...
- 每天5分钟复习OpenStack(十一)Ceph部署
在之前的章节中,我们介绍了Ceph集群的组件,一个最小的Ceph集群包括Mon.Mgr和Osd三个部分.为了更好地理解Ceph,我建议在进行部署时采取手动方式,这样方便我们深入了解Ceph的底层.今天 ...
- windows下tomcat开机自启动
在Windows下,可以通过以下步骤将Tomcat设置为开机自启动: 1. 打开Tomcat安装目录:通常情况下,Tomcat的安装目录位于`C:\Program Files\Apache Softw ...
- 解决报错:Java 8 date/time type `java.time.Duration` not supported by default: add Module "com.fasterxml.jackson.datatype:jackson-datatype-jsr310" to enable handling
1.错误信息: Java 8 date/time type java.time.Duration not supported by default: add Module "com.fast ...
- Chrome扩展开发系列开篇
大家好,我是 dom 哥.这是我关于 Chrome 扩展开发的系列文章,感兴趣的可以 点个小星星. 浏览器现状 研究机构 Statcounter 发布了 2023 年 9 月报告,揭示了有关浏览器的最 ...
- bash shell笔记整理——head命令
作用 我直接搬运官方英文了,这英文真的简单直白了,我英语辣鸡...毕竟我学历不行,觉得翻译不好就直接自己来吧.. Print the first 10 lines of each FILE to st ...
- 37. 干货系列从零用Rust编写负载均衡及代理,负载均衡中try_files实现
wmproxy wmproxy已用Rust实现http/https代理, socks5代理, 反向代理, 静态文件服务器,四层TCP/UDP转发,七层负载均衡,内网穿透,后续将实现websocket代 ...
- 牛客刷Java记录第6天
第一题 一个文件中的字符要写到另一个文件中,首先需要( ). A System.out.print (buffer[i]); B FileOutputStream fout = new FileOut ...
- Blazor快速开发框架Known-V2.0.0
Known2.0 Known是基于Blazor的企业级快速开发框架,低代码,跨平台,开箱即用,一处代码,多处运行. 官网:http://known.pumantech.com Gitee: https ...