为什么要封装组件

最近写MAUI Blazor的时候,总是苦于对移动端没有什么好的支持,没有一个能左右滑动的tab切换组件。

既然没有,那就自己封装一个。

简单了解轮播图、tab切换的库之后,决定使用根据Swiper来封装。

Swiper是js的插件,所以需要js与.NET互操作,有以下两点

  • 当js中的索引变化时,调用.NET方法改变Blazor中的索引;
  • 当Blazor中的索引变化时,调用js方法改变js中的索引。

准备工作

下载Swiper,一个js文件和一个css文件,swiper-bundle.min.jsswiper-bundle.min.css

最好用7的版本,8的版本会有一些兼容问题。

将文件引入index.html

Blazor部分

如官网所示,swiper的结构如下

<div class="swiper">
<div class="swiper-wrapper">
<div class="swiper-slide">slider1</div>
<div class="swiper-slide">slider2</div>
<div class="swiper-slide">slider3</div>
</div>
</div>

所以我们的组件分为两部分,一个外壳,一个是可重复的内容。

分别命名为SwiperTabItems.razorSwiperTabItem.razor

SwiperTabItems.razor
@inject IJSRuntime JSRuntime

<div class="@("swiper " + Id)">
<div class="swiper-wrapper">
@ChildContent
</div>
</div>
@code{
[Parameter]
public int Value
{
get => _value;
set
{
if (_value != value)
{
_value = value;
if(AfterFirstRender)
{
UpdateSwiper(value);
}
}
}
}
[Parameter]
public EventCallback<int> ValueChanged { get; set; }
[Parameter]
public RenderFragment? ChildContent { get; set; } private int _value = 0;
private bool AfterFirstRender;
private string Id = "swiper" + Guid.NewGuid().ToString(); protected override async Task OnAfterRenderAsync(bool firstRender)
{
if (firstRender && ChildContent is not null)
{
var dotNetCallbackRef = DotNetObjectReference.Create(this);
await JSRuntime!.InvokeVoidAsync("swiperInit", new object[4] { dotNetCallbackRef, "UpdateValue", Id, Value });
AfterFirstRender = true;
}
}
private async void UpdateSwiper(int value)
{
await JSRuntime!.InvokeVoidAsync($"{Id}.slideTo", new object[1] { value });
}
[JSInvokable]
public async Task UpdateValue(int value)
{
_value = value;
if (ValueChanged.HasDelegate)
{
await ValueChanged.InvokeAsync(value);
}
StateHasChanged();
} public async void Dispose()
{
await JSRuntime!.InvokeVoidAsync($"{Id}.destroy", new object[2] { true, true });
}
}
SwiperTabItem.razor
<div class="swiper-slide">
@ChildContent
</div> @code {
[Parameter]
public RenderFragment? ChildContent { get; set; }
}

注:

1.id使用guid是为了如果存在多个组件,js的实例化不会重复

2.OnAfterRenderAsync是blazor的一个生命周期,位于渲染之后,将swiper的初始化实例放在这个位置才能正确初始化。firstRender 代表第一次渲染。

js部分

如swiper官网所示,需要初始化实例

<script>
var mySwiper = new Swiper('.swiper', {
autoplay: true,//可选选项,自动滑动
})
</script>

所以我们需要写一个init-swiper.js文件,包含初始化实例的函数,以供Blazor的调用

init-swiper.js
function swiperInit(dotNetCallbackRef, callbackMethod,id,index) {
console.log('Entered initSwiper!');
let className = "." + id;
window[id] = new Swiper(className, {
observer: true,
observeParents: true,
observeSlideChildren: true,
autoHeight: true,
initialSlide: index,
on: {
slideChangeTransitionStart: function () {
dotNetCallbackRef.invokeMethodAsync(callbackMethod, this.activeIndex);
//alert(this.activeIndex);//切换结束时,告诉我现在是第几个slide
},
}
});
}

init-swiper.js引入index.html

使用

<SwiperTabItems @bind-Value="tabs">
<SwiperTabItem>
你的内容1
</SwiperTabItem>
<SwiperTabItem>
你的内容2
</SwiperTabItem>
<SwiperTabItem>
你的内容3
</SwiperTabItem>
</SwiperTabItems>

演示

这个封装的比较简单,如果需要更多的属性,可以看看swiper的文档

封装一个可以左右滑动的Blazor组件的更多相关文章

  1. 封装一个优雅的element ui表格组件

    现在做后台系统用vue + elementUI 的越来越多,那element ui的 el-table 组件肯定也离不开.虽然element ui的table组件很好.但是表格和分页是分离的.每次写表 ...

  2. 使用vue.js封装一个包含图片的跑马灯组件

    初衷: 学习完Vuejs后,来准备练习仿写一下老东家的门户页面,主要是为了熟悉一下常用插件的使用,比如video.js,wow.js,swiper等等:而其中涉及到一个包含图片跑马灯组件,大概长这样( ...

  3. Vue 2.x折腾记 - (17) 基于Ant Design Vue 封装一个配置式的表单组件

    前言 写了个类似上篇搜索的封装,但是要考虑的东西更多. 具体业务比展示的代码要复杂,篇幅太长就不引入了. 效果图 2019-04-25 添加了下拉多选的渲染,并搜索默认过滤文本而非值 简化了渲染的子组 ...

  4. 小程序封装一个有输入框的modal层组件

    其实很简单,就是在modal中添加新的 input <view> <modal class="modal" wx:if="{{!hiddenModal} ...

  5. Blazor组件自做五 : 使用JS隔离封装Google地图

    Blazor组件自做五: 使用JS隔离封装Google地图 运行截图 演示地址 正式开始 1. 谷歌地图API 谷歌开发文档 开始学习 Maps JavaScript API 的最简单方法是查看一个简 ...

  6. Blazor组件自做六 : 使用JS隔离封装Baidu地图

    1. 运行截图 演示地址 2. 在文件夹wwwroot/lib,添加baidu子文件夹,添加baidumap.js文件 2.1 跟上一篇类似,用代码方式异步加载API,脚本生成新的 body > ...

  7. Blazor组件自做一 : 使用JS隔离封装viewerjs库

    Viewer.js库是一个实用的js库,用于图片浏览,放大缩小翻转幻灯片播放等实用操作 本文相关参考链接 JavaScript 模块中的 JavaScript 隔离 Viewer.js工程 Blazo ...

  8. Go/Python/Erlang编程语言对比分析及示例 基于RabbitMQ.Client组件实现RabbitMQ可复用的 ConnectionPool(连接池) 封装一个基于NLog+NLog.Mongo的日志记录工具类LogUtil 分享基于MemoryCache(内存缓存)的缓存工具类,C# B/S 、C/S项目均可以使用!

    Go/Python/Erlang编程语言对比分析及示例   本文主要是介绍Go,从语言对比分析的角度切入.之所以选择与Python.Erlang对比,是因为做为高级语言,它们语言特性上有较大的相似性, ...

  9. Blazor组件自做八 : 使用JS隔离封装屏幕键盘kioskboard.js组件

    1. 运行截图 演示地址 2. 在文件夹wwwroot/lib,添加kioskboard子文件夹,添加kioskboards.js文件 2.1 常规操作,懒加载js库, export function ...

  10. Blazor组件自做三 : 使用JS隔离封装ZXing扫码

    Blazor组件自做三 : 使用JS隔离封装ZXing扫码 本文基础步骤参考前两篇文章 Blazor组件自做一 : 使用JS隔离封装viewerjs库 Blazor组件自做二 : 使用JS隔离制作手写 ...

随机推荐

  1. 33-webpack详细配置output

    const { resolve } = require('path') const HtmlWebpackPlugin = require('html-webpack-plugin') module. ...

  2. javasec(四)序列化与反序列化基本原理

    title: javasec(四)序列化与反序列化基本原理 tags: - javasec - 反序列化 categories: - javasec cover: 'https://blog-1313 ...

  3. ffmpeg音视频基础学习

    ffmpeg音视频基础学习 从去年开始了解音视频,中间也由于项目的需要,学习过ffmpeg.live555.以及QTAV框架,一直没总结过,现在大致总结下音视频中的常见词汇,后续慢慢更新添加!博客也会 ...

  4. 笔记八:linux系统编程之IO

    笔记:linux系统编程之IO 应用层     内核层     硬件层 应用层:数据结构 .java.android.C.C++,C#: l inux高级编程:涉及内核为应用层提供接口函数: 内核五大 ...

  5. 【机器学习与深度学习理论要点】07.A/B测试的概念及用法

    1)什么是A/B测试? A/B测试就是两种模型同时运行,并在实际环境中验证其效果的方式.在互联网公司中,A/B测试是验证新模块.新功能.新产品是否有效,新算法.新模型的效果是否有提升,新设计是否收到用 ...

  6. 聚合短信PHP代码示例短信接口调用CURL方法

    聚合的短信相信大家已经做多了吧,网上的代码看了下就是感觉太繁琐了,不过网上的也是比较好的,用的是post方法,更安全,因我们的项目是在服务器上请求,又绑定了白名单 ,所以弄了个简单点的自己用,参考如下 ...

  7. Vue中使用富文本编辑器

    原文链接:https://blog.csdn.net/qq_45695853/article/details/114635009

  8. 2022-04-27:用go语言重写ffmpeg的remuxing.c示例。

    2022-04-27:用go语言重写ffmpeg的remuxing.c示例. 答案2022-04-27: ffmpeg的remuxing.c是一个用于将多媒体文件从一种容器格式转换为另一种容器格式的命 ...

  9. docker无法启动,报错grpc: addrConn.createTransport failed to connect to {unix:///run/containerd/containerd.

    docker无法启动,报错.k8s的pod镜像加载失败. 解法方法: 删除/var/lib/docker/和/var/lib/containerd/ 这两个文件夹,重起docker服务. 问题完美解决 ...

  10. svn is already locked 最终解决方案

    今日执行项目更新时,手贱点击了cancel 中断了操作,最后导致项目被锁,杯具了. 首先想到了Clean up 直接提示 看来不行呀 -// 省略 n 多种尝试 最后使用删除db 中的 lock 表来 ...