06 antdesign react Anchor 不同页面之间实现锚点

news2024/11/14 21:13:41

react Anchor 不同页面之间实现锚点

  • 一、定义
  • 二、使用步骤
  • 三、开发流程
    • (一)、组件
    • (二)、页面布局
    • (三)、点击事件
    • (四)、总结说明
      • 一、react单页面应用,当前页面的锚点
      • 二、react单页面应用,不同页面的锚点
        • 思路:锚点只能在当前页面使用,所以用useEffect()拦截
      • 三、总结
  • 五、其它
    • 解决办法:加上location.hash可以解决

一、定义

Anchor锚点是:用于跳转到页面指定位置。

何时使用
需要展现当前页面上可供跳转的锚点链接,以及快速在锚点之间跳转。

二、使用步骤

1、首先在react项目中引用antd的锚点

import {Anchor} from 'antd';

const { Link } = Anchor;

2、发现页面进行跳转而不是点位到页面的锚点,发现url有所改变
浏览器支持的锚点必须是通过hash来实现的,

三、开发流程

(一)、组件

在这里插入图片描述

(二)、页面布局

onClick()点击事件

<Anchor onClick={onAnchorChange}>
  <div className={styles.digUlItem}>
    {item.des.map((littleItem, littleIndex) => {
      return (
        <Link
          key={littleItem}
          href={`#${littleItem}`}
          title={littleItem}
        ></Link>
      );
    })}
  </div>
</Anchor>

(三)、点击事件

 const onAnchorChange = (e,link) => {
    e.preventDefault();
    console.log('ddd e anchor>>>', e);
    console.log('ddd hash anchor>>>', link);
  };

(四)、总结说明

一、react单页面应用,当前页面的锚点

简单使用

  • antd中Anchor组件阻止默认路由跳转
  • 使用Anchor组件时,添加click方法:
    1、阻止默认事件 e.preventDefault();阻止link的href跳转路由事件
    2、使用H5的scrollToAnchor添加页面滚动效果:
    当前页面实现锚点,的点击事件,代码如下:
    链接: react+antd中Anchor锚点踩坑===当前页面跳转,阻止点击事件的默认事件
handleClickFun= (e, link) => {
    e.preventDefault();
    if (link.href) {
        // 找到锚点对应得的节点
        let element = document.getElementById(link.href);
        // 如果对应id的锚点存在,就跳滚动到锚点顶部
        element && element .scrollIntoView({block:'start', behavior:'smooth'});
    }
};
// block:表示垂直方向的滚动对齐方式,“start”, “center”, “end”, 或 “nearest”
// behavior:表示动画效果,auto/smooth(滚动效果) 

使用加强版

  • 点击事件阻止a标签的默认
    e.preventDefault();
  • 原理:

【antdesign中anchor点击锚点后手动刷新页面,显示空白】是因为antdesign anchor底层代码是使用a标签来进行锚点跳转的,所以我们需要阻止a标签的默认行为。
antd Anchor底层是使用的a标签实现锚点功能的,而a标签的默认行为就是路由跳转(跳转到href),只要阻止a标签的默认行为就可以了。

  • 控制台打印效果,比较如下:

  • 阻止默认事件前:
    在这里插入图片描述

  • 阻止默认事件后:
    在这里插入图片描述

二、react单页面应用,不同页面的锚点

思路:锚点只能在当前页面使用,所以用useEffect()拦截

  • 详细思路:我在项目中使用的锚点并不是真正意义上的锚点,只是利用了点击a便签会在浏览器的地址栏url出现#后的参数,可以理解为vue中的路由传参query。
    通过react的函数组件的钩子函数useEffect()。通过useEffect(handleFun,[第二个参数])的第二个参数对浏览器的地址栏进行拦截。我拦截的是:location.hash 对#号后的部分进行拦截。这里而外说一句:location对象,他是window对象和document对象的属性,它表示载入窗口的url,它可以解析url。
  • 代码如下:
    在这里插入图片描述
    在这里插入图片描述
 useEffect(() => {
    let myTimer
    clearTimeout(myTimer);
    if (!!location.hash) {
      const scrollToAnchor = (anchorName) => {
        if (anchorName) {
          let anchorElement = document.getElementById(anchorName);
          console.log('element>>>', anchorElement)

          if (anchorElement !== null) {
            let scrollHeight = anchorElement.offsetTop - 65;
            if (!!scrollHeight) {
              myTimer = setTimeout(() => {
                window.scrollTo({
                  left: 0,  //x轴
                  top: scrollHeight,  //y轴
                  behavior: 'smooth'
                })
              }, 500);
              //useEffect()中清除定时器,return出去就可以了。
              return () => clearTimeout(myTimer)
            } else {
              return () => clearTimeout(myTimer)
            }
          }
        }
      }
      scrollToAnchor(location.hash)
    }
  }, [location.hash])
useEffect(() => {
    console.log('location.hash>>>', location.hash);
    if (!!location.hash) {
      const scrollToAnchor = (anchorName) => {
        console.log('anchorName>>>', anchorName)
        if (anchorName) {
          let anchorElement = document.getElementById(anchorName);
          console.log('element>>>', anchorElement)
          if (anchorElement !== null) {
            let scrollHeight = anchorElement.offsetTop - 65;
            console.log('anchorElement.offsetTop>>>', anchorElement.offsetTop)
            console.log('scrollHeight>>>', scrollHeight)
            let myTimer
            if (!!scrollHeight) {
              myTimer = setTimeout(() => {
                console.log('定时器>>>');
                window.scrollTo({
                  left: 0,  //x轴
                  top: scrollHeight,  //y轴
                  behavior: 'smooth'
                })
              }, 500);
              //useEffect()中清除定时器,return出去就可以了。
              return () => clearTimeout(myTimer)
            } else {
              //useEffect()中清除定时器,return出去就可以了。
              return () => clearTimeout(myTimer)
            }
          }
        }
      }
      scrollToAnchor(location.hash)
    }
  }, [location.hash])
  • 完整代码:
import React,{useEffect} from 'react';
import styles from './index.less';
import { Image, Space } from 'antd';
import FarmServerList from '@/components/productServer/FarmServerList';
import farm1 from '@/assets/productServer/farm1.png';
export default function index({ content = [] }) {
  useEffect(() => {
    console.log('location.hash>>>', location.hash);
    if (!!location.hash) {
      const scrollToAnchor = (anchorName) => {
        console.log('anchorName>>>', anchorName)
        if (anchorName) {
          let anchorElement = document.getElementById(anchorName);
          console.log('element>>>', anchorElement)
          if (anchorElement !== null) {
            let scrollHeight = anchorElement.offsetTop - 65;
            console.log('anchorElement.offsetTop>>>', anchorElement.offsetTop)
            console.log('scrollHeight>>>', scrollHeight)
            let myTimer
            if (!!scrollHeight) {
              myTimer = setTimeout(() => {
                console.log('定时器>>>');
                window.scrollTo({
                  left: 0,  //x轴
                  top: scrollHeight,  //y轴
                  behavior: 'smooth'
                })
              }, 500);
              //useEffect()中清除定时器,return出去就可以了。
              return () => clearTimeout(myTimer)
            } else {
              //useEffect()中清除定时器,return出去就可以了。
              return () => clearTimeout(myTimer)
            }
          }
        }
      }
      scrollToAnchor(location.hash)
    }
  }, [location.hash])
  return (
    <div className={styles.home_box}>
      {content.map((item,index) => {
        if (item.id % 2 == 0) {
          return (
            <div className={styles.content} key={item.id}>
              <div className={styles.left}>
                <Space direction={'vertical'} size={'large'}>
                  <span className={styles.span1} id={`#${index}`}>
                    {item.title}
                  </span>
                  <FarmServerList list={item.list} align={item.id % 2} />
                </Space>
              </div>
              <div className={styles.right}>
                <Image src={item.img} preview={{ mask: false }} />
              </div>
            </div>
          );
        } else {
          return (
            <div className={styles.content} key={item.id}>
              <div className={styles.left}>
                <Image src={item.img} preview={{ mask: false }} />
              </div>
              <div className={styles.right}>
                <Space direction={'vertical'} size={'large'}>
                  <span className={styles.span} id={`#${index}`}>{item.title}</span>
                  <FarmServerList list={item.list} />
                </Space>
              </div>
            </div>
          );
        }
      })}
    </div>
  );
}

三、总结

实际效果:能够实现从一个页面跳转到另外一个页面,并且根据浏览器地址栏location.hash获取到#号后的内容,从一个页面的锚点->点击->跳转到另一个页面->根据getElementById获取真实节点->windowm.scrollTo页面滚动
BUG点:
现象:点击第一次跳转不能滚动,重新第二次跳转能够实现滚动。
原因分析:react框架有虚拟Dom,在useEffect不能获取到真实的Dom节点,因为页面组件的useEffect会在浏览器加载一个页面渲染的时候调用useEffect()两次,第一次拿不到,第二次能拿到节点。
但我们想要第一节进入页面就实现滚动
所以:考虑使用react的函数组件的另一个狗子函数useRef()
链接: useEffect执行时机
链接: useRef()操作真实dom节点

五、其它

解决办法:加上location.hash可以解决

浏览器支持的锚点必须是通过hash来实现的,
不行的话就不能用锚点的形式,只能自己写个组件,注册点击事件,然后获取要滚动到的元素的位置,设置window的scrollTop
React不引入antd如何实现锚点跳转

代码如下:

scrollToAnchor (id){

document.getElementById(id).scrollIntoView(false);

}

render:

  • this.scrollToAnchor(‘Summarize’)}>Summarize

  • this.scrollToAnchor(‘ProductFunction’)}>ProductFunction

  • this.scrollToAnchor(‘ToHelpAnswer’)}>ToHelpAnswer

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/351030.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

central cache设计及实现

你好&#xff0c;我是安然无虞。 central cache 设计及实现 central cache 也是一个哈希桶结构, 而且它的哈希桶的映射关系跟 thread cache 是一样的, 不同的是 central cache 每个哈希桶位置上挂的是 SpanList 双向链表结构, 而且每个哈希桶下面的 span 中的大块内存被按映射…

大数据之数据中台

目录数据仓库传统数据仓库无法支撑互联网时代的商业智能数据埋点数据仓库 数据仓库是在企业管理和决策中面向主题的、集成的、与时间相关的、不可修改的数据集合。 在电商场景中&#xff0c;有一个数据库专门存放订单的数据&#xff0c;另外一个数据库存放会员相关的数据。构建…

订单服务:订单流程

订单流程 订单流程是指从订单产生到完成整个流转的过程&#xff0c;从而行程了一套标准流程规则。而不同的产品类型或业务类型在系统中的流程会千差万别&#xff0c;比如上面提到的线上实物订单和虚拟订单的流程&#xff0c;线上实物订单与 O2O 订单等&#xff0c;所以需要根据…

[Datawhale][CS224W]图机器学习(三)

目录一、简介与准备二、教程2.1 下载安装2.2 创建图2.2.1 常用图创建&#xff08;自定义图创建&#xff09;1.创建图对象2.添加图节点3.创建连接2.2.2 经典图结构1.全连接无向图2.全连接有向图3.环状图4.梯状图5.线性串珠图6.星状图7.轮辐图8.二项树2.2.3 栅格图1.二维矩形栅格…

如何修改 类名::静态变量?

好久没更新了&#xff0c;其中经历了备战省赛&#xff0c;重装电脑&#xff0c;换服务器&#xff0c;重新搭建博客等一系列事&#xff0c;受到许多技术大牛的指点&#xff0c;而且新人太厉害了&#xff0c;卷卷卷&#xff01; Geek Challenge 2021有一道BabyPOP&#xff0c;这…

vue写一个翻页的时间插件

读秒效果有一个从上向下的翻页效果 效果图 clock代码 <template><div class"dateClock"><div class"todayClass"><p class"datep">{{dateToday}}</p><span style"float:right;font-size:16px">…

基于 VPX 总线的工件台运动控制系统研究与开发-DSP+FPGA硬件架构(一)

作为光刻机核心单元之一&#xff0c;超精密工件台主要负责实现快速扫描、上下片、精密定位、调平调焦等功能。目前&#xff0c;较为成熟的方案大多采用 VME 并行总线架构来建立超精密工件台控制系统&#xff0c;由于随着系统性能要求的提升&#xff0c;VME 总线以及相应的处理器…

最大连续子列和

给定一个数组&#xff0c;求它的最大连续子列和。这个问题有四种解法。 1、暴力循环(O(n^3))分析这个问题&#xff0c;既然是子列&#xff0c;那么它最长为n&#xff0c;最短为1。要想求和我们一般需要知道这个子列的左端下标和右端下标&#xff0c;再求这个子列的和。最简单的…

深入浅出带你学习Nginx中间件常见漏洞

前言 上篇文章总结了APACHE中间件的常见漏洞&#xff0c;本文介绍一个同样很常用的中间件——nginx&#xff0c;本文会讲解关于nginx中间件的常见漏洞利用方式与危害&#xff0c;下面展开来给大家讲解。 Nginx是什么&#xff1f; 在讲漏洞之前我们需要先知道什么是nginx,简单…

《SQL基础》11. 索引

SQL - 索引索引概述结构B-TreeBTreeHash思考分类语法SQL性能分析SQL执行频率慢查询日志profile详情explain执行计划索引失效情况范围查询索引列运算字符串不加引号模糊查询or连接条件数据分布影响使用规则最左前缀法则SQL提示覆盖索引前缀索引设计原则索引 概述 索引&#xf…

linux安装minio,注册服务实现开机自启动

一、安装1.创建安装目录mkdir -p /usr/local/soft/minio2.下载文件cd /usr/local/soft/miniowget -q http://dl.minio.org.cn/server/minio/release/linux-amd64/minio3.新建数据存储目录mkdir -p /home/minio/data4.授予可执行权限chmod x minio5.设置账号密码export MINIO_AC…

yolov8代码梳理 训练自己的数据 最终版

1.总结一下 最开始为了检测不规则的麻包袋&#xff0c;所以采用了目标检测。yolov3&#xff0c;fasterrcnn&#xff0c;ssd。这种矩形框还是可以用。后面检测的物体变成了规则的纸箱&#xff0c;我们还用目标检测发现&#xff0c;没有旋转角度&#xff0c;因为箱子的摆放不是正…

深度学习算法面试常问问题(三)

pooling层是如何进行反向传播的&#xff1f; average pooling&#xff1a; 在前向传播中&#xff0c;就是把一个patch的值取平均传递给下一层的一个像素。因此&#xff0c;在反向传播中&#xff0c;就是把某个像素的值平均分成n份 分配给上一层。 max pooling&#xff1a; 在前…

一文教你如何编写高效的接口测试,别再浪费时间了

目录 前言 1、什么是接口测试 2、为什么要做接口测试 3、接口测试的范围 4、接口测试的重点 5、测试原则 总结 前言 在所有的开发测试中&#xff0c;接口测试是必不可少的一项。有效且覆盖完整的接口测试&#xff0c;不仅能保障新功能的开发质量&#xff0c;还能让开发在…

PMP项目管理未来的发展与趋势

什么是项目管理&#xff1f;关于项目管理的解释主要是基于国际项目管理三大体系不同的解释及本领域权威专家的解释。 项目管理就是以项目为对象的系统管理方法&#xff0c;通过一个临时性的、专门的柔性组织&#xff0c;对项目进行高效率的计划、组织、指导和控制&#xff0c;以…

深度学习术语解释:backbone、head、neck,etc

backbone&#xff1a;翻译为主干网络的意思&#xff0c;既然说是主干网络&#xff0c;就代表其是网络的一部分&#xff0c;那么是哪部分呢&#xff1f;这个主干网络大多时候指的是提取特征的网络&#xff0c;其作用就是提取图片中的信息&#xff0c;共后面的网络使用。这些网络…

深入浅出带你学习利用session.upload_progress进行文件包含

前言 该思路是很久之前在CTF比赛中学习到的&#xff0c;可以简单理解为利用session.upload_progress来进行文件竞争从而达到上传文件进行文件包含或者命令执行的目的&#xff0c;可能大部分人会不理解&#xff0c;我们下面我们展开来讲。 基础知识 session.upload_progress …

Java并发编程面试题——线程安全(原子性、可见性、有序性)

文章目录一、原子性高频问题1.1 Java中如何实现线程安全?1.2 CAS底层实现1.3 CAS的常见问题1.4 四种引用类型 ThreadLocal的问题&#xff1f;二、可见性高频问题2.1 Java的内存模型2.2 保证可见性的方式2.3 volatile修饰引用数据类型2.4 有了MESI协议&#xff0c;为啥还有vol…

2022年全国职业院校技能大赛网络安全竞赛A模块(2)

2022年全国职业院校技能大赛&#xff08;中职组&#xff09; 网络安全竞赛试题 &#xff08;2&#xff09; 目录 2022年全国职业院校技能大赛&#xff08;中职组&#xff09; 网络安全竞赛试题 &#xff08;2&#xff09; 模块A 基础设施设置与安全加固 A-1任务一 登录安…

ChatGPT引爆资本狂潮,AI大战升级版打响!谷歌紧急上线对标竞品,微软测试AI搜索引擎,百度宣国产版定档,谁才是最后的赢家?

文 BFT机器人 如果你迄今为止还没有听说过ChatGPT&#xff0c;那就已经彻彻底底被人工智能的新时代甩到身后了。 如果说去年年底ChatGPT刚推出的时候&#xff0c;大多数网友对其的态度还是认为它和Clubhouse一样仅会是昙花一现的话&#xff0c;那么2个月时间过去&#xff0c;C…