gpt4 book ai didi

javascript - 自从更新react-dom和react-router-dom包: excessive li onClick behavior

转载 作者:行者123 更新时间:2023-12-02 22:42:52 24 4
gpt4 key购买 nike

我继承了一个react/node/prismic应用程序,我们需要更新prismic-reactjs包,这导致需要更新其他一些包——总的来说,我们改变了:

prismic-reactjs:0.2.0 → 1.1.0

react :15.6.1 → 16.0.0

webpack:3.12.0 → 4.39.2

react-dom:15.6.1 → 16.0.0

react-router-dom:4.1.2 → 5.0.1

extract-text-webpack-plugin(已弃用)→ mini-css-extract-plugin

然后删除了“withRouter()”的一个使用,因为启动本地服务器时出现了新错误(但我在另一个分支中确认,单独进行该编辑不会产生下面的症状)

上下文:我们有一个 personMap,它显示一组人的图标,您单击其中一个即可打开 PersonMapStory 模式,该模式显示该人的故事,然后链接到该人的其他人底部的列表。如果您向下滚动并单击其中一个链接,顶部的故事将相应替换(用新人的故事),下面列出的是其他所有人。在代码中,这些底部链接的按钮 onClick 行为是 setActivePerson() 给这个新人。

新错误:在这个新分支中,当我们单击底部链接时,它会为#2 调用 setActivePerson(),但随后也会再次为 #1 调用 setActivePerson()! (从原始 PersonMap 里的 onClick 行为返回)。因此,症状是您看到第一个故事,向下滚动并单击另一个人的链接,但模式似乎根本没有更新(尽管它实际上更新了两次)。在当前/生产/工作分支中,它只更新一次给新人,仅此而已。这种新行为听起来确实类似于 this onClick 传播问题。我尝试将 stopPropagation() 添加到适当的方法中,但没有成功。

更多信息:在这个分支中,看起来来自该模式内部的任何点击(即使我点击顶部显示的故事)都会调用原始人的 SetActivePerson() ,来自当 thisPersonIsQueued 为 true 时,PersonMap 订单项 onClick 行为。 (因此,如果我向下滚动并单击一个新人,我们会 SetActivePerson person #2,然后立即 SetActivePerson person #1,因此模式会保持相同的故事并且永远不会更新)。在 PersonMap 状态(在 Chrome 开发者工具/组件中查看)中,有一个 queuedUpPatient:通常在这种情况下(在原始/生产代码中)queuedUpPatient 为 null,但在这个新分支中,queuedUpPatient 保留 Person #1 的值。 (或者已清除并重新设置)。

调试器中可能的线索:在我们为 #2 调用 SetActivePerson 之后(这是我们常规/生产分支中发生的最后一步),在我们看到对 person #1 的 setActivePerson 调用之前它是 onClick 行为,调试器引导我完成react-dom.development.js 中的几个方法,从 callCallback() 的最后一行开始,在以下情况的内部代码中(见下文):

// Check that the browser supports the APIs we need to implement our special
// DEV version of invokeGuardedCallback
if (typeof window !== 'undefined' && typeof window.dispatchEvent === 'function' && typeof document

关于可能发生的情况以及如何使其再次正常工作以便我们可以继续进行软件包升级的其他想法?这是 map 和模态的两个相关文件(如下)。

PersonMap.js:

import Modal from 'react-modal'
import PropTypes from 'prop-types'
import React from 'react'

import PersonMapPoint from './PersonMapPoint'
import PersonMapStory from './PersonMapStory'
import PersonMapCallout from './PersonMapCallout'
import PersonMapLocator from './PersonMapLocator'
import PersonMapBackground from './PersonMapBackground'

const CUSTOM_STYLES = {
content: {
top: 0,
left: 0,
right: 0,
bottom: 0,
backgroundColor: '#fff',
zIndex: 10,
border: 'none'
}
}

class PersonMap extends React.Component {
constructor(props) {
super(props)

this.setActivePerson = this.setActivePerson.bind(this)
this.setNoActivePerson = this.setNoActivePerson.bind(this)
this.setQueuedUpPerson = this.setQueuedUpPerson.bind(this)
this.setNoQueuedUpPerson = this.setNoQueuedUpPerson.bind(this)
this.checkBlurEvent = this.checkBlurEvent.bind(this)
this.setIsDesktop = this.setIsDesktop.bind(this)
this.checkHasBeenScrolledTo = this.checkHasBeenScrolledTo.bind(this)
this.setTopRef = this.setTopRef.bind(this)
this.handleKeyPress = this.handleKeyPress.bind(this)

this.state = {
activePerson: null,
queuedUpPerson: null,
scrollPos: 0,
isDesktop: false,
hasBeenScrolledTo: false,
lastQueuedPerson: null
}
}

componentDidMount() {
this.setIsDesktop()
this.checkHasBeenScrolledTo()
window.addEventListener('resize', this.setIsDesktop)
window.addEventListener('scroll', this.checkHasBeenScrolledTo)
}

componentWillUnmount() {
window.removeEventListener('resize', this.setIsDesktop)
window.removeEventListener('scroll', this.checkHasBeenScrolledTo)
}

setTopRef(element) {
this.topRef = element
}

setActivePerson(personName) {
this.setState({
activePerson: personName,
scrollPos: window.scrollY
})
event.stopPropagation()
}

setNoActivePerson() {
this.setState({
queuedUpPerson: this.state.activePerson,
activePerson: null
}, () => {
setTimeout(() => {
window.scrollTo(0, this.state.scrollPos)
}, 50)
})
}

setQueuedUpPerson(personName) {
this.setState({
queuedUpPerson: personName,
lastQueuedPerson: personName
})
}

setNoQueuedUpPerson() {
this.setState({
queuedUpPerson: null
})
event.stopPropagation()
}

handleKeyPress(e, name) {
if (e.key !== ' ' && e.key !== 'Enter') {
return
}
this.setActivePerson(name)
}

checkBlurEvent(e) {
if (Array.from(e.currentTarget.childNodes[0].childNodes).includes(e.relatedTarget)) {
return
}
this.setNoQueuedUpPerson()
}

render() {
return (
<section className="slice-area person-map CONSTRAIN">
<div className="person-map-headers">
<div className="person-map-headers-inner">
<h1 className="person-map-title">
{this.props.title}
</h1>
<p className="person-map-disclaimer">
{this.props.disclaimer}
</p>
</div>
</div>
<div
className={`person-map-list-wrap${ this.state.isDesktop ? ' --desktop' : '' }`}
ref={this.state.isDesktop && this.setTopRef}
>
{ this.state.isDesktop &&
<PersonMapBackground isVisible={this.state.hasBeenScrolledTo}/>
}
<ul className="person-map-list">
{
this.props.personStories.map((person) => {
const thisPersonIsQueued = this.state.queuedUpPerson === person.name
const thisPersonIsActive = this.state.activePerson === person.name
const thisPersonWasLastQueued = this.state.lastQueuedPerson === person.name

return (
<li
key={person.name} className={`person-map-list-item${thisPersonWasLastQueued ? ' --active' : ''}`}
onMouseEnter={this.state.isDesktop ? () => this.setQueuedUpPerson(person.name) : null}
onMouseLeave={this.state.isDesktop ? this.setNoQueuedUpPerson : null}
onFocus={this.state.isDesktop ? () => this.setQueuedUpPerson(person.name) : null}
onBlur={this.state.isDesktop ? this.checkBlurEvent : null}
onClick={thisPersonIsQueued || !this.state.isDesktop ? () => this.setActivePerson(person.name) : null}
onKeyPress={(e) => this.handleKeyPress(e, person.name)}
>
{
<PersonMapLocator
x={person.x}
y={person.y}
>
}
<Modal
isOpen={thisPersonIsActive}
onRequestClose={this.setNoActivePerson}
style={CUSTOM_STYLES}
>
<PersonMapStory
name={person.name}
photo={person.photo_url}
story={person.story}
setNoActivePerson={this.setNoActivePerson}
setActivePerson={this.setActivePerson}
isActive={thisPersonIsActive}
otherPersons={this.props.personStories.filter((item) => item.name !== person.name).map((item) => ({ name: item.name, photo: item.photo_url }))}
isDesktop={this.state.isDesktop}
/>
</Modal>
</li>
)
})
}
</ul>
</div>
</section>
)
}
}

PersonMap.propTypes = {
title: PropTypes.string,
disclaimer: PropTypes.string,
personStories: PropTypes.arrayOf(PropTypes.shape({
name: PropTypes.string,
photo_url: PropTypes.string,
x: PropTypes.number,
y: PropTypes.number,
story: PropTypes.shape({
title: PropTypes.string,
link: PropTypes.string,
content: PropTypes.arrayOf(PropTypes.shape({
question: PropTypes.string,
answer: PropTypes.string
}))
})
}))
}
export default PersonMap

PersonMapStory.js:

import PropTypes from 'prop-types'
import React from 'react'

import PersonMapPerson from './PersonMapPerson'


class PersonMapStory extends React.Component {
constructor(props) {
console.log("PersonMapStory constructor")
super(props)

this.deactivateThisPerson = this.deactivateThisPerson.bind(this)
this.check = 0
}

deactivateThisPerson(backPressed = false) {
if (!backPressed) {
history.back()
}
console.log("Set no active person ")
this.props.setNoActivePerson()
}

setActivePerson(name) {
this.props.setActivePerson(name)
event.stopPropagation()
}


componentDidMount() {
const loc = window.location.pathname.substr(1)
this.setState({ location: loc })
const openState = { modalOpen: true }
history.pushState(openState, 'signup-open', loc)
}

render() {
return (
<div className={`person-map-story${this.props.isActive ? ' --active' : ''}${this.props.isDesktop ? ' --desktop' : ''}`}>
<h2 className="person-map-story-title">
{ this.props.story.title }
</h2>
<button className="person-map-story-close-button" onClick={() => {this.deactivateThisPerson(false)}}>
<span className="person-map-story-close-button-text">
Close Story
</span>
</button>
<div className="person-map-story-body">
<span className="person-map-story-person-wrap">
<PersonMapPerson
photo={this.props.photo}
name={this.props.name}
/>
</span>
{
this.props.story.content.map((section) => {
if (section) {
return (
<div key={section.question}>
<h3 className="person-map-story-question">{section.question}</h3>
<p className="person-map-story-answer">
{section.answer}
</p>
</div>
)
}
})
}
{
this.props.story.link &&
<a href={this.props.story.link}
target="_blank"
className="person-map-story-more-link header-and-text-link"
> Read More Stories
</a>
}

<ul className="person-map-story-other-list">
{
this.props.otherPersons.map((person) => (
<li
key={person.name}
className="person-map-story-other-list-item"
>
<button className="person-map-story-other-button" onClick={() => this.setActivePerson(person.name)}>
<PersonMapPerson
name={person.name}
photo={person.photo}
ctaText="View their story"
/>
</button>
</li>
))
}
</ul>
</div>
</div>
)
}
}

PersonMapStory.propTypes = {
name: PropTypes.string,
photo: PropTypes.string,
story: PropTypes.shape({
title: PropTypes.string,
link: PropTypes.string,
content: PropTypes.arrayOf(PropTypes.shape({
question: PropTypes.string,
answer: PropTypes.string
}))
}),
setNoActivePerson: PropTypes.func,
setActivePerson: PropTypes.func,
isActive: PropTypes.bool,
otherPersons: PropTypes.arrayOf(PropTypes.shape({
name: PropTypes.string,
photo: PropTypes.string
})),
isDesktop: PropTypes.bool
}

export default PersonMapStory

最佳答案

更新 PersonMapStory 中的 onClick 以访问 event 对象:

<button className="person-map-story-other-button" onClick={e => this.setActivePerson(e, person.name)}>

并更改 PersonMapStory 中的 setActivePerson(name) 函数:

setActivePerson(event, name) {
this.props.setActivePerson(name)
event.stopPropagation()
}

关于javascript - 自从更新react-dom和react-router-dom包: excessive li onClick behavior,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58530473/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com