React Router v4 и loadable-компоненты всегда отображают один и тот же компонент, даже если маршрут совпадает с другим

У меня есть стандартное приложение React-Redux с реактивным маршрутизатором v4, webpack 4, и я пытаюсь выполнить отложенную загрузку компонентов через библиотеку loadable-components, чтобы webpack мог создавать фрагменты и загружать их по требованию. Проблемы, кажется, это всегда рендеринг Dashboard (см. код ниже) Компонент внутри <Switch>, независимо от маршрута.

Я не понимаю причину. Я обнаружил похожую проблему: React-Router v4 рендерил неправильный компонент, но соответствовал правильно, но следовал другому шаблону, и я не могу или не понимаю, как применить это решение в моей проблеме.

Я публикую основной AppContainer и файл маршрутов:

AppContainer.js:

import React, { Component } from 'react'
import PropTypes from 'prop-types'
import { Provider } from 'react-redux'
import ReduxToastr from 'react-redux-toastr'
import 'react-redux-toastr/src/styles/index.scss'
import { Offline, Online } from 'react-detect-offline'
import Footer from 'components/Footer/'
import { Modal, ModalHeader, ModalBody } from 'reactstrap'
import { Translate } from 'react-redux-i18n'
import { PersistGate } from 'redux-persist/integration/react'
import { Router, Switch, Route } from 'react-router-dom'
import PrivateRoute from 'components/PrivateRoute'
import { Dashboard, PasswordReset, PasswordResetEdit, SignIn } from 'routes'

// Layout
import CoreLayout from 'layouts/CoreLayout'

class AppContainer extends Component {
  static propTypes = {
    history: PropTypes.object.isRequired,
    persistor: PropTypes.object.isRequired, // redux-persist
    store  : PropTypes.object.isRequired
  }

  render () {
    const { history, store, persistor } = this.props
    const opened = true
    const toastrDefaultTimeout = 8000
    const newToastrAlwaysOnTop = false

    return (
      <div>
        <Online>
          <Provider store={store}>
            <PersistGate loading={null} persistor={persistor}>
              <div style={{ height: '100%' }}>
                <ReduxToastr
                  timeOut={toastrDefaultTimeout}
                  newestOnTop={newToastrAlwaysOnTop}
                  preventDuplicates
                  position='top-right'
                  transitionIn='fadeIn'
                  transitionOut='fadeOut'
                  progressBar />
                <Router history={history}>
                  <CoreLayout {...this.props} >
                    <Switch>
                      <Route exact path='/sign-in' component={SignIn} />
                      <Route exact path='/passwords/new' component={PasswordReset} />
                      <Route exact path='/passwords/edit' component={PasswordResetEdit} />
                      <PrivateRoute exact path='/' component={Dashboard} persistor={persistor} />
                    </Switch>
                  </CoreLayout>
                </Router>
              </div>
            </PersistGate>
          </Provider>
        </Online>
        <Offline>
          <div className='app'>
            <div className='app-body'>
              <main className='main align-items-center align-self-center'>
                <div className='container'>
                  <div className='animated fadeIn'>
                    <Modal isOpen={opened} toggle={this.toggle}>
                      <ModalHeader toggle={this.toggle}><Translate value={`views.shared.no-internet.title`} /></ModalHeader>
                      <ModalBody>
                        <div className='card-block'>
                          <div className='row'>
                            <div className='col-sm-12'>
                              <Translate value='views.shared.no-internet.description' />
                            </div>
                          </div>
                        </div>
                      </ModalBody>
                    </Modal>
                  </div>
                </div>
              </main>
            </div>
            <Footer />
          </div>
        </Offline>
      </div>
    )
  }
}

export default AppContainer

маршруты /index.js:

// We only need to import the modules necessary for initial render
import loadable from 'loadable-components'
import { hot } from 'react-hot-loader'

// Component split-code (lazy load)
let Dashboard,
SignIn,
PasswordReset,
PasswordResetEdit

// Needed for HMR
if (__DEV__) {
  Dashboard = hot(module)(loadable(() => import('./Dashboard')))
  PasswordReset = hot(module)(loadable(() => import('./PasswordReset')))
  PasswordResetEdit = hot(module)(loadable(() => import('./PasswordResetEdit')))
  SignIn = hot(module)(loadable(() => import('./SignIn')))
} else {
  Dashboard = loadable(() => import('./Dashboard'))
  SignIn = loadable(() => import('./SignIn'))
  PasswordReset = loadable(() => import('./PasswordReset'))
  PasswordResetEdit = loadable(() => import('./PasswordResetEdit'))
}

export { Dashboard, PasswordReset, PasswordResetEdit, SignIn }

И для любого любопытного, вот PrivateRoute составная часть:

import React, { Component } from 'react'
import PropTypes from 'prop-types'
import { Route, Redirect } from 'react-router-dom'
import { connect } from 'react-redux'
import { validateSession } from 'modules/sessions/session'

const mapStateToProps = state => ({
  session: state.session
})

const mapDispatchToProps = (dispatch) => {
  return {
    validateSession: () => { dispatch(validateSession()) }
  }
}

class PrivateRoute extends Component {
  componentDidMount () {
    this.props.validateSession(this.props.persistor)
  }

  render () {
    const { component: Component, session, ...rest } = this.props

    return (
      <Route {...rest} render={(routeProps) =>
      this.props.session.userSignedIn ? <Component {...routeProps} />
      : <Redirect to={{
        pathname: '/sign-in',
        state: { from: routeProps.location }
      }} />
    }
  />)
  }
}

PrivateRoute.propTypes = {
  component: PropTypes.func.isRequired,
  persistor: PropTypes.object.isRequired,
  session: PropTypes.object.isRequired,
  validateSession: PropTypes.func.isRequired
}

export default connect(mapStateToProps, mapDispatchToProps)(PrivateRoute)

Если я удалю if (_DEV_) блокировать в файле routs/index.js и всегда делать это, как в блоке else, все работает нормально, однако я теряю HMR, то есть мне нужно обновить браузер, чтобы изменения в коде вступили в силу.

редактировать

Для тех, кто бродит, откуда берутся исторические данные: Основная точка входа в приложение (в webpack.config.js). SRC / main.js:

import React from 'react'
import ReactDOM from 'react-dom'
import createStore from './store/createStore'
import { hot } from 'react-hot-loader'
import AppContainer from './containers/AppContainer'
import { persistStore } from 'redux-persist'

// ========================================================
// Store Instantiation
// ========================================================

const initialState = window.___INITIAL_STATE__
const { history, store } = createStore(initialState)

// begin periodically persisting the store
let persistor = persistStore(store)

// ========================================================
// Render Setup
// ========================================================
const MOUNT_NODE = document.getElementById('root')

let render = () => {
  ReactDOM.render(
    <AppContainer store={store} persistor={persistor} history={history} />,
    MOUNT_NODE
  )
}

// This code is excluded from production bundle
if (__DEV__) {
  if (module.hot) {
    // Development render functions
    const renderApp = render
    const renderError = (error) => {
      const RedBox = require('redbox-react').default

      ReactDOM.render(<RedBox error={error} />, MOUNT_NODE)
    }

    // Wrap render in try/catch
    render = () => {
      try {
        renderApp()
      } catch (error) {
        console.error(error)
        renderError(error)
      }
    }

    // Setup hot module replacement
    module.hot.accept('./containers/AppContainer', () =>
      render(require('./containers/AppContainer').default)
    )
  }
}

// ========================================================
// Go!
// ========================================================
export default hot(module)(render)
render()

createStore.js:

import { applyMiddleware, compose, createStore } from 'redux'
import thunk from 'redux-thunk'
import { apiMiddleware } from 'redux-api-middleware'
import { createLogger } from 'redux-logger'
import makeRootReducer from './reducers'
import { routerMiddleware } from 'react-router-redux'
// I18n
import { syncTranslationWithStore, loadTranslations, setLocale } from 'react-redux-i18n'
import { translationsObject } from 'translations/index'
// Router history
import createHistory from 'history/createBrowserHistory'
// Raven for Sentry
import Raven from 'raven-js'
import createRavenMiddleware from 'raven-for-redux'

export default (initialState = {}) => {
  // ======================================================
  // Middleware Configuration
  // ======================================================
  const logger = createLogger()
  const history = createHistory()
  const historyMiddleware = routerMiddleware(history)

  const middleware = [historyMiddleware, apiMiddleware, thunk, logger]
  if (__PROD__) {
    Raven.config(`${__SENTRY_DSN__}`).install()
    middleware.push(createRavenMiddleware(Raven))
  }

  // ======================================================
  // Store Enhancers
  // ======================================================
  const enhancers = []

  let composeEnhancers = compose

  const composeWithDevToolsExtension = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__
  if (typeof composeWithDevToolsExtension === 'function') {
    composeEnhancers = composeWithDevToolsExtension
  }

  // ======================================================
  // Store Instantiation and HMR Setup
  // ======================================================
  const store = createStore(
    makeRootReducer(),
    initialState,
    composeEnhancers(
      applyMiddleware(...middleware),
      ...enhancers
    )
  )

  store.asyncReducers = {}

  // DEPRECATED in react-router v4: To unsubscribe, invoke `store.unsubscribeHistory()` anytime
  // store.unsubscribeHistory = browserHistory.listen(updateLocation(store))

  if (module.hot) {
    const reducers = require('./reducers').default
    module.hot.accept('./reducers', () => {
      store.replaceReducer(reducers(store.asyncReducers))
    })
  }

  syncTranslationWithStore(store)
  store.dispatch(loadTranslations(translationsObject))
  store.dispatch(setLocale('es_AR'))

  return { history, store }
}

0 ответов

Другие вопросы по тегам