前端实现文档内部导航效果

1. 前言

  • 实现文档内部导航效果,功能包括:
    • 滚动:用户滚动到相应的标题处,导航对应的标题元素加样式,并且导航框的内部滚动轴也会跟随滚动到合适的位置。
    • 点击:点击导航对应的标题元素或者置顶按钮,文档则跳到相应的位置并添加样式,并且导航框的内部滚动轴也会跟随滚动到合适的位置。
  • 代码由html、css和javascript组成;代码中的文本是由lorem(乱数假文)生成的
  • 每个代码块有注释解释,如果由不清楚的欢迎留言

2. html代码

  • 由于文档内容太长就没有放,生成文档结构命令为section[class="doc"]>(h2{navigator:$}+p*20>lorem5)*20
html 复制代码
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
    <link rel="stylesheet" href="./index.css" />
  </head>
  <body>
    <div class="content">
      <!-- 文档内容使用 section[class="doc"]>(h2{navigator:$}+p*20>lorem5)*20  生成-->
      
      <!-- 导航 -->
      <div class="doc-markup">
        <div class="doc-markup-header">
          <span>目录</span>
        </div>
        <div class="doc-markup-body"></div>
      </div>
    </div>
    <!-- 回到顶部 -->
    <div class="broadsideImg">
      <button class="gotop">
        <svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg" class="">
          <path
            fill-rule="evenodd"
            clip-rule="evenodd"
            d="M5.00806 3.0625C4.70158 3.0625 4.45312 3.31095 4.45312 3.61743C4.45312 3.92391 4.70158 4.17236 5.00806 4.17236H14.9968C15.3033 4.17236 15.5518 3.92391 15.5518 3.61743C15.5518 3.31095 15.3033 3.0625 14.9968 3.0625H5.00806ZM10.3948 5.9997C10.1781 5.78297 9.82676 5.78297 9.61005 5.9997L6.07849 9.53126C5.86176 9.74797 5.86176 10.0993 6.07849 10.316C6.2952 10.5328 6.64656 10.5328 6.86327 10.316L9.44751 7.73182V17.3809C9.44751 17.6873 9.69596 17.9358 10.0024 17.9358C10.3089 17.9358 10.5574 17.6873 10.5574 17.3809V7.73182L13.1416 10.316C13.3583 10.5328 13.7097 10.5328 13.9264 10.316C14.1431 10.0993 14.1431 9.74797 13.9264 9.53126L10.3948 5.9997Z"
            fill="#515767"
            stroke="#515767"
            stroke-width="0.5"
          ></path>
        </svg>
      </button>
    </div>
  </body>

  <script src="./index.js"></script>
</html>

3. js代码

1. 创建导航元素

js 复制代码
// 创建导航元素
function creatDocumentNavigation() {
  if (doms.length === 0) {
    docMarkup.style.display = "none"
    return false
  }
  docMarkup.style.display = "block"
  let ul = document.createElement("ul")
  ul.setAttribute("class", "doc-markup_ul")
  for (let i = 0; i < doms.length; i++) {
    // 创建li标签,设置内容
    const bookmarkContent = doms[i].textContent
    let li = document.createElement("li")
    li.textContent = bookmarkContent
    li.setAttribute("class", "doc-markup_li")
    li.setAttribute("title", bookmarkContent)
    //添加点击事件
    li.addEventListener("click", e => {
      e.preventDefault()
      document.documentElement.scrollTop = Math.floor(doms[i].offsetTop)
      isClickToJump = true
    })
    ul.appendChild(li)
  }
  docMarkupBody.appendChild(ul)
  // 监听文档滚动事件
  watchDocumentScroll()
  // 监听回到顶部事件
  watchBroadsideImgGotopClick()
}

2. 监听滚动事件

js 复制代码
function watchDocumentScroll() {
  const markupLi = document.querySelectorAll(".doc-markup-body .doc-markup_ul .doc-markup_li")
  let lastScrollTop = 0
  window.addEventListener("scroll", () => {
    let selectiIndex = null
    const currentScrollTop = document.documentElement.scrollTop
    // 这里是是对向上还是向下滚动做处理;大于零向下,反之向上
    if (currentScrollTop - lastScrollTop > 0 || isClickToJump) {
      isClickToJump = false
      doms.forEach((item, index) => {
        let reat = item.getBoundingClientRect()
        // 这里的判断是找到需要高亮的导航目录元素,当标题dom距离顶部小于50那么此标题dom就是高亮标题(reat.bottom指的是元素底部距离顶部的距离);向上滚动就请coder脑补一下子把
        if (reat.top < 50 && reat.bottom > 0) {
          //清除导航样式函数
          clearMarkupSelectClass()
          selectiIndex = index
        }
      })
    } else {
      doms.forEach((item, index) => {
        let reat = item.getBoundingClientRect()
        if (reat.top > 50 && reat.bottom < 150) {
          //清除导航样式函数
          clearMarkupSelectClass()
          index === 0 ? (selectiIndex = index) : (selectiIndex = index - 1)
        }
      })
    }
    if (selectiIndex !== null) {
      markupLi[selectiIndex].classList.add("markup-select")
      //大于4的原因是:我将导航高度设置了300,每个标题30,因此导航最高展示10个元素所以当下标大于4的即5个元素时开始触发导航目录的滚动轴
      if(selectiIndex >= 4){
        docMarkupBody.scrollTo({
          top: Math.floor(markupLi[selectiIndex-4].offsetTop) - 55,
          behavior: "smooth",
        })
      }else{
        docMarkupBody.scrollTo({
          top: 0,
          behavior: "smooth",
        })
      }
    }

    lastScrollTop = currentScrollTop
  })
}

3. 回到顶部按钮

js 复制代码
function watchBroadsideImgGotopClick() {
  const broadsideImgGotop = document.querySelector(".broadsideImg .gotop")
  broadsideImgGotop.addEventListener("click", (e) => {
    e.preventDefault()
    isClickToJump = true
    document.documentElement.scrollTop = 0
  })
}

4. 清除导航样式函数

js 复制代码
// 清除标记的样式
function clearMarkupSelectClass() {
  const markupLi = document.querySelectorAll(".doc-markup-body .doc-markup_ul .doc-markup_li")
  markupLi.forEach(item => {
    item.classList.remove("markup-select")
  })
}

4. 效果静态展示

5. 写在最后

  • 代码的简洁程度不够高,请大家不要介意;另外滚动函数可以加防抖来提交性能
  • 如果对代码有疑惑的地方欢迎留言。
相关推荐
Python大数据分析@几秒前
通俗的讲,网络爬虫到底是什么?
前端·爬虫·网络爬虫
不爱学英文的码字机器18 分钟前
[操作系统] 环境变量详解
开发语言·javascript·ecmascript
Lysun00122 分钟前
vue2的$el.querySelector在vue3中怎么写
前端·javascript·vue.js
jerry-8941 分钟前
Centos类型服务器等保测评整/etc/pam.d/system-auth
java·前端·github
工业甲酰苯胺43 分钟前
深入解析 Spring AI 系列:解析返回参数处理
javascript·windows·spring
小爬菜1 小时前
Django学习笔记(启动项目)-03
前端·笔记·python·学习·django
想要打 Acm 的小周同学呀1 小时前
前端Vue2项目使用md编辑器
前端·编辑器·vue2·markdown 语法
计算机-秋大田1 小时前
基于SSM的家庭记账本小程序设计与实现(LW+源码+讲解)
java·前端·后端·微信小程序·小程序·课程设计
海的预约2 小时前
VUE之路由Props、replace、编程式路由导航、重定向
前端·vue.js·智能路由器
西柚与蓝莓3 小时前
报错:{‘csrf_token‘: [‘The CSRF token is missing.‘]}
前端·flask