1
0
Fork 0
mirror of https://github.com/terribleplan/next.js.git synced 2024-01-19 02:48:18 +00:00

Always load pages with ".js" extension (#3393)

* Always fetch pages with '.js' extention from client side.

* Load error page always from _error.js rather _error/index.js

* Load pages from page.js instead of page/index.js from the client for static exports.

* Update index.js

* Simplify the path re-write logic in the webpack pages-plugin.
This commit is contained in:
Arunoda Susiripala 2017-12-05 04:49:53 +05:30 committed by GitHub
parent a8c344fa19
commit 820e3ff716
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
9 changed files with 50 additions and 37 deletions

View file

@ -5,15 +5,15 @@ import Clock from './Clock'
@inject('store') @observer @inject('store') @observer
class Page extends React.Component { class Page extends React.Component {
componentDidMount() { componentDidMount () {
this.props.store.start() this.props.store.start()
} }
componentWillUnmount() { componentWillUnmount () {
this.props.store.stop() this.props.store.stop()
} }
render() { render () {
return ( return (
<div> <div>
<h1>{this.props.title}</h1> <h1>{this.props.title}</h1>

View file

@ -5,18 +5,18 @@ import { initStore } from '../store'
import Page from '../components/Page' import Page from '../components/Page'
export default class Counter extends React.Component { export default class Counter extends React.Component {
static getInitialProps({ req }) { static getInitialProps ({ req }) {
const isServer = !!req const isServer = !!req
const store = initStore(isServer) const store = initStore(isServer)
return { initialState: getSnapshot(store), isServer } return { initialState: getSnapshot(store), isServer }
} }
constructor(props) { constructor (props) {
super(props) super(props)
this.store = initStore(props.isServer, props.initialState) this.store = initStore(props.isServer, props.initialState)
} }
render() { render () {
return ( return (
<Provider store={this.store}> <Provider store={this.store}>
<Page title='Index Page' linkTo='/other' /> <Page title='Index Page' linkTo='/other' />

View file

@ -5,18 +5,18 @@ import { initStore } from '../store'
import Page from '../components/Page' import Page from '../components/Page'
export default class Counter extends React.Component { export default class Counter extends React.Component {
static getInitialProps({ req }) { static getInitialProps ({ req }) {
const isServer = !!req const isServer = !!req
const store = initStore(isServer) const store = initStore(isServer)
return { initialState: getSnapshot(store), isServer } return { initialState: getSnapshot(store), isServer }
} }
constructor(props) { constructor (props) {
super(props) super(props)
this.store = initStore(props.isServer, props.initialState) this.store = initStore(props.isServer, props.initialState)
} }
render() { render () {
return ( return (
<Provider store={this.store}> <Provider store={this.store}>
<Page title='Other Page' linkTo='/' /> <Page title='Other Page' linkTo='/' />

View file

@ -5,32 +5,31 @@ let store = null
const Store = types const Store = types
.model({ .model({
lastUpdate: types.Date, lastUpdate: types.Date,
light: false, light: false
}) })
.actions((self) => { .actions((self) => {
let timer; let timer
function start() { function start () {
timer = setInterval(() => { timer = setInterval(() => {
// mobx-state-tree doesn't allow anonymous callbacks changing data // mobx-state-tree doesn't allow anonymous callbacks changing data
// pass off to another action instead // pass off to another action instead
self.update(); self.update()
}) })
} }
function update() { function update () {
self.lastUpdate = Date.now() self.lastUpdate = Date.now()
self.light = true self.light = true
} }
function stop() { function stop () {
clearInterval(timer); clearInterval(timer)
} }
return { start, stop, update } return { start, stop, update }
}) })
export function initStore (isServer, snapshot = null) {
export function initStore(isServer, snapshot = null) {
if (isServer) { if (isServer) {
store = Store.create({ lastUpdate: Date.now() }) store = Store.create({ lastUpdate: Date.now() })
} }

View file

@ -1,4 +1,4 @@
/* global window, document, __NEXT_DATA__ */ /* global window, document */
import EventEmitter from './EventEmitter' import EventEmitter from './EventEmitter'
const webpackModule = module const webpackModule = module
@ -69,10 +69,7 @@ export default class PageLoader {
loadScript (route) { loadScript (route) {
route = this.normalizeRoute(route) route = this.normalizeRoute(route)
route = route === '/' ? '/index.js' : `${route}.js`
if (__NEXT_DATA__.nextExport) {
route = route === '/' ? '/index.js' : `${route}/index.js`
}
const script = document.createElement('script') const script = document.createElement('script')
const url = `${this.assetPrefix}/_next/${encodeURIComponent(this.buildId)}/page${route}` const url = `${this.assetPrefix}/_next/${encodeURIComponent(this.buildId)}/page${route}`

View file

@ -28,6 +28,16 @@ export default class PagesPlugin {
routeName = `/${routeName.replace(/(^|\/)index$/, '')}` routeName = `/${routeName.replace(/(^|\/)index$/, '')}`
// If there's file named pageDir/index.js
// We are going to rewrite it as pageDir.js
// With this, we can statically decide the filepath of the page
// based on the page name.
const rule = /^bundles[/\\]pages[/\\].*[/\\]index\.js$/
if (rule.test(chunk.name)) {
delete compilation.assets[chunk.name]
chunk.name = chunk.name.replace(/[/\\]index\.js$/, `.js`)
}
const content = page.source() const content = page.source()
const newContent = ` const newContent = `
window.__NEXT_REGISTER_PAGE('${routeName}', function() { window.__NEXT_REGISTER_PAGE('${routeName}', function() {

View file

@ -80,13 +80,13 @@ export class Head extends Component {
render () { render () {
const { head, styles, __NEXT_DATA__ } = this.context._documentProps const { head, styles, __NEXT_DATA__ } = this.context._documentProps
const { pathname, buildId, assetPrefix, nextExport } = __NEXT_DATA__ const { pathname, buildId, assetPrefix } = __NEXT_DATA__
const pagePathname = getPagePathname(pathname, nextExport) const pagePathname = getPagePathname(pathname)
return <head {...this.props}> return <head {...this.props}>
{(head || []).map((h, i) => React.cloneElement(h, { key: i }))} {(head || []).map((h, i) => React.cloneElement(h, { key: i }))}
<link rel='preload' href={`${assetPrefix}/_next/${buildId}/page${pagePathname}`} as='script' /> <link rel='preload' href={`${assetPrefix}/_next/${buildId}/page${pagePathname}`} as='script' />
<link rel='preload' href={`${assetPrefix}/_next/${buildId}/page/_error/index.js`} as='script' /> <link rel='preload' href={`${assetPrefix}/_next/${buildId}/page/_error.js`} as='script' />
{this.getPreloadDynamicChunks()} {this.getPreloadDynamicChunks()}
{this.getPreloadMainLinks()} {this.getPreloadMainLinks()}
{styles || null} {styles || null}
@ -174,8 +174,8 @@ export class NextScript extends Component {
render () { render () {
const { staticMarkup, __NEXT_DATA__, chunks } = this.context._documentProps const { staticMarkup, __NEXT_DATA__, chunks } = this.context._documentProps
const { pathname, nextExport, buildId, assetPrefix } = __NEXT_DATA__ const { pathname, buildId, assetPrefix } = __NEXT_DATA__
const pagePathname = getPagePathname(pathname, nextExport) const pagePathname = getPagePathname(pathname)
__NEXT_DATA__.chunks = chunks __NEXT_DATA__.chunks = chunks
@ -197,15 +197,17 @@ export class NextScript extends Component {
` `
}} />} }} />}
<script async id={`__NEXT_PAGE__${pathname}`} type='text/javascript' src={`${assetPrefix}/_next/${buildId}/page${pagePathname}`} /> <script async id={`__NEXT_PAGE__${pathname}`} type='text/javascript' src={`${assetPrefix}/_next/${buildId}/page${pagePathname}`} />
<script async id={`__NEXT_PAGE__/_error`} type='text/javascript' src={`${assetPrefix}/_next/${buildId}/page/_error/index.js`} /> <script async id={`__NEXT_PAGE__/_error`} type='text/javascript' src={`${assetPrefix}/_next/${buildId}/page/_error.js`} />
{staticMarkup ? null : this.getDynamicChunks()} {staticMarkup ? null : this.getDynamicChunks()}
{staticMarkup ? null : this.getScripts()} {staticMarkup ? null : this.getScripts()}
</div> </div>
} }
} }
function getPagePathname (pathname, nextExport) { function getPagePathname (pathname) {
if (!nextExport) return pathname if (pathname === '/') {
if (pathname === '/') return '/index.js' return '/index.js'
return `${pathname}/index.js` }
return `${pathname}.js`
} }

View file

@ -141,11 +141,13 @@ function copyPages (nextDir, outDir, buildId) {
} }
let destFilePath = null let destFilePath = null
if (/index\.js$/.test(filename)) { if (relativeFilePath === `${sep}index.js`) {
destFilePath = join(outDir, '_next', buildId, 'page', relativeFilePath) destFilePath = join(outDir, '_next', buildId, 'page', relativeFilePath)
} else { } else if (/index\.js$/.test(filename)) {
const newRelativeFilePath = relativeFilePath.replace(/\.js/, `${sep}index.js`) const newRelativeFilePath = relativeFilePath.replace(`${sep}index.js`, '.js')
destFilePath = join(outDir, '_next', buildId, 'page', newRelativeFilePath) destFilePath = join(outDir, '_next', buildId, 'page', newRelativeFilePath)
} else {
destFilePath = join(outDir, '_next', buildId, 'page', relativeFilePath)
} }
cp(fullFilePath, destFilePath) cp(fullFilePath, destFilePath)

View file

@ -201,7 +201,10 @@ export default class Server {
'/_next/:buildId/page/:path*': async (req, res, params) => { '/_next/:buildId/page/:path*': async (req, res, params) => {
const paths = params.path || [''] const paths = params.path || ['']
const page = `/${paths.join('/')}` // We need to remove `.js` from the page otherwise it won't work with
// page rewrites
// eg:- we re-write page/index.js into page.js
const page = `/${paths.join('/')}`.replace('.js', '')
if (!this.handleBuildId(params.buildId, res)) { if (!this.handleBuildId(params.buildId, res)) {
const error = new Error('INVALID_BUILD_ID') const error = new Error('INVALID_BUILD_ID')