useHover 监听鼠标悬停

1. 需求描述--有啥用?

使用场景 当用户将鼠标放置在产品预览图上时,我们在预览图中以鼠标为中心,以 50 像素为半径划出部分区域并将该区域进行放大,将放大的图片展示在预览图旁边。当用户将鼠标在预览图上移动鼠标时,他可以改到对应区块的细节部分。

2. 需求分析--分析问题!

我们需要一个工具函数(或者hooks),这个工具函数(或者hooks)需要有以下几个功能:

【入参】

参数 含义
target DOM 节点或者 Ref 对象
options 额外的配置项

options额外参数

参数 含义 类型
onEnter hover 时触发 ( )=>{ }
onLeave 取消 hover 时触发 ( )=>{ }
onChange onChange ( )=>{ }

【返回值】

参数 含义 类型
isHovering 鼠标元素是否处于 hover boolean

综上所述,梳理思路如下:

首先我们要理解 hover 事件的本质其实就是鼠标进入事件和鼠标离开事件的语法糖

其次我们可以通过为传入的 dom 元素添加对应的事件处理函数达到我们的目的。

最后 当事件回调函数被触发时,执行对应的用户传入的函数后,还要更新鼠标元素是否处于 hover 的状态。

3. 手动实现--独立自主!

jsx 复制代码
import React, { useEffect, useRef, useState } from 'react'

function useHover(...props) {
  let [dom, options] = props
  // 是否处于悬停状态
  let [isHovering, setIsHovering] = useState(false)
  // 悬停状态发生改变时,执行用户传入的函数
  useEffect(() => {
    options?.onChange()
  }, [isHovering])

  useEffect(() => {
    if (!dom.current) return
    //  dom 节点存在时,为 dom 节点注册鼠标移入事件
    dom.current.onmouseenter = () => {
      setIsHovering(true)
      options?.onEnter()
    }
    //  dom 节点存在时,为 dom 节点注册鼠标移出事件
    dom.current.onmouseleave = () => {
      setIsHovering(false)
      options?.onLeave()
    }
  }, [dom])
  // 悬停状态作为 hooks 的返回值!
  return isHovering
}


// 测试案例
export default () => {
  let dom = useRef(null)
  let isHovering = useHover(dom, {
    onEnter: () => console.log('onEnter'),
    onLeave: () => console.log('onLeave'),
    onChange: () => console.log('onChange'),
  })
  return <>
    <div ref={dom} style={{ width: '550px', height: '550px', backgroundColor: 'pink' }}>标题</div>
    <span>{isHovering ? 'yse' : 'no'}</span>
  </>
}
相关推荐
镜宇秋霖丶2 小时前
2026.5.6@霖宇博客制作中遇见的问题
前端·javascript·vue.js
吴声子夜歌3 小时前
Vue3——TypeScript基础
javascript·typescript
小李子呢02113 小时前
前端八股Vue---Vue-router路由管理器
前端·javascript·vue.js
百锦再4 小时前
Auto.js变成基础知识学习
开发语言·javascript·学习·sqlite·kotlin·android studio·数据库开发
洛_尘5 小时前
Python 5:使用库
java·前端·python
Bigger5 小时前
Bun 能上生产吗?我的实战结论
前端·node.js·bun
kyriewen7 小时前
你的前端滤镜慢得像PPT?用Rust+WebAssembly,一秒处理4K图
前端·rust·webassembly
kyriewen117 小时前
你等的Babel编译,够喝三杯咖啡了——用Rust重写的SWC,只需眨个眼
开发语言·前端·javascript·后端·性能优化·rust·前端框架
IT_陈寒7 小时前
SpringBoot自动配置坑了我,原来要这样绕过去
前端·人工智能·后端
东方小月7 小时前
Claude Code 完整上手指南:MCP、Skills、第三方模型配置一次搞定
前端·人工智能·后端