Javascript React 手机上的 右键点击(长按)事件

右键点击事件:

世上本没问题,闲多了,就有了问题: 开会不准玩电脑,只能玩手机,玩儿了阵子扫雷,决定自己做一个网页版。三下五除二做完之后,猛然发现,手机没有右键。。。。。。没有右键怎么插小红旗?!
在这里插入图片描述
想起来app上是用长按来插小红旗的,于是开始了长达几个小时的纠结。

React Component监听点击事件

最开始的时候,格子接受左键和右键点击不同的event。
onClick用来监听左键点击,onContextMenu用来监听右键点击,可以愉快的玩耍,么得问题的。

import React from 'react';
import '../styles/grid.css'

class SingleGrid extends React.Component {
    constructor(props) {
        super(props);
        this.state = {status: 'default'};
        this.handleClick = this.handleClick.bind(this);
        this.handleRightClick = this.handleRightClick.bind(this);
    }

    handleClick() {
		//....一些左键点击事件处理逻辑,与本文无关。。。。
    }

    handleRightClick(e) {
       	//此处省略若干行
        // marked是插小红旗,default是啥都没有,
        this.setState({
            status: this.state.status === 'marked' ? 'default' : 'marked'
        });
    }
    render () {
        let currentStatue = '';
        //。。。此处省略若干行。。。
        return <span
            className={`grid${currentStatue} grid ${currentStatue && currentStatue!== 'marked' ? 'open' : 'default' }`}
            onClick={this.handleClick}
            onContextMenu={this.handleRightClick}
        >{/*格子内容省略*/}</span>
    }
}
export default SingleGrid

手机用长按代替右键点击

直到,开会不准玩电脑呀!手机怎么搞啊?祭出longPress大法:网上找了好几个react的lib,都不老好用,不是install时出问题就是build时出问题(时间主要浪费在这里了)。放弃lib,自己搞。
网上代码很多,大概是按住之后setTimeout,松开之后cancel timeout。

/**
 * Created by xzou2 on 11/13/18.
 */
import React from 'react';
import '../styles/grid.css'

class SingleGrid extends React.Component {
    constructor(props) {
        super(props);
        this.state = {status: 'default'};
        this.handleClick = this.handleClick.bind(this);
        this.handleRightClick = this.handleRightClick.bind(this);
        this.handleTouchEnd = this.handleTouchEnd.bind(this);
        this.touchStart = this.touchStart.bind(this);
    }

    handleClick() {
		//一直是个不惹麻烦的乖宝宝
    }

    handleRightClick(e) {
        //同上
    }

    handleTouchEnd(){
        clearTimeout(this.pressTime);
    }

    touchStart(){
        this.pressTime  = setTimeout(()=>{
            this.setState({
                status: this.state.status === 'marked' ? 'default' : 'marked'
            });
        }, '1000');
    }

    render () {
        let currentStatue = '';
        //此处省略乖宝宝代码
        return <span
            className={`grid${currentStatue} grid ${currentStatue && currentStatue!== 'marked' ? 'open' : 'default' }`}
            onClick={this.handleClick}
            onContextMenu={this.handleRightClick}
            onTouchStart={this.touchStart}
            onTouchEnd={this.handleTouchEnd}
        >{/*格子内容省略*/}</span>
    }
}
export default SingleGrid

onTouchStart和onContextMenu重复执行

问题出现了,长按之后,小红旗插上了,但是只要稍微松开晚一些,小红旗就又取消了。
经过观察,按住之后超过一段时间,render()方法执行了两遍。经过漫长的debug,原来handleRightClick方法也被执行了。负负得正,一个插旗一个拔旗。
花了一些时间研究了一下为啥浏览器会trigger 右键点击事件,未果,一拍脑门,既然不能阻止浏览器触发事件,那就让他触发,但是不干活就是了。所以handleRightClick方法在touchStart被触发的时候(触屏)就不要干活了。

    handleRightClick(e) {
        e.stopPropagation();//这两行是为了阻止出现浏览器菜单
        e.preventDefault();
        if(this.pressTime) {return;} //如果touchstart触发,说明是触屏,这个方法不往下执行。
        this.setState({
            status: this.state.status === 'marked' ? 'default' : 'marked'
        });
    }

到目前为止,大概,可能。。。已经解决了这个事儿了吧,如果又有新的bug,再说吧。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
React 17 并没有直接提供事件的功能,但你可以通过使用原生 DOM 的方式来实现。你可以在 React 组件中添加一个 ref 属性,并将其设置为一个函数,然后在该函数中获取对应的 DOM 元素。接着,你可以在该 DOM 元素上监听 "mousedown" 和 "mouseup" 事件,并使用 setTimeout 函数来实现按功能。 具体实现如下所示: ```jsx import React, { useRef } from 'react'; function LongPressButton() { const buttonRef = useRef(null); let timer = null; const handleMouseDown = () => { timer = setTimeout(() => { console.log('Long press triggered!'); }, 1000); }; const handleMouseUp = () => { clearTimeout(timer); }; return ( <button ref={buttonRef} onMouseDown={handleMouseDown} onMouseUp={handleMouseUp} > Long Press Button </button> ); } export default LongPressButton; ``` 在上面的代码中,我们使用了 useRef 钩子来创建了一个 buttonRef 引用。在组件渲染时,我们将该引用赋值给了 button 元素。然后,我们在 button 元素上监听了 onMouseDown 和 onMouseUp 事件,并分别触发 handleMouseDown 和 handleMouseUp 函数。在 handleMouseDown 函数中,我们使用 setTimeout 函数来实现按功能。在 handleMouseUp 函数中,我们清除了定时器,以避免事件被错误地触发。 需要注意的是,上面的代码只是一个简单的示例,并没有考虑到一些复杂的情况。例如,在触发事件之前,用户可能会拖动鼠标或者取消点击等操作。因此,在实际开发中,你可能需要对代码进行一些修改和优化。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值