使用js實(shí)現(xiàn)一個(gè)簡(jiǎn)單的滾動(dòng)條過(guò)程解析
當(dāng)我們給元素加上 overflow: auto; 的時(shí)候,就會(huì)出現(xiàn)滾動(dòng)條,然而瀏覽的不同,滾動(dòng)條的樣式大不一樣,有些甚至非常丑。
于是就想著自己寫(xiě)一個(gè)滾動(dòng)條,大概需要弄清楚一下這幾個(gè)點(diǎn):
1、滾動(dòng)條 bar 是根據(jù)內(nèi)容的多少,高度不一樣的,這個(gè)需要?jiǎng)討B(tài)的計(jì)算
2、滾動(dòng)條 bar 的 top 位置 和 內(nèi)容scrollTop 的關(guān)系。
思路:
使用嵌套的布局,如下:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<style type="text/css">
*{
padding: 0;
margin: 0;
}
.wrap{
width: 400px;
height: 400px;
border: 2px solid deeppink;
margin: 0 auto;
overflow: hidden;
position: relative;
}
.box-middle{
height: 100%;
overflow: auto;
width: 200%;
}
.box{
width: 50%;
}
.bar{
background: #000;
width: 10px;
position: absolute;
top: 0;
right: 0;
}
.s1{
height: 400px;
background: pink;
}
.s2{
height: 400px;
background: deeppink;
}
.s3{
height: 400px;
background: deepskyblue;
}
</style>
</head>
<body>
<div class="wrap" id="wrap">
<div class="box-middle" id="boxMidle">
<div class="box" id="content">
<div class="s1">內(nèi)容1</div>
<div class="s2">內(nèi)容2</div>
<div class="s3">內(nèi)容3</div>
</div>
</div>
<div class="bar" id="bar"></div>
</div>
</body>
</html>
wrap 為最外層,給overflow:hidden;。
box-middle 是中間層,也是有滾動(dòng)條的一層,可以寬度給多一點(diǎn),這樣就看不見(jiàn)滾動(dòng)條了。
box就是內(nèi)容層,通過(guò)js,計(jì)算使得 box 的寬度和wrap 保持一致,這樣就完全看不見(jiàn)滾動(dòng)條了
bar 為滾動(dòng)條
寫(xiě)js之前,首先要弄懂一下三個(gè)屬性:
offsetHeight : height + padding + border clientHeight : height + padding scrollHeight : 內(nèi)容的高度(所有子元素高度和) + padding
1、計(jì)算比例:
bar的高度 / wrap的高度 = wrap的高度 / wrap 內(nèi)容部子元素的高度和 ; 此時(shí)忽略 wrap 的padding:0
bar的top / wrap的scrollTop = wrap的高度 / wrap 內(nèi)容部子元素的高度和 ;
需要注意,當(dāng)比例 的 值 小于 1 的時(shí)候,說(shuō)明 這個(gè)時(shí)候沒(méi)有出現(xiàn)滾動(dòng)條。
知道算法之后,寫(xiě)代碼就簡(jiǎn)單很多,普通版代碼如下:
var $wrap = document.getElementById("wrap");
var $boxMidle = document.getElementById("boxMidle");
var $content = document.getElementById("content");
var $bar = document.getElementById("bar");
$content.style.width = $wrap.clientWidth + "px"; //內(nèi)容的寬度
var rate = $boxMidle.clientHeight/ $boxMidle.scrollHeight; //滾動(dòng)條高度的比例,也是滾動(dòng)條top位置的比例
var barHeight = rate * $boxMidle.clientHeight; //滾動(dòng)條的 bar 的高度
if(rate < 1){
//需要出現(xiàn)滾動(dòng)條,并計(jì)算滾動(dòng)條的高度
$bar.style.height = barHeight + "px";
}else{
//不需要出現(xiàn)滾動(dòng)條
$bar.style.display = "none";
}
$boxMidle.onscroll = function(e){
console.log("offsetHeight"+this.offsetHeight); //height + padding + border
console.log("clientHeight"+this.clientHeight); // height + padding
console.log("scrollHeight"+this.scrollHeight); //內(nèi)容的高度(所有子元素高度和) + padding
console.log(this.scrollTop);
$bar.style.top = this.scrollTop*rate + "px";
}
使用面向?qū)ο蟀妫?/p>
function ScrollBar(opt){
var me = this;
me.$wrap = document.getElementById(opt.wrap);
me.$boxMidle = document.getElementById(opt.boxMidle);
me.$content = document.getElementById(opt.content);
me.$bar = document.getElementById(opt.bar);
me.init();
me.$boxMidle.onscroll = function(e){
//console.log("offsetHeight"+this.offsetHeight); //content + padding + border
//console.log("clientHeight"+this.clientHeight); // content + padding
//console.log("scrollHeight"+this.scrollHeight); //內(nèi)容的高度 + padding
console.log(this.scrollTop);
me.scrollToY(this.scrollTop * me.rate)
}
}
ScrollBar.prototype.init = function(){
this.$content.style.width = this.$wrap.clientWidth + "px"; //內(nèi)容的寬度
this.rate = this.$boxMidle.clientHeight/this.$boxMidle.scrollHeight; //滾動(dòng)條高度的比例,也是滾動(dòng)條top位置的比例
this.barHeight = this.rate * this.$boxMidle.clientHeight; //滾動(dòng)條的 bar 的高度
if(this.rate < 1){
//需要出現(xiàn)滾動(dòng)條,并計(jì)算滾動(dòng)條的高度
this.$bar.style.height = this.barHeight + "px";
}else{
//不需要出現(xiàn)滾動(dòng)條
this.$bar.style.display = "none";
}
}
ScrollBar.prototype.scrollToY = function(y){
if(this.rate < 1){
this.$bar.style.top = y + 'px';
}
}
var obj = new ScrollBar({"wrap":"wrap","boxMidle":"boxMidle","content":"content","bar":"bar"});
最后看一下效果:
雖然效果很丑,但是可控,自己調(diào)一下就可以了

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- 原生js實(shí)現(xiàn)自定義滾動(dòng)條組件
- 原生js實(shí)現(xiàn)自定義滾動(dòng)條
- js實(shí)現(xiàn)滾動(dòng)條自動(dòng)滾動(dòng)
- Vue.js桌面端自定義滾動(dòng)條組件之美化滾動(dòng)條VScroll
- JS自定義滾動(dòng)條效果
- JavaScript實(shí)現(xiàn)簡(jiǎn)易聊天對(duì)話框(加滾動(dòng)條)
- js滾輪事件 js自定義滾動(dòng)條的實(shí)現(xiàn)
- JS實(shí)現(xiàn)滾動(dòng)條觸底加載更多
- layer.js open 隱藏滾動(dòng)條的例子
- vue 純js監(jiān)聽(tīng)滾動(dòng)條到底部的實(shí)例講解
- JavaScript 獲取滾動(dòng)條位置并將頁(yè)面滑動(dòng)到錨點(diǎn)
相關(guān)文章
基于JavaScript實(shí)現(xiàn)猜數(shù)字游戲代碼實(shí)例
這篇文章主要介紹了基于JavaScript實(shí)現(xiàn)猜數(shù)字游戲代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07
javascript 進(jìn)度條 實(shí)現(xiàn)代碼
這個(gè)例子是通過(guò)測(cè)試的。是真真正正根據(jù)記錄的條數(shù)掛鉤的。2009-07-07
GitHub上77.9K的Axios項(xiàng)目有哪些值得借鑒的地方詳析
提到axios,相信大家應(yīng)該都不會(huì)陌生,這篇文章主要給大家介紹了關(guān)于GitHub上77.9K的Axios項(xiàng)目有哪些值得借鑒的地方,需要的朋友可以參考下2021-06-06
使用UrlConnection實(shí)現(xiàn)后臺(tái)模擬http請(qǐng)求的簡(jiǎn)單實(shí)例
這篇文章主要介紹了使用UrlConnection實(shí)現(xiàn)后臺(tái)模擬http請(qǐng)求的簡(jiǎn)單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-01-01
H5實(shí)現(xiàn)手機(jī)拍照和選擇上傳功能
這篇文章主要為大家詳細(xì)介紹了H5實(shí)現(xiàn)手機(jī)拍照和選擇上傳功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-12-12
JS實(shí)現(xiàn)在線統(tǒng)計(jì)一個(gè)頁(yè)面內(nèi)鼠標(biāo)點(diǎn)擊次數(shù)的方法
這篇文章主要介紹了JS實(shí)現(xiàn)在線統(tǒng)計(jì)一個(gè)頁(yè)面內(nèi)鼠標(biāo)點(diǎn)擊次數(shù)的方法,實(shí)例分析了javascript操作Cookie實(shí)現(xiàn)計(jì)數(shù)的技巧,需要的朋友可以參考下2015-02-02
禁用Enter鍵表單自動(dòng)提交實(shí)現(xiàn)代碼
這篇文章主要介紹了禁用Enter鍵表單自動(dòng)提交實(shí)現(xiàn)代碼,需要的朋友可以參考下2014-05-05
JavaScript實(shí)現(xiàn)圖片懶加載(Lazyload)
這篇文章主要介紹了JavaScript實(shí)現(xiàn)圖片懶加載(Lazyload)的相關(guān)資料,需要的朋友可以參考下2016-11-11

