草庐IT

javascript - 如何通过React和鼠标事件传播实现可重用组件?

coder 2024-07-15 原文

考虑以下典型的React文档结构:

Component.jsx

<OuterClickableArea>
    <InnerClickableArea>
        content
    </InnerClickableArea>
</OuterClickableArea>

这些组件的组成如下:

OuterClickableArea.js
export default class OuterClickableArea extends React.Component {
    constructor(props) {
        super(props)

        this.state = {
            clicking: false
        }

        this.onMouseDown = this.onMouseDown.bind(this)
        this.onMouseUp = this.onMouseUp.bind(this)
    }

    onMouseDown() {
        if (!this.state.clicking) {
            this.setState({ clicking: true })
        }
    }

    onMouseUp() {
        if (this.state.clicking) {
            this.setState({ clicking: false })
            this.props.onClick && this.props.onClick.apply(this, arguments)
            console.log('outer area click')
        }
    }

    render() {
        return (
            <div
                onMouseDown={this.onMouseDown}
                onMouseUp={this.onMouseUp}
            >
                { this.props.children }
            </div>
        )
    }
}

出于这个原因,InnerClickableArea.js具有几乎相同的代码,除了类名和控制台日志语句。

现在,如果您要运行此应用程序,并且用户单击内部可点击区域,则会发生以下情况(按预期):
  • 外围区域注册鼠标事件处理程序
  • 内部区域注册鼠标事件处理程序
  • 用户在内部区域
  • 中按下鼠标
  • 内部区域的mouseDown侦听器触发
  • 外围区域的mouseDown侦听器触发
  • 用户释放鼠标向上
  • 内部区域的mouseUp侦听器触发
  • 控制台记录“内部区域单击”
  • 外围区域的mouseUp侦听器触发
  • 控制台记录“外部单击”

  • 这显示了典型的事件冒泡/传播-到目前为止没有任何惊喜。

    它将输出:
    inner area click
    outer area click
    

    现在,如果我们正在创建一个应用程序,一次只能进行一次交互,该怎么办?例如,假设有一个编辑器,可以在其中单击鼠标来选择元素。在内部区域中按下时,我们只希望选择内部元素。

    一个简单而明显的解决方案是在InnerArea组件内部添加一个stopPropagation:

    InnerClickableArea.js
        onMouseDown(e) {
            if (!this.state.clicking) {
                e.stopPropagation()
                this.setState({ clicking: true })
            }
        }
    

    这按预期工作。它将输出:
    inner area click
    

    有问题吗?
    InnerClickableArea在此隐式选择OuterClickableArea(及所有其他父项)以使其不能够接收事件。即使InnerClickableArea不应该知道OuterClickableArea的存在。就像OuterClickableArea并不了解InnerClickableArea(遵循关注点和可重用性概念的分离)。我们在这两个组件之间创建了一个隐式依赖关系,其中OuterClickableArea“知道”它不会错误地触发其侦听器,因为它“记住” InnerClickableArea将停止任何事件传播。这似乎是错误的。

    我尝试不使用stopPropagation来提高应用程序的可伸缩性,因为添加依赖事件的新功能会更容易,而不必记住哪个组件在何时使用stopPropagation

    相反,我想使上述逻辑更具说明性,例如通过在Component.jsx内声明性地定义每个区域当前是否可单击。我将使它成为应用程序状态感知器,传递区域是否可单击,并将其重命名为Container.jsx。像这样:

    Container.jsx
    <OuterClickableArea
        clickable={!this.state.interactionBusy}
        onClicking={this.props.setInteractionBusy.bind(this, true)} />
    
        <InnerClickableArea
            clickable={!this.state.interactionBusy}
            onClicking={this.props.setInteractionBusy.bind(this, true)} />
    
                content
    
        </InnerClickableArea>
    
    </OuterClickableArea>
    

    其中this.props.setInteractionBusy是一个Redux Action ,它将导致应用状态被更新。另外,此容器会将应用程序状态映射到其 Prop (上面未显示),因此将定义this.state.ineractionBusy

    OuterClickableArea.js(与InnerClickableArea.js几乎相同)
    export default class OuterClickableArea extends React.Component {
        constructor(props) {
            super(props)
    
            this.state = {
                clicking: false
            }
    
            this.onMouseDown = this.onMouseDown.bind(this)
            this.onMouseUp = this.onMouseUp.bind(this)
        }
    
        onMouseDown() {
            if (this.props.clickable && !this.state.clicking) {
                this.setState({ clicking: true })
                this.props.onClicking && this.props.onClicking.apply(this, arguments)
            }
        }
    
        onMouseUp() {
            if (this.state.clicking) {
                this.setState({ clicking: false })
                this.props.onClick && this.props.onClick.apply(this, arguments)
                console.log('outer area click')
            }
        }
    
        render() {
            return (
                <div
                    onMouseDown={this.onMouseDown}
                    onMouseUp={this.onMouseUp}
                >
                    { this.props.children }
                </div>
            )
        }
    }
    

    问题在于Javascript事件循环似乎按以下顺序运行这些操作:
  • OuterClickableAreaInnerClickableArea都是用等于clickable的prop true创建的(因为应用程序状态交互忙状态默认为false)
  • 外围区域注册鼠标事件处理程序
  • 内部区域注册鼠标事件处理程序
  • 用户在内部区域
  • 中按下鼠标
  • 内部区域的mouseDown侦听器触发
  • 内部区域的onClicking被触发
  • 容器运行'setInteractionBusy'操作
  • redux应用程序状态interactionBusy设置为true
  • 外围区域的mouseDown侦听器触发(它的clickable属性仍然是true,因为即使应用程序状态interactionBusytrue,react仍未导致重新渲染;渲染操作位于Javascript事件循环的末尾)
  • 用户释放鼠标向上
  • 内部区域的mouseUp侦听器触发
  • 控制台记录“内部区域单击”
  • 外围区域的mouseUp侦听器触发
  • 控制台记录“外部单击”
  • react重新渲染Component.jsx并将clickable作为false传递给两个组件,但是现在
  • 为时已晚

    这将输出:
    inner area click
    outer area click
    

    而所需的输出是:
    inner area click
    

    因此,应用程序状态无助于使这些组件更加独立和可重用。原因似乎是,即使状态已更新,容器也只会在事件循环结束时重新渲染,并且鼠标事件传播触发器已在事件循环中排队。

    因此,我的问题是:是否可以使用声明状态的应用状态来替代使用鼠标状态传播的应用状态,还是可以通过应用(redux)状态来实现/执行上述设置? ?

    最佳答案

    最简单的替代方法是在触发stopPropogation之前添加一个标志,这种情况下该标志是一个参数。

    const onMouseDown = (stopPropagation) => {
    stopPropagation && event.stopPropagation();
    }
    

    现在,即使是应用程序状态或 Prop 也可以决定触发停止传播
    <div onMouseDown={this.onMouseDown(this.state.stopPropagation)} onMouseUp={this.onMouseUp(this.props.stopPropagation)} >
        <InnerComponent stopPropagation = {true}>
    </div>
    

    关于javascript - 如何通过React和鼠标事件传播实现可重用组件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50298440/

    有关javascript - 如何通过React和鼠标事件传播实现可重用组件?的更多相关文章

    1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

      我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

    2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

      总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

    3. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

      关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

    4. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

      给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

    5. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

      我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

    6. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

      我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

    7. ruby - 通过 rvm 升级 ruby​​gems 的问题 - 2

      尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub

    8. ruby - 如何指定 Rack 处理程序 - 2

      Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

    9. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

      在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

    10. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

      我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

    随机推荐