Vue实现一个返回顶部backToTop组件
发布时间 - 2026-01-11 02:30:28 点击率:次最近在学习VUE。自己就在研究怎么用VUE实现一个组件的封装,今日就算留个笔记

前言
返回顶部这个功能用jq实现,好容易实现,一个animate配合scrollTo就搞定了
今天我们来试试vue封装一个原生js实现的返回顶部;
写起来够呛,借助github,看了别人的gist,稍微封装了下;
当然不是用scrollTo直接调位那种,没有过渡效果怎么说得过去!!还是捣鼓出来了.
废话不多说,看效果图…
效果图
实现思路
- 过渡用的是requestAnimationFrame,这货只支持IE10+,所以必须做兼容
- 滚动视图是window.pageYOffset,这货支持IE9+;
- 为了让可控性更强,图标采用iconfont,具体瞅代码
你能学到什么?
- 学到一些页面计算相关的东东
- 动画API的一些知识
- Vue封装组件相关知识和生命周期和事件监听销毁相关知识的运用
实现功能
- 视图默认在350处显示返回顶部的按钮和图标
- 提示文字和颜色,在图标上下左右的自定义,字段都限制了格式和默认值
- 图标颜色和形状,大小的自定义,字段都限制了格式和默认值
- 过渡动效的自定义,用法:scrollIt(0, 1500, 'easeInOutCubic', callback);
- 返回到视图的point,也就是滚动到哪里
- 过渡时间(ms级别)
- 一堆过渡效果,字符串格式,其实就是滚动的计算函数..
- 当然少不了默认参数了,除了callback
- 兼容性是IE9+,特意开了虚拟机去尝试
代码
scrollIt.js –过渡滚动实现
export function scrollIt(
destination = 0,
duration = 200,
easing = "linear",
callback
) {
// define timing functions -- 过渡动效
let easings = {
// no easing, no acceleration
linear(t) {
return t;
},
// accelerating from zero velocity
easeInQuad(t) {
return t * t;
},
// decelerating to zero velocity
easeOutQuad(t) {
return t * (2 - t);
},
// acceleration until halfway, then deceleration
easeInOutQuad(t) {
return t < 0.5 ? 2 * t * t : -1 + (4 - 2 * t) * t;
},
// accelerating from zero velocity
easeInCubic(t) {
return t * t * t;
},
// decelerating to zero velocity
easeOutCubic(t) {
return --t * t * t + 1;
},
// acceleration until halfway, then deceleration
easeInOutCubic(t) {
return t < 0.5 ? 4 * t * t * t : (t - 1) * (2 * t - 2) * (2 * t - 2) + 1;
},
// accelerating from zero velocity
easeInQuart(t) {
return t * t * t * t;
},
// decelerating to zero velocity
easeOutQuart(t) {
return 1 - --t * t * t * t;
},
// acceleration until halfway, then deceleration
easeInOutQuart(t) {
return t < 0.5 ? 8 * t * t * t * t : 1 - 8 * --t * t * t * t;
},
// accelerating from zero velocity
easeInQuint(t) {
return t * t * t * t * t;
},
// decelerating to zero velocity
easeOutQuint(t) {
return 1 + --t * t * t * t * t;
},
// acceleration until halfway, then deceleration
easeInOutQuint(t) {
return t < 0.5 ? 16 * t * t * t * t * t : 1 + 16 * --t * t * t * t * t;
}
};
// requestAnimationFrame()的兼容性封装:先判断是否原生支持各种带前缀的
//不行的话就采用延时的方案
(function() {
var lastTime = 0;
var vendors = ["ms", "moz", "webkit", "o"];
for (var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
window.requestAnimationFrame =
window[vendors[x] + "RequestAnimationFrame"];
window.cancelAnimationFrame =
window[vendors[x] + "CancelAnimationFrame"] ||
window[vendors[x] + "CancelRequestAnimationFrame"];
}
if (!window.requestAnimationFrame)
window.requestAnimationFrame = function(callback, element) {
var currTime = new Date().getTime();
var timeToCall = Math.max(0, 16 - (currTime - lastTime));
var id = window.setTimeout(function() {
callback(currTime + timeToCall);
}, timeToCall);
lastTime = currTime + timeToCall;
return id;
};
if (!window.cancelAnimationFrame)
window.cancelAnimationFrame = function(id) {
clearTimeout(id);
};
})();
function checkElement() {
// chrome,safari及一些浏览器对于documentElemnt的计算标准化,reset的作用
document.documentElement.scrollTop += 1;
let elm =
document.documentElement.scrollTop !== 0
? document.documentElement
: document.body;
document.documentElement.scrollTop -= 1;
return elm;
}
let element = checkElement();
let start = element.scrollTop; // 当前滚动距离
let startTime = Date.now(); // 当前时间
function scroll() { // 滚动的实现
let now = Date.now();
let time = Math.min(1, (now - startTime) / duration);
let timeFunction = easings[easing](time);
element.scrollTop = timeFunction * (destination - start) + start;
if (element.scrollTop === destination) {
callback; // 此次执行回调函数
return;
}
window.requestAnimationFrame(scroll);
}
scroll();
}
backToTop.vue
<template>
<div class="back-to-top" @click="backToTop" v-show="showReturnToTop" @mouseenter="show" @mouseleave="hide">
<i :class="[bttOption.iClass]" :style="{color:bttOption.iColor,'font-size':bttOption.iFontsize}"></i>
<span class="tips" :class="[bttOption.iPos]" :style="{color:bttOption.textColor}" v-show="showTooltips">{{bttOption.text}}</span>
</div>
</template>
<script>
import { scrollIt } from './scrollIt'; // 引入动画过渡的实现
export default {
name: 'back-to-top',
props: {
text: { // 文本提示
type: String,
default: '返回顶部'
},
textColor: { // 文本颜色
type: String,
default: '#f00'
},
iPos: { // 文本位置
type: String,
default: 'right'
},
iClass: { // 图标形状
type: String,
default: 'fzicon fz-ad-fanhuidingbu1'
},
iColor: { // 图标颜色
type: String,
default: '#f00'
},
iFontsize: { // 图标大小
type: String,
default: '32px'
},
pageY: { // 默认在哪个视图显示返回按钮
type: Number,
default: 400
},
transitionName: { // 过渡动画名称
type: String,
default: 'linear'
}
},
data: function () {
return {
showTooltips: false,
showReturnToTop: false
}
},
computed: {
bttOption () {
return {
text: this.text,
textColor: this.textColor,
iPos: this.iPos,
iClass: this.iClass,
iColor: this.iColor,
iFontsize: this.iFontsize
}
}
},
methods: {
show () { // 显示隐藏提示文字
return this.showTooltips = true;
},
hide () {
return this.showTooltips = false;
},
currentPageYOffset () {
// 判断滚动区域大于多少的时候显示返回顶部的按钮
window.pageYOffset > this.pageY ? this.showReturnToTop = true : this.showReturnToTop = false;
},
backToTop () {
scrollIt(0, 1500, this.transitionName, this.currentPageYOffset);
}
},
created () {
window.addEventListener('scroll', this.currentPageYOffset);
},
beforeDestroy () {
window.removeEventListener('scroll', this.currentPageYOffset)
}
}
</script>
<style scoped lang="scss">
.back-to-top {
position: fixed;
bottom: 5%;
right: 100px;
z-index: 9999;
cursor: pointer;
width: auto;
i {
font-size: 32px;
display: inline-block;
position: relative;
text-align: center;
padding: 5px;
background-color: rgba(234, 231, 231, 0.52);
border-radius: 5px;
transition: all 0.3s linear;
&:hover {
border-radius: 50%;
background: #222;
color: #fff !important;
}
}
.tips {
display: inline-block;
position: absolute;
word-break: normal;
white-space: nowrap;
width: auto;
font-size: 12px;
color: #fff;
z-index: -1;
}
.left {
right: 0;
top: 50%;
margin-right: 50px;
transform: translateY(-50%);
}
.right {
left: 0;
top: 50%;
margin-left: 50px;
transform: translateY(-50%);
}
.bottom {
bottom: 0;
margin-top: 50px;
}
.top {
top: 0;
margin-bottom: 50px;
}
}
</style>
总结
从心血来潮到折腾出来,为了兼顾兼容性和拓展性,好像几个小时了.
不过实现了.你再搬到其他语言,类似ng4,也就是十来分钟的事情,
思路会了,实现更多的是写法而已,至于性能优化,可以一边写一边考虑,也可以实现后有空再优化.
希望对大家的学习有所帮助,也希望大家多多支持。
# Vue
# 返回顶部
# vue实现返回顶部
# vue返回顶部组件
# Vue实现返回顶部按钮实例代码
# vue实现移动端返回顶部
# vue进入页面时不在顶部
# 检测滚动返回顶部按钮问题及解决方法
# vue项目tween方法实现返回顶部的示例代码
# vue开发公共组件之返回顶部
# 自定义
# 的是
# 相关知识
# 默认值
# 就在
# 看了
# 说得过去
# 开了
# 你能
# 上下左右
# 可以实现
# 你再
# 多说
# 更强
# 装了
# 搬到
# 心血来潮
# 十来
# 几个小时
# 回调
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Laravel如何设置定时任务(Cron Job)_Laravel调度器与任务计划配置
如何在服务器上三步完成建站并提升流量?
Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】
Laravel定时任务怎么设置_Laravel Crontab调度器配置
php json中文编码为null的解决办法
图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?
php嵌入式断网后怎么恢复_php检测网络重连并恢复硬件控制【操作】
中国移动官方网站首页入口 中国移动官网网页登录
Laravel如何设置自定义的日志文件名_Laravel根据日期或用户ID生成动态日志【技巧】
深圳网站制作培训,深圳哪些招聘网站比较好?
JS经典正则表达式笔试题汇总
uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址
香港服务器建站指南:免备案优势与SEO优化技巧全解析
Edge浏览器提示“由你的组织管理”怎么解决_去除浏览器托管提示【修复】
Laravel如何发送系统通知_Laravel Notifications实现多渠道消息通知
Laravel如何实现API速率限制?(Rate Limiting教程)
Laravel如何监控和管理失败的队列任务_Laravel失败任务处理与监控
无锡营销型网站制作公司,无锡网选车牌流程?
七夕网站制作视频,七夕大促活动怎么报名?
javascript中的try catch异常捕获机制用法分析
用v-html解决Vue.js渲染中html标签不被解析的问题
php增删改查怎么学_零基础入门php数据库操作必知基础【教程】
南京网站制作费用,南京远驱官方网站?
LinuxCD持续部署教程_自动发布与回滚机制
如何快速登录WAP自助建站平台?
Laravel如何升级到最新版本?(升级指南和步骤)
Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】
java中使用zxing批量生成二维码立牌
Laravel Admin后台管理框架推荐_Laravel快速开发后台工具
网站图片在线制作软件,怎么在图片上做链接?
千库网官网入口推荐 千库网设计创意平台入口
成都品牌网站制作公司,成都营业执照年报网上怎么办理?
HTML透明颜色代码怎么让下拉菜单透明_下拉菜单透明背景指南【技巧】
Laravel如何获取当前用户信息_Laravel Auth门面获取用户ID
JS中对数组元素进行增删改移的方法总结
大型企业网站制作流程,做网站需要注册公司吗?
如何在IIS中新建站点并配置端口与IP地址?
使用PHP下载CSS文件中的所有图片【几行代码即可实现】
Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】
Win11怎么更改系统语言为中文_Windows11安装语言包并设为显示语言
Laravel策略(Policy)如何控制权限_Laravel Gates与Policies实现用户授权
原生JS实现图片轮播切换效果
实例解析Array和String方法
Laravel如何使用Sanctum进行API认证?(SPA实战)
如何在云主机上快速搭建多站点网站?
PHP 500报错的快速解决方法
详解jQuery停止动画——stop()方法的使用
网页设计与网站制作内容,怎样注册网站?
合肥制作网站的公司有哪些,合肥聚美网络科技有限公司介绍?
香港服务器选型指南:免备案配置与高效建站方案解析

