I have created a bare-bones Meteor app, using React. It uses the three files shown below (and no others) in a folder called client
. In the Console, the App prints out:
withTracker
rendering
withTracker
rendering
props {} {}
state null null
In other words, the App component is rendered twice. The last two lines of output indicate that neither this.props
nor this.state
changed between renders.
index.html
<body>
<div id="react-target"></div>
</body>
main.jsx
import React from 'react'
import { render } from 'react-dom'
import App from './App.jsx'
Meteor.startup(() => {
render(<App/>, document.getElementById('react-target'));
})
App.jsx
import React from 'react'
import { withTracker } from 'meteor/react-meteor-data'
class App extends React.Component {
render() {
console.log("rendering")
return "Rendered"
}
componentDidUpdate(prevProps, prevState) {
console.log("props", prevProps, this.props)
console.log("state", prevState, this.state)
}
}
export default withTracker(() => {
console.log("withTracker")
})(App)
If I change App.jsx to the following (removing the withTracker
wrapper), then the App prints only rendering
to the Console, and it only does this once.
import React from 'react'
import { withTracker } from 'meteor/react-meteor-data'
export default class App extends React.Component {
render() {
console.log("rendering")
return "Rendered"
}
componentDidUpdate(prevProps, prevState) {
console.log(prevProps, this.props)
console.log(prevState, this.state)
}
}
What is withTracker
doing that triggers this second render? Since I cannot prevent it from occurring, can I be sure that any component that uses withTracker
will always render twice?
Context: In my real project, I use withTracker
to read data from a MongoDB collection, but I want my component to reveal that data only after a props
change triggers the component to rerender. I thought that it would be enough to set a flag after the first render, but it seems that I need to do something more complex.
Unsure if you're running into the same error I discovered, or if this is just standard React behavior that you're coming into here as suggested by other answers, but:
When running an older (
0.2.x
) version ofreact-meteor-data
on the2.0
Meteor, I was seeing two sets of distinct renders, one of which was missing crucialprops
and causing issues with server publications due to the missing data. Consider the following:For whatever reason, I was seeing not only N render calls but I was seeing N render calls that were missing properties in a duplicate manner. For those reading this and wonder, there was one and only one use of the component, one and only one use of the
withTracker
HoC, the parent HoCs had no logic that would cause conditional passing of props.Unfortunately, I have not discovered a root-cause of the bug. However, creating a fresh Meteor application and moving the code over was the only solution which removed the bug. An in-place update of the Meteor application (
2.0
to2.1
) and dependencies DID NOT solve the issue... however a fresh installation and running agit mv client imports server
did solve my problems.I've regrettably had to chalk this up to some form of drift due to subsequent Meteor updates over the two years of development.