前言

前几天工作中在做项目的时候,需要实现一个动态锚点的效果,具体效果如下:
如果是传统项目,这个效果就非常简单。但是放到 Vue 中,就有两大难题:
1. 在没有 jQuery 的 animate() 方法的情况下,如何实现平滑滚动?
2. 如何监听页面滚动事件?
在浏览了大量文章、进行多次尝试之后,终于解决了这些问题
期间主要涉及到了 setTimeout 的递归用法,和 Vue 生命周期中的 mounted
一、锚点实现
在实现平滑滚动之前,得先确保基本的锚点功能
如果没有其他要求,直接用 <a href="#id" rel="external nofollow" > 是最简单粗暴的办法
但是为了满足后续的要求,必须另外想办法
首先在父组件(表单)中添加 class="d_jump" 作为钩子
然后在子组件中添加一个 jump 方法
jump (index) {
let jump = document.querySelectorAll('.d_jump')
// 获取需要滚动的距离
let total = jump[index].offsetTop
// Chrome
document.body.scrollTop = total
// Firefox
document.documentElement.scrollTop = total
// Safari
window.pageYOffset = total
},
通过 offsetTop 获取对象到窗体顶部的距离,然后赋值给 scrollTop,就能实现锚点的功能
需要注意的是:各浏览器下获取 scrollTop 有所差异
Chrome: document.body.scrollTop
Firefox: document.documentElement.scrollTop
Safari: window.pageYOffset
二、平滑滚动
仅仅是锚点是不够的,这样的跳转十分突兀
为了更好的用户体验 ,需要将滚动的过程展现出来
如果有 jQuery 实现平滑滚动就非常简单:
$('html body').animate({scrollTop: total}, 500);
可惜没如果~~
在看了好些文章之后,有了一个大概的开发思路:
将总距离分成很多小段,然后每隔一段时间跳一段
只要每段的时间足够短,频次足够多,在视觉上就是正常的平滑滚动了
// 平滑滚动,时长500ms,每10ms一跳,共50跳
// 获取当前滚动条与窗体顶部的距离
let distance = document.documentElement.scrollTop || window.pageYOffset || document.body.scrollTop
// 计算每一小段的距离
let step = total / 50
(function smoothDown () {
if (distance < total) {
distance += step
// 移动一小段
document.body.scrollTop = distance
document.documentElement.scrollTop = distance
window.pageYOffset = distance
// 设定每一次跳动的时间间隔为10ms
setTimeout(smoothDown, 10)
} else {
// 限制滚动停止时的距离
document.body.scrollTop = total
document.documentElement.scrollTop = total
window.pageYOffset = total
}
})()
实际情况下,得考虑向上滚动和向下滚动两种情况,完整的代码为:
jump (index) {
// 用 class="d_jump" 添加锚点
let jump = document.querySelectorAll('.d_jump')
let total = jump[index].offsetTop
let distance = document.documentElement.scrollTop || window.pageYOffset || document.body.scrollTop
// 平滑滚动,时长500ms,每10ms一跳,共50跳
let step = total / 50
if (total > distance) {
smoothDown()
} else {
let newTotal = distance - total
step = newTotal / 50
smoothUp()
}
function smoothDown () {
if (distance < total) {
distance += step
document.body.scrollTop = distance
document.documentElement.scrollTop = distance
window.pageYOffset = distance
setTimeout(smoothDown, 10)
} else {
document.body.scrollTop = total
document.documentElement.scrollTop = total
window.pageYOffset = total
}
}
function smoothUp () {
if (distance > total) {
distance -= step
document.body.scrollTop = distance
document.documentElement.scrollTop = distance
window.pageYOffset = distance
setTimeout(smoothUp, 10)
} else {
document.body.scrollTop = total
document.documentElement.scrollTop = total
window.pageYOffset = total
}
}
}
三、修改锚点状态
在上面展示的效果中,当页面滚动的时候,锚点的激活状态会有相应的改变
其实这个效果并不难,只需要监听页面滚动事件,然后根据滚动条的距离修改锚点状态就可以了
但是在 Vue 中,应该在什么地方监听滚动事件呢?
mounted: function () {
this.$nextTick(function () {
window.addEventListener('scroll', this.onScroll)
})
},
methods: {
onScroll () {
let scrolled = document.documentElement.scrollTop || window.pageYOffset || document.body.scrollTop
// 586、1063分别为第二个和第三个锚点对应的距离
if (scrolled >= 1063) {
this.steps.active = 2
} else if (scrolled < 1063 && scrolled >= 586) {
this.steps.active = 1
} else {
this.steps.active = 0
}
}
}
上面的代码中,我先写了一个修改锚点状态的方法 onScroll,然后在 mounted 中监听 scroll 事件,并执行 onScroll 方法
mounted 是 Vue 生命周期中的一个状态,在这个状态下,$el (vue 实例的根元素)已经创建完毕,但还没有加载数据
从结果上看,也可以在 created 状态下监听 scroll 事件
如果对 mounted 和 created 还不够了解,可以参考官方文档·生命周期图示
总结
上面只能算是一个应急之法,而且这种操作 DOM 的方法,并不符合 Vue 的设计理念,待我研究出更合理更高效的办法之后,再发出来分享~
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。
# vue
# 监听滚动事件
# 监听页面滚动
# vuejs
# 锚点
# vue实现横屏滚动公告效果
# vue实现公告栏文字上下滚动效果的示例代码
# Vue中消息横向滚动时setInterval清不掉的问题及解决方法
# vue实现消息的无缝滚动效果的示例代码
# vue实现列表垂直无缝滚动
# vue实现文字滚动效果
# 解决Vue页面固定滚动位置的处理办法
# vue elementUI table表格数据 滚动懒加载的实现方法
# vue监听滚动事件实现滚动监听
# vue实现竖屏滚动公告效果
# 小段
# 递归
# 时长
# 滚动条
# 的是
# 情况下
# 会有
# 看了
# 在这个
# 状态下
# 就能
# 就有
# 两种
# 第二个
# 如果没有
# 两大
# 仅仅是
# 写了
# 我先
# 上看
相关文章:
如何快速搭建FTP站点实现文件共享?
网站制作壁纸教程视频,电脑壁纸网站?
如何在建站之星网店版论坛获取技术支持?
南宁网站建设制作定制,南宁网站建设可以定制吗?
详解jQuery停止动画——stop()方法的使用
详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)
如何选择高性价比服务器搭建个人网站?
详解ASP.NET 生成二维码实例(采用ThoughtWorks.QRCode和QrCode.Net两种方式)
如何零基础在云服务器搭建WordPress站点?
C++ static_cast和dynamic_cast区别_C++静态转换与动态类型安全转换
网站制作服务平台,有什么网站可以发布本地服务信息?
如何配置支付宝与微信支付功能?
网页制作模板网站推荐,网页设计海报之类的素材哪里好?
巅云智能建站系统:可视化拖拽+多端适配+免费模板一键生成
大连网站制作公司哪家好一点,大连买房网站哪个好?
如何使用Golang table-driven基准测试_多组数据测量函数效率
如何快速搭建二级域名独立网站?
建站之星后台管理:高效配置与模板优化提升用户体验
网站企业制作流程,用什么语言做企业网站比较好?
定制建站哪家更专业可靠?推荐榜单揭晓
如何用PHP快速搭建高效网站?分步指南
如何快速使用云服务器搭建个人网站?
网站视频怎么制作,哪个网站可以免费收看好莱坞经典大片?
广德云建站网站建设方案与建站流程优化指南
代刷网站制作软件,别人代刷火车票靠谱吗?
如何用y主机助手快速搭建网站?
专业网站设计制作公司,如何制作一个企业网站,建设网站的基本步骤有哪些?
招贴海报怎么做,什么是海报招贴?
c# 在ASP.NET Core中管理和取消后台任务
C++如何编写函数模板?(泛型编程入门)
已有域名和空间如何快速搭建网站?
高防服务器租用首荐平台,企业级优惠套餐快速部署
网站制作公司,橙子建站是合法的吗?
建站之星如何快速生成多端适配网站?
如何通过西部建站助手安装IIS服务器?
相亲简历制作网站推荐大全,新相亲大会主持人小萍萍资料?
,石家庄四十八中学官网?
建站之星价格显示格式升级,你的预算足够吗?
购物网站制作公司有哪些,哪个购物网站比较好?
网站制作企业,网站的banner和导航栏是指什么?
建站之星备案流程有哪些注意事项?
php8.4新语法match怎么用_php8.4match表达式替代switch【方法】
如何通过宝塔面板实现本地网站访问?
linux top下的 minerd 木马清除方法
为什么Go需要go mod文件_Go go mod文件作用说明
seo网站制作优化,网站SEO优化步骤有哪些?
太原网站制作公司有哪些,网约车营运证查询官网?
北京专业网站制作设计师招聘,北京白云观官方网站?
网站设计制作公司地址,网站建设比较好的公司都有哪些?
建站之星安装路径如何正确选择及配置?
*请认真填写需求信息,我们会在24小时内与您取得联系。