hashjavascript
hash 属性是一个可读可写的字符串,该字符串是 URL 的锚部分(从 # 号开始的部分)。
location.hash=anchorname。html
锚点java
<!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
虽然能够直接定位到制定的位置,可是效果不好,没有平缓的过渡效果。github
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.htmldom
最后,附上完整示例源码测试
<!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/aefa75666905url