一、科普時間
hash
hash 屬性是一個可讀可寫的字符串,該字符串是 URL 的錨部分(從 # 號開始的部分)。
location.hash=anchorname。
錨點
二、錨點簡單的栗子
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>偽錨點</title>
<style>
.anchor1, .anchor2{width:100px;height:100px;margin-top:2000px;margin-bottom:2000px;}
.anchor1{background:red;}
.anchor2{background:green;}
</style>
</head>
<body>
<p>
<a href="#anchor1">錨點1</a>
</p>
<p>
<a href="#anchor2">錨點2</a>
</p>
<div id="anchor1" class="anchor1">
錨點1
</div>
<div id="anchor2" class="anchor2">
錨點2
</div>
</body>
</html>
解析
訪問該頁面的地址:http://127.0.0.1/anchor.html(我是在本地服務器上測試的)
點擊a鏈接錨點1,則頁面會直接跳到紅色的div(錨點1),同時,瀏覽器地址改變為http://127.0.0.1/anchor.html#anchor1
雖然可以直接定位到制定的位置,但是效果很差,沒有平緩的過渡效果。
三、改進過渡效果
1)前期理論准備
既然hash值是對應錨點的id值,那如果改為js動態獲取hash值,然后再根據hash值獲得dom對象。最后,用js進行平緩過渡。
基於這個思路,就必須要求hash的取名有獨特性,不能跟頁面中的任何一個id一致,否則會觸發瀏覽器默認的錨點定位行為。
2)確定特殊hash命名
hash命名直接取特殊的前綴:w_,比如錨點1對應的hash值為w_anchor1
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>偽錨點</title>
<style>
.anchor1, .anchor2{width:100px;height:100px;}
.anchor1{background:red;}
.anchor2{background:green;}
.spacing1, .spacing2{height:200px;}
.spacing1{background:yellow;}
.spacing2{background:gray;}
</style>
</head>
<body>
<p>
<a href="#w_anchor1">錨點1</a>
</p>
<p>
<a href="#w_anchor2">錨點2</a>
</p>
<p class="spacing1">間隔1</p>
<p class="spacing2">間隔2</p>
<p class="spacing1">間隔3</p>
<p class="spacing2">間隔4</p>
<p class="spacing1">間隔5</p>
<p class="spacing2">間隔6</p>
<p class="spacing1">間隔7</p>
<p class="spacing2">間隔8</p>
<div id="anchor1" class="anchor1">錨點1</div>
<p class="spacing1">間隔1</p>
<p class="spacing2">間隔2</p>
<p class="spacing1">間隔3</p>
<p class="spacing2">間隔4</p>
<p class="spacing1">間隔5</p>
<p class="spacing2">間隔6</p>
<p class="spacing1">間隔7</p>
<p class="spacing2">間隔8</p>
<div id="anchor2" class="anchor2">錨點2</div>
<p class="spacing1">間隔1</p>
<p class="spacing2">間隔2</p>
<p class="spacing1">間隔3</p>
<p class="spacing2">間隔4</p>
<p class="spacing1">間隔5</p>
<p class="spacing2">間隔6</p>
<p class="spacing1">間隔7</p>
<p class="spacing2">間隔8</p>
</body>
</html>
3)編寫讀取特殊hasn值的方法以及緩動方法(本示例不考慮兼容性)
(function(window, undefined){
// 監聽頁面加載完成后,檢查是否需要定位錨點
window.onload = function(){
scrollToAnchor();
};
// 監聽地址欄url的hash值改變時,檢查是否需要定位錨點
window.onhashchange = function(){
scrollToAnchor();
};
// 滾動到自定義的偽錨點
function scrollToAnchor(){
var hash = getHash(), // 獲取url的hash值
anchor = getAnchor(hash), // 獲取偽錨點的id
anchorDom, // 偽錨點dom對象
anchorScrollTop; // 偽錨點距離頁面頂部的距離
// 如果不存在偽錨點,則直接結束
if(anchor.length < 1){
return;
}
anchorDom = getDom(anchor);
anchorScrollTop = anchorDom.offsetTop;
animationToAnchor(document.body.scrollTop, anchorScrollTop);
}
/*
@function 滾動到指定位置方法
@param startNum {int} -- 開始位置
@param stopNum {int} -- 結束位置
*/
function animationToAnchor(startNum, stopNum){
var nowNum = startNum + 10; // 步進為10
if(nowNum > stopNum){
nowNum = stopNum;
}
// 緩動方法
window.requestAnimationFrame(function(){
document.body.scrollTop = nowNum; // 當前示例頁面,滾動條在body,所以滾動body
// 滾動到預定位置則結束
if(nowNum == stopNum){
return;
}
animationToAnchor(nowNum, stopNum); // 只要還符合緩動條件,則遞歸調用
});
}
// 獲取錨點id
function getAnchor(str){
return checkAnchor(str) ? str.split("w_")[1] : "";
}
// 判斷是否為特殊的hash值,也即是否為偽錨點
function checkAnchor(str){
return str.indexOf("w_") == 0 ? true : false;
}
// 獲取hash值
function getHash(){
return window.location.hash.substring(1);
}
// 獲取dom對象
function getDom(id){
return document.getElementById(id);
}
})(window);
在線演示:https://wall-wxk.github.io/blogDemo/anchor/anchor.html
最后,附上完整示例源碼
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>偽錨點</title>
<style>
.anchor1, .anchor2{width:100px;height:100px;}
.anchor1{background:red;}
.anchor2{background:green;}
.spacing1, .spacing2{height:200px;}
.spacing1{background:yellow;}
.spacing2{background:gray;}
</style>
</head>
<body>
<p>
<a href="#w_anchor1">錨點1</a>
</p>
<p>
<a href="#w_anchor2">錨點2</a>
</p>
<p class="spacing1">間隔1</p>
<p class="spacing2">間隔2</p>
<p class="spacing1">間隔3</p>
<p class="spacing2">間隔4</p>
<p class="spacing1">間隔5</p>
<p class="spacing2">間隔6</p>
<p class="spacing1">間隔7</p>
<p class="spacing2">間隔8</p>
<div id="anchor1" class="anchor1">錨點1</div>
<p class="spacing1">間隔1</p>
<p class="spacing2">間隔2</p>
<p class="spacing1">間隔3</p>
<p class="spacing2">間隔4</p>
<p class="spacing1">間隔5</p>
<p class="spacing2">間隔6</p>
<p class="spacing1">間隔7</p>
<p class="spacing2">間隔8</p>
<div id="anchor2" class="anchor2">錨點2</div>
<p class="spacing1">間隔1</p>
<p class="spacing2">間隔2</p>
<p class="spacing1">間隔3</p>
<p class="spacing2">間隔4</p>
<p class="spacing1">間隔5</p>
<p class="spacing2">間隔6</p>
<p class="spacing1">間隔7</p>
<p class="spacing2">間隔8</p>
<script>
(function(window, undefined){
// 監聽頁面加載完成后,檢查是否需要定位錨點
window.onload = function(){
scrollToAnchor();
};
// 監聽地址欄url的hash值改變時,檢查是否需要定位錨點
window.onhashchange = function(){
scrollToAnchor();
};
// 滾動到自定義的偽錨點
function scrollToAnchor(){
var hash = getHash(), // 獲取url的hash值
anchor = getAnchor(hash), // 獲取偽錨點的id
anchorDom, // 偽錨點dom對象
anchorScrollTop; // 偽錨點距離頁面頂部的距離
// 如果不存在偽錨點,則直接結束
if(anchor.length < 1){
return;
}
anchorDom = getDom(anchor);
anchorScrollTop = anchorDom.offsetTop;
animationToAnchor(document.body.scrollTop, anchorScrollTop);
}
/*
@function 滾動到指定位置方法
@param startNum {int} -- 開始位置
@param stopNum {int} -- 結束位置
*/
function animationToAnchor(startNum, stopNum){
var nowNum = startNum + 10; // 步進為10
if(nowNum > stopNum){
nowNum = stopNum;
}
// 緩動方法
window.requestAnimationFrame(function(){
document.body.scrollTop = nowNum; // 當前示例頁面,滾動條在body,所以滾動body
// 滾動到預定位置則結束
if(nowNum == stopNum){
return;
}
animationToAnchor(nowNum, stopNum); // 只要還符合緩動條件,則遞歸調用
});
}
// 獲取錨點id
function getAnchor(str){
return checkAnchor(str) ? str.split("w_")[1] : "";
}
// 判斷是否為特殊的hash值,也即是否為偽錨點
function checkAnchor(str){
return str.indexOf("w_") == 0 ? true : false;
}
// 獲取hash值
function getHash(){
return window.location.hash.substring(1);
}
// 獲取dom對象
function getDom(id){
return document.getElementById(id);
}
})(window);
</script>
</body>
</html>
閱讀原文:www.jianshu.com/p/aefa75666905
