一、科普时间
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