最近、プロジェクトをReact v15.2.1から16.4.1にアップグレードしました。サイドバーコンポーネントは次のエラーをスローしています:
Error: Unable to find node on an unmounted component. bundle.js line 1326 > eval:42:15
invariant
webpack:///./node_modules/fbjs/lib/invariant.js?:42:15
findCurrentFiberUsingSlowPath
webpack:///./node_modules/react-dom/cjs/react-dom.development.js?:3817:7
findCurrentHostFiber
webpack:///./node_modules/react-dom/cjs/react-dom.development.js?:3886:23
findHostInstance
webpack:///./node_modules/react-dom/cjs/react-dom.development.js?:16824:19
findDOMNode
webpack:///./node_modules/react-dom/cjs/react-dom.development.js?:17310:12
handleClickOutside
webpack:///./src/components/simulator/sidebar/Sidebar.js?:99:31
handleClickOutside self-hosted:984:17
エラーメッセージに基づいて、handleClickOutside(event)メソッドでReactDOM.findDOMNode(this)を呼び出すときにエラーが発生していると思います。
私が使用しているコンポーネントはここにあります: https://ashiknesin.com/blog/build-custom-sidebar-component-react/ 、これを少し変更しました:
import React from 'react'
import ReactDOM from 'react-dom'
import classNames from 'classnames'
import SimulatorForm from './SimulatorForm'
import './Sidebar.scss'
const openForm = require('../../../public/icons/si-glyph-arrow-left.svg');
const closeForm = require('../../../public/icons/si-glyph-arrow-right.svg');
const pinForm = require('../../../public/icons/si-glyph-pin-location-love.svg');
const unpinForm = require('../../../public/icons/si-glyph-pin-location-delete.svg');
class Sidebar extends React.Component {
constructor(props) {
super(props)
this.state = {
showMenu: false,
isMenuPinned: false,
formIcon: openForm,
pinIcon: pinForm,
modelsDescription: props.modelsDescription
}
// Methods to pin/unpin the Form
this.toggleMenu = this.toggleMenu.bind(this)
this.pinMenu = this.pinMenu.bind(this)
// Handlers
this.handleSelectedModelChange = this.props.handleSelectedModelChange
this.handleNewMasterGraphsData = this.props.handleNewMasterGraphsData
this.handleNewResults = this.props.handleNewResults
}
componentWillReceiveProps(nextProps) {
this.setState({ modelsDescription: nextProps.modelsDescription});
}
componentDidMount() {
document.addEventListener('click', this.handleClickOutside.bind(this), true);
}
componentWillUnmount() {
document.removeEventListener('click', this.handleClickOutside.bind(this), true);
}
pinMenu() {
this.setState({
isMenuPinned: !this.state.isMenuPinned,
pinIcon: this.state.isMenuPinned ? pinForm : unpinForm
});
}
toggleMenu() {
this.setState({
showMenu: !this.state.showMenu,
formIcon: this.state.showMenu ? openForm : closeForm
});
}
handleClickOutside(event) {
if (!this.state.isMenuPinned) {
const domNode = ReactDOM.findDOMNode(this);
if ((!domNode || !domNode.contains(event.target))) {
this.setState({
showMenu: false,
formIcon: openForm
});
}
}
}
render() {
const showMenu = this.state.showMenu;
const sidebarClass = classNames({
'sidebar': true,
'sidebar-menu-expanded': showMenu,
'sidebar-menu-collapsed': !showMenu
});
const elementsClass = classNames({
'expanded-element': true,
'is-hidden': !showMenu,
});
return (
<nav className={sidebarClass}>
<img className="menuIcon" src={this.state.formIcon} height="42" width="42" onClick={this.toggleMenu} />
<ul>
<li>
{
this.state.showMenu ? <img className="pinIcon" src={this.state.pinIcon} height="42" width="42" onClick={this.pinMenu} /> : null
}
</li>
<li>
{
this.state.showMenu ? <SimulatorForm modelsDescription={this.state.modelsDescription} handleSelectedModelChange={this.handleSelectedModelChange}
handleNewMasterGraphsData={this.handleNewMasterGraphsData} handleNewResults={this.handleNewResults} /> : null
}
</li>
</ul>
</nav>
)
}
}
export default Sidebar
最後に、エラーはページをリロードしたときにのみスローされます。そうでない場合は、完全に正常に動作しているようです。エラーが再度スローされないようにするための提案や推奨事項はありますか?
私はこれについてオンラインで読んでいましたが、それに対する修正を見つけることができませんでした。また、これが重大な変更としてリストされているとは思いませんが、非常に間違っている可能性があります。
さて、コードの元の作者の助けのおかげで解決策を見つけることになりました。リンクを見つけることができます こちら 。それは私のバインディング、リンクの詳細に関する問題でした。
だから、ここで私がやったことだ。私は変わりました:
componentDidMount() {
document.addEventListener('click', this.handleClickOutside.bind(this), true);
}
componentWillUnmount() {
document.removeEventListener('click', this.handleClickOutside.bind(this), true);
}
に
componentDidMount = () => {
document.addEventListener("click", this.handleClickOutside, true);
};
componentWillUnmount = () => {
document.removeEventListener("click", this.handleClickOutside, true);
};