2020-06-10 20:54:23 +00:00
|
|
|
/* eslint-disable no-restricted-globals */
|
2019-12-12 17:42:21 +00:00
|
|
|
import {
|
2020-08-19 20:11:29 +00:00
|
|
|
ConcreteComponent,
|
2019-12-12 17:42:21 +00:00
|
|
|
ComponentInternalInstance,
|
|
|
|
ComponentOptions,
|
2020-09-18 04:14:59 +00:00
|
|
|
InternalRenderFunction,
|
|
|
|
ClassComponent,
|
|
|
|
isClassComponent
|
2019-12-12 17:42:21 +00:00
|
|
|
} from './component'
|
2019-12-12 23:13:59 +00:00
|
|
|
import { queueJob, queuePostFlushCb } from './scheduler'
|
2020-06-10 20:54:23 +00:00
|
|
|
import { extend } from '@vue/shared'
|
2020-11-21 21:06:50 +00:00
|
|
|
import { warn } from './warning'
|
2019-12-12 17:42:21 +00:00
|
|
|
|
2020-06-12 18:53:48 +00:00
|
|
|
export let isHmrUpdating = false
|
|
|
|
|
2020-08-19 20:11:29 +00:00
|
|
|
export const hmrDirtyComponents = new Set<ConcreteComponent>()
|
2020-06-12 18:53:48 +00:00
|
|
|
|
2019-12-16 22:57:34 +00:00
|
|
|
export interface HMRRuntime {
|
|
|
|
createRecord: typeof createRecord
|
|
|
|
rerender: typeof rerender
|
|
|
|
reload: typeof reload
|
|
|
|
}
|
|
|
|
|
2019-12-12 17:42:21 +00:00
|
|
|
// Expose the HMR runtime on the global object
|
|
|
|
// This makes it entirely tree-shakable without polluting the exports and makes
|
|
|
|
// it easier to be used in toolings like vue-loader
|
|
|
|
// Note: for a component to be eligible for HMR it also needs the __hmrId option
|
|
|
|
// to be set so that its instances can be registered / removed.
|
2020-11-27 15:34:45 +00:00
|
|
|
if (__DEV__) {
|
2019-12-12 17:42:21 +00:00
|
|
|
const globalObject: any =
|
|
|
|
typeof global !== 'undefined'
|
|
|
|
? global
|
|
|
|
: typeof self !== 'undefined'
|
|
|
|
? self
|
|
|
|
: typeof window !== 'undefined'
|
|
|
|
? window
|
|
|
|
: {}
|
|
|
|
|
|
|
|
globalObject.__VUE_HMR_RUNTIME__ = {
|
|
|
|
createRecord: tryWrap(createRecord),
|
|
|
|
rerender: tryWrap(rerender),
|
|
|
|
reload: tryWrap(reload)
|
2019-12-16 22:57:34 +00:00
|
|
|
} as HMRRuntime
|
2019-12-12 17:42:21 +00:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:52:16 +00:00
|
|
|
type HMRRecord = {
|
|
|
|
component: ComponentOptions
|
|
|
|
instances: Set<ComponentInternalInstance>
|
|
|
|
}
|
2019-12-12 17:42:21 +00:00
|
|
|
|
|
|
|
const map: Map<string, HMRRecord> = new Map()
|
|
|
|
|
|
|
|
export function registerHMR(instance: ComponentInternalInstance) {
|
2020-04-20 08:03:00 +00:00
|
|
|
const id = instance.type.__hmrId!
|
|
|
|
let record = map.get(id)
|
|
|
|
if (!record) {
|
2020-10-26 21:52:16 +00:00
|
|
|
createRecord(id, instance.type as ComponentOptions)
|
2020-04-20 08:03:00 +00:00
|
|
|
record = map.get(id)!
|
|
|
|
}
|
2020-10-26 21:52:16 +00:00
|
|
|
record.instances.add(instance)
|
2019-12-12 17:42:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function unregisterHMR(instance: ComponentInternalInstance) {
|
2020-10-26 21:52:16 +00:00
|
|
|
map.get(instance.type.__hmrId!)!.instances.delete(instance)
|
2019-12-12 17:42:21 +00:00
|
|
|
}
|
|
|
|
|
2020-10-26 21:52:16 +00:00
|
|
|
function createRecord(
|
|
|
|
id: string,
|
|
|
|
component: ComponentOptions | ClassComponent
|
|
|
|
): boolean {
|
2020-11-21 21:06:50 +00:00
|
|
|
if (!component) {
|
|
|
|
warn(
|
|
|
|
`HMR API usage is out of date.\n` +
|
|
|
|
`Please upgrade vue-loader/vite/rollup-plugin-vue or other relevant ` +
|
2020-12-31 10:15:57 +00:00
|
|
|
`dependency that handles Vue SFC compilation.`
|
2020-11-21 21:06:50 +00:00
|
|
|
)
|
|
|
|
component = {}
|
|
|
|
}
|
2019-12-12 17:42:21 +00:00
|
|
|
if (map.has(id)) {
|
2019-12-12 23:13:59 +00:00
|
|
|
return false
|
2019-12-12 17:42:21 +00:00
|
|
|
}
|
2020-10-26 21:52:16 +00:00
|
|
|
map.set(id, {
|
|
|
|
component: isClassComponent(component) ? component.__vccOpts : component,
|
|
|
|
instances: new Set()
|
|
|
|
})
|
2019-12-12 23:13:59 +00:00
|
|
|
return true
|
2019-12-12 17:42:21 +00:00
|
|
|
}
|
|
|
|
|
2020-05-01 14:37:40 +00:00
|
|
|
function rerender(id: string, newRender?: Function) {
|
2020-04-20 08:03:00 +00:00
|
|
|
const record = map.get(id)
|
|
|
|
if (!record) return
|
2020-10-26 21:52:16 +00:00
|
|
|
if (newRender) record.component.render = newRender
|
2019-12-22 17:25:04 +00:00
|
|
|
// Array.from creates a snapshot which avoids the set being mutated during
|
|
|
|
// updates
|
2020-10-26 21:52:16 +00:00
|
|
|
Array.from(record.instances).forEach(instance => {
|
2019-12-18 02:28:24 +00:00
|
|
|
if (newRender) {
|
2020-05-01 14:37:40 +00:00
|
|
|
instance.render = newRender as InternalRenderFunction
|
2019-12-18 02:28:24 +00:00
|
|
|
}
|
2019-12-12 17:42:21 +00:00
|
|
|
instance.renderCache = []
|
2019-12-12 23:13:59 +00:00
|
|
|
// this flag forces child components with slot content to update
|
2020-06-12 18:53:48 +00:00
|
|
|
isHmrUpdating = true
|
2019-12-12 17:42:21 +00:00
|
|
|
instance.update()
|
2020-06-12 18:53:48 +00:00
|
|
|
isHmrUpdating = false
|
2019-12-12 17:42:21 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-09-18 04:14:59 +00:00
|
|
|
function reload(id: string, newComp: ComponentOptions | ClassComponent) {
|
2020-04-20 08:03:00 +00:00
|
|
|
const record = map.get(id)
|
|
|
|
if (!record) return
|
2019-12-22 17:25:04 +00:00
|
|
|
// Array.from creates a snapshot which avoids the set being mutated during
|
|
|
|
// updates
|
2020-10-26 21:52:16 +00:00
|
|
|
const { component, instances } = record
|
|
|
|
|
|
|
|
if (!hmrDirtyComponents.has(component)) {
|
|
|
|
// 1. Update existing comp definition to match new one
|
|
|
|
newComp = isClassComponent(newComp) ? newComp.__vccOpts : newComp
|
|
|
|
extend(component, newComp)
|
|
|
|
for (const key in component) {
|
2021-05-06 12:21:54 +00:00
|
|
|
if (key !== '__file' && !(key in newComp)) {
|
2020-10-26 21:52:16 +00:00
|
|
|
delete (component as any)[key]
|
2020-05-03 17:52:09 +00:00
|
|
|
}
|
|
|
|
}
|
2020-10-26 21:52:16 +00:00
|
|
|
// 2. Mark component dirty. This forces the renderer to replace the component
|
|
|
|
// on patch.
|
|
|
|
hmrDirtyComponents.add(component)
|
|
|
|
// 3. Make sure to unmark the component after the reload.
|
|
|
|
queuePostFlushCb(() => {
|
|
|
|
hmrDirtyComponents.delete(component)
|
|
|
|
})
|
|
|
|
}
|
2020-05-03 17:52:09 +00:00
|
|
|
|
2020-10-26 21:52:16 +00:00
|
|
|
Array.from(instances).forEach(instance => {
|
2019-12-12 23:13:59 +00:00
|
|
|
if (instance.parent) {
|
2020-05-03 17:52:09 +00:00
|
|
|
// 4. Force the parent instance to re-render. This will cause all updated
|
2019-12-12 23:13:59 +00:00
|
|
|
// components to be unmounted and re-mounted. Queue the update so that we
|
|
|
|
// don't end up forcing the same parent to re-render multiple times.
|
|
|
|
queueJob(instance.parent.update)
|
2019-12-22 17:25:04 +00:00
|
|
|
} else if (instance.appContext.reload) {
|
|
|
|
// root instance mounted via createApp() has a reload method
|
|
|
|
instance.appContext.reload()
|
2019-12-12 23:13:59 +00:00
|
|
|
} else if (typeof window !== 'undefined') {
|
2019-12-22 17:25:04 +00:00
|
|
|
// root instance inside tree created via raw render(). Force reload.
|
2019-12-12 23:13:59 +00:00
|
|
|
window.location.reload()
|
|
|
|
} else {
|
|
|
|
console.warn(
|
|
|
|
'[HMR] Root or manually mounted instance modified. Full reload required.'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})
|
2019-12-12 17:42:21 +00:00
|
|
|
}
|
|
|
|
|
2019-12-12 23:13:59 +00:00
|
|
|
function tryWrap(fn: (id: string, arg: any) => any): Function {
|
2019-12-12 17:42:21 +00:00
|
|
|
return (id: string, arg: any) => {
|
|
|
|
try {
|
2019-12-12 23:13:59 +00:00
|
|
|
return fn(id, arg)
|
2019-12-12 17:42:21 +00:00
|
|
|
} catch (e) {
|
|
|
|
console.error(e)
|
|
|
|
console.warn(
|
2019-12-12 23:13:59 +00:00
|
|
|
`[HMR] Something went wrong during Vue component hot-reload. ` +
|
2019-12-12 17:42:21 +00:00
|
|
|
`Full reload required.`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|