Vue is essentially a function that completes a series of initialization procedures when it is called through the new keyword construction. Development through the Vue framework is basically done by passing different parameter options into Vue functions. Parameter options often need to be combined in two main cases:

1. The Vue function itself has some static properties that the developer passes in with the same name when instantiating. 2. When using Vue by inheritance, you need to merge attributes of the same name from the parent and child classes.

Vue function definition in the/SRC/core/instance/index, js.

function Vue (options) {
  if(process.env.NODE_ENV ! = ='production' &&
    !(this instanceof Vue)
  ) {
    warn('Vue is a constructor and should be called with the `new` keyword')}this._init(options)
}

initMixin(Vue)
Copy the code

Options are passed to the _init method on the instance prototype to initialize the Vue instantiation. InitMixin function role is the prototype of the object to add to the Vue instance _init method, initMixin function in the/SRC/core/instance/init. Js is defined. In the _init function, the passed option sets are merged.

// merge options
if (options && options._isComponent) {
    initInternalComponent(vm, options)
} else {
    vm.$options = mergeOptions(
      resolveConstructorOptions(vm.constructor),
      options || {},
      vm
    )
}
Copy the code

The _isComponent option is rarely passed in during development, so the else branch is used for instantiation. The mergeOptions function returns the merged options and assigns them to the $options property of the instance. MergeOptions function takes three parameters, including the first parameter is the generated instance constructor into the resolveConstructorOptions after processing the return value of a function.

export function resolveConstructorOptions (Ctor: Class<Component>) {
  let options = Ctor.options
  if (Ctor.super) {
    const superOptions = resolveConstructorOptions(Ctor.super)
    const cachedSuperOptions = Ctor.superOptions
    if(superOptions ! == cachedSuperOptions) {// super option changed,
      // need to resolve new options.
      Ctor.superOptions = superOptions
      // check if there are any late-modified/attached options (#4976)
      const modifiedOptions = resolveModifiedOptions(Ctor)
      // update base extend options
      if (modifiedOptions) {
        extend(Ctor.extendOptions, modifiedOptions)
      }
      options = Ctor.options = mergeOptions(superOptions, Ctor.extendOptions)
      if (options.name) {
        options.components[options.name] = Ctor
      }
    }
  }
  return options
}
Copy the code

ResolveConstructorOptions function parameters for instance constructor, no parent, in the constructor simply return to the constructor of the options properties. Otherwise, the if branch is used to merge the options property of the constructor and its parent. If the parent class of the constructor still exists, the method is recursively called and finally returns the unique options property. In the study of instantiation merge options, for the convenience of writing, the value returned by this function is uniformly called the parent option set of option merge, and the option set passed in the instantiation is called the child option set.

Vue constructor static property Options

When merging options, the first argument passed without inheritance is the static property options on the Vue constructor. What does this static property contain? To figure this out, first figure out what happens when you run the NPM run dev command to generate the /dist/vue.js file. In the package.json file, the scripts object has:

"dev": "rollup -w -c scripts/config.js --environment TARGET:web-full-dev".Copy the code

When packaging with rollup, follow the configuration in scripts/config.js and use web-full-dev as the value of the environment variable TARGET.

// Runtime+compiler development build (Browser)
'web-full-dev': {
    entry: resolve('web/entry-runtime-with-compiler.js'),
    dest: resolve('dist/vue.js'),
    format: 'umd'.env: 'development'.alias: { he: './entity-decoder' },
    banner
},
Copy the code

The above file paths are aliased in the scripts/alias.js file. If you run the NPM run dev command, the entry file is SRC /platforms/web/entry-runtime-with-compiler.js, vue.js files that comply with umD specifications will be generated. Follow the entry file’s reference to the Vue function to find the file where the Vue constructor resides. As shown below:

Vue constructor defined in/SRC/core/instance/index in js. In this js file, some properties and methods are mounted to vue. prototype through various mixins. Then in/SRC /core/index.js, add static properties and methods to the Vue constructor via the initGlobalAPI function.

import Vue from './instance/index'
import { initGlobalAPI } from './global-api/index'
import { isServerRendering } from 'core/util/env'
import { FunctionalRenderContext } from 'core/vdom/create-functional-component'

initGlobalAPI(Vue)
Copy the code

In the initGlobalAPI function there is a definition to add the options property to the Vue constructor.

Vue.options = Object.create(null)
ASSET_TYPES.forEach(type= > {
    Vue.options[type + 's'] = Object.create(null)})// this is used to identify the "base" constructor to extend all plain-object
// components with in Weex's multi-instance scenarios.
Vue.options._base = Vue

extend(Vue.options.components, builtInComponents)
Copy the code

After this code is processed, vue. options looks like this:

Vue.options = {
	components: {
		KeepAlive
	},
	directives: Object.create(null),
	filters: Object.create(null),
  _base: Vue
}
Copy the code

In/SRC/platforms/web/runtime/index, js, by the following code to the Vue. Options to add platform instruction and built-in component properties.

import platformDirectives from './directives/index'
import platformComponents from './components/index'

// install platform runtime directives & components
extend(Vue.options.directives, platformDirectives)
extend(Vue.options.components, platformComponents)
Copy the code

The final vue. options property contents are as follows:

Vue.options = {
    components: {
        KeepAlive,
        Transition,
        TransitionGroup
    },
    directives: {
        model,
        show
     },
    filters: Object.create(null),
    _base: Vue
}
Copy the code

The mergeOptions function is used to mergeOptions

Merge option function mergeOptions in/SRC/core/util/options. The js is defined.

export function mergeOptions (parent: Object, child: Object, vm? : Component) :Object {
  if(process.env.NODE_ENV ! = ='production') {
    checkComponents(child)
  }

  if (typeof child === 'function') {
    child = child.options
  }

  normalizeProps(child, vm)
  normalizeInject(child, vm)
  normalizeDirectives(child)

  if(! child._base) {if (child.extends) {
      parent = mergeOptions(parent, child.extends, vm)
    }
    if (child.mixins) {
      for (let i = 0, l = child.mixins.length; i < l; i++) {
        parent = mergeOptions(parent, child.mixins[i], vm)
      }
    }
  }

  const options = {}
  let key
  for (key in parent) {
    mergeField(key)
  }
  for (key in child) {
    if(! hasOwn(parent, key)) { mergeField(key) } }function mergeField (key) {
    const strat = strats[key] || defaultStrat
    options[key] = strat(parent[key], child[key], vm, key)
  }
  return options
}
Copy the code

1. Component naming rules

When merging options, the declared component name is first checked for compliance in a non-production environment:

if(process.env.NODE_ENV ! = ='production') {
  checkComponents(child)
}
Copy the code

The checkComponents function checks the validity of names using the validateComponentName function for each attribute in the components attribute of a suboption set.

function checkComponents (options: Object) {
  for (const key in options.components) {
    validateComponentName(key)
  }
}
Copy the code

The validateComponentName function defines the rules for naming components:

export function validateComponentName (name: string) {
  if (!/^[a-zA-Z][\w-]*$/.test(name)) {
    warn(
      'Invalid component name: "' + name + '". Component names ' +
      'can only contain alphanumeric characters and the hyphen, ' +
      'and must start with a letter.')}if (isBuiltInTag(name) || config.isReservedTag(name)) {
    warn(
      'Do not use built-in or reserved HTML elements as component ' +
      'id: ' + name
    )
  }
}
Copy the code

As can be seen from the above code, there are two valid naming rules:

1. The component name can contain letters, digits, underscores (_), and hyphens (-) and must start with a letter. 2. Component names cannot be slot and Component built-in Vue tags; Cannot be HTML built-in tags; Partial SVG tags cannot be used.

2. Normalize options

The options passed into Vue often come in many forms, which is convenient for developers. When merging options within Vue, the various forms are standardized and eventually converted into one form to be merged.

normalizeProps(child, vm)
normalizeInject(child, vm)
normalizeDirectives(child)
Copy the code

The three function calls above are standardized props, Inject, and directives, respectively.

(A) Standardization of props options

The props option comes in two forms: arrays, objects, and ultimately, objects. If the props option is an array, the values in the array must all be strings. If the string has a hyphen, it is changed to a hump name. Such as:

props: ['propOne'.'prop-two']
Copy the code

The props will be normalized to:

props: {
  propOne: {type: null
  },
  propTwo: {type: null}}Copy the code

If the props option is an object, its properties take two forms: string, object. The hyphenated attribute name is converted to a hump name. If the property is an object, it does not change; If the property is a string it is converted to an object and the value of the property becomes the type property of the new object. Such as:

props: {
  propOne: Number."prop-two": Object.propThree: {
    type: String.default: ' '}}Copy the code

The props will be normalized to:

props: {
  propOne: {
    type: Number
  },
  propTwo: {
    type: Object
  },
  propThree: {
    type: String.default: ' '}}Copy the code

If the props object is an object, there are four valid props:

1. Type: Basic type checking. Required: An attribute that must be passed in. 3. Default: indicates the default value. 4. Validator: Custom validation function.

(2) Standardization of inject options

The Inject option comes in two forms: arrays, objects, and eventually objects. If the Inject option is an array, it is converted to an object whose property name is the value of the array. The value of the property is the object that only has the FROM property. The value of the FROM property is the same as the corresponding value of the array. Such as:

inject: ['test']
Copy the code

The Inject will be normalized into:

inject: {
  test: {
    from: 'test'}}Copy the code

If the Inject option is an object, its attribute takes three forms: string, symbol, and object. If it is an object, add the attribute from, whose value is equal to the attribute name. If it is a string or symbol, it is converted to an object that has the attribute FROM, whose value is equal to the string or symbol. Such as:

inject: {
  a: 'value1'.b: {
    default: 'value2'}}Copy the code

The Inject will be normalized into:

inject: {
  a: {
    from: 'value1'
  },
  b: {
    from: 'b'.default: 'value2'}}Copy the code
Standards of directives

The custom directive option Directives accepts only the object type. Directives A general concrete custom directive is an object. The directives option is fairly uniform, so why is there this standardized step? That’s because the properties of a specific custom instruction object are typically the hook functions. But Vue provides a shorthand: you can define a custom directive as a function, rather than an object, when bind and update trigger the same behavior, regardless of the other hooks. When Vue internally incorporates the directives option, it abbreviates this function to the form of an object. Caching As follows:

Directive: {'color':function (el, binding) {
    el.style.backgroundColor = binding.value
  })
}
Copy the code

The directive will be formalized as:

Directive: {'color': {bind:function (el, binding) {
      el.style.backgroundColor = binding.value
    }),
    update: function (el, binding) {
      el.style.backgroundColor = binding.value
    })
  }
}
Copy the code

3. Handling of extends and mixins options

The mixins option accepts an array of mixins. These mixin instance objects can contain options just like normal instance objects. As follows:

var mixin = {
  created: function () { console.log(1)}}var vm = new Vue({
  created: function () { console.log(2)},mixins: [mixin]
})
/ / = > 1
/ / = > 2
Copy the code

The extends option allows you to declare an extension to another component, which can be a simple option object or constructor. As follows:

var CompA = { ... }

// Inherit CompA when 'vue.extend' is not called
var CompB = {
  extends: CompA,
  ...
}
Copy the code

The extends or mixins option is first handled internally by a recursive call to the mergeOptions function, which merges the objects from the extends object or mixins array as a collection of child options with the parent collection. This is the basis for the merge rule when the content of the extends and mixins options conflicts with the other options in the list.

4. Merge options using policy patterns

The number of options is large, and the merging rules are different. Vue uses the policy pattern internally to merge options. The various policy methods are implemented outside of the mergeOptions function, and the environment object is the Strats object. Strats objects are derived from the optionMergeStrategies object in the/SRC /core/config.js file by adding a series of policy functions. The environment object accepts the request to decide which policy to delegate to handle. This is why users can customize option merge rules by globally configuring optionMergeStrategies.

Option combination strategy

The attributes that the environment object has on Strats and their corresponding functions are shown below:

1. Merge strategy for option EL, propsData, and property objects not included in strats objects

The options el, propsData, and options not shown in the figure are all merged using the default policy function defaultStrat.

const defaultStrat = function (parentVal: any, childVal: any) :any {
  return childVal === undefined
    ? parentVal
    : childVal
}
Copy the code

The default policy is simple: if there is an option in the suboption set, the value of the suboption is used directly. Otherwise, use the value of the parent option.

2. Select data and provide

The data and provide policy functions are both mergeDataOrFn, but when the provide option is merged, three parameters are passed to the mergeDataOrFn function: parent option, child option, and instance. The data option is merged in two cases: when the child component option is handled through vue.extends () and when it is instantiated normally. In the former case, there is no instance VM and two arguments are passed to the mergeDataOrFn function: parent and child; The latter case is the same as the argument passed in with the option provide. The mergeDataOrFn function code is shown below, and only goes to the if branch if the data option is merged and the child component option is handled through vue.extends (). When dealing with normal instantiation options data and provide, the else branch is used.

export function mergeDataOrFn (parentVal: any,childVal: any,vm? : Component): ?Function 
{
  if(! vm) {if(! childVal) {return parentVal
    }
    if(! parentVal) {return childVal
    }
    return function mergedDataFn () {
      return mergeData(
        typeof childVal === 'function' ? childVal.call(this.this) : childVal,
        typeof parentVal === 'function' ? parentVal.call(this.this) : parentVal
      )
    }
  } else {
    return function mergedInstanceDataFn () {
      const instanceData = typeof childVal === 'function'
        ? childVal.call(vm, vm)
        : childVal
      const defaultData = typeof parentVal === 'function'
        ? parentVal.call(vm, vm)
        : parentVal
      if (instanceData) {
        return mergeData(instanceData, defaultData)
      } else {
        return defaultData
      }
    }
  }
}
Copy the code

In the case that the instance VM does not exist, there are three cases:

1. If the child option does not exist, return the parent option. If the parent option does not exist, return the child option. 3. If both parent and child options exist, return mergedDataFn.

The mergedDataFn function extracts the return value of the parent and child option functions, passes the pure object to mergeData, and finally returns the return value of mergeData. If neither parent option exists, it will not go to this function, so it will not be considered. Why do we say that parent and child options in an if branch are functions? Because you can only go through vue.extends () when dealing with the child component’s data option. When a component is defined, data must be declared as a function that returns a pure object, preventing multiple component instances from sharing a single data object. When defining a component, the data option is a pure object, and Vue will have error warnings in non-production environments. In the else branch, the return function mergedInstanceDataFn, in which the return value of the parent and child option functions is extracted if the child option exists, passes the pure object to mergeData. Otherwise, the parent option is returned as a pure object. In this scenario, the function of mergeData is to add the attribute that exists in the parent option object but does not exist in the child option object to the child option object through the set method and change it into a responsive data attribute. After analyzing the various cases, it is found that the options data and provide policy functions are higher-order functions, and the return value is a function that returns the merged object. Why is that? This reason, as mentioned earlier, is to ensure that each component instance has a unique copy of its data, preventing component instances from sharing the same data object. Why is the result of the data or provide option merge processing a function that is not executed during the merge phase, but at initialization? In/SRC/core/instance/init. Initialized js, have the following code:

initInjections(vm)
initState(vm)
initProvide(vm) 
Copy the code

The initState function has the following code:

if (opts.props) initProps(vm, opts.props)
if (opts.methods) initMethods(vm, opts.methods)
if (opts.data) {
  initData(vm)
} else {
  observe(vm._data = {}, true /* asRootData */)}Copy the code

As can be seen from the above code: Data and provide are initialized after Inject and props. The return function of the merge function is executed during initialization, which initializes the values of data and provide using the values of Inject and props.

Merge strategy for lifecycle hook options

The lifecycle hook options are merged using the mergeHook function.

function mergeHook (parentVal: ? Array
       
        , childVal: ? Function | ? Array
        
       ): ?Array<Function> {
  const res = childVal
    ? parentVal
      ? parentVal.concat(childVal)
      : Array.isArray(childVal)
        ? childVal
        : [childVal]
    : parentVal
  return res
    ? dedupeHooks(res)
    : res
}
Copy the code

The Vue API documentation states that the lifecycle hook option can only be of function type, as shown in this source code, developers can pass in the lifecycle option of function array type. Because you can combine functions in an array, passing in an array of functions is not very useful. Another interesting point is that if the parent option exists, it must be an array. Lifecycle options can be arrays, but developers usually pass in functions, so why does the parent have to be an array? This is because there are two cases where the lifecycle parent option exists: vue.extends () and Mixins. As mentioned above in the handling of options extends, mixins, in both cases, the mergeOptions function is called recursively as a suboption to merge the options. If a lifecycle parent exists, it must be an array of functions. The mergeHook function returns a value, if present, that is passed to the dedupeHooks function for processing. The purpose is to remove duplicate values from the option merge array.

function dedupeHooks (hooks) {
  const res = []
  for (let i = 0; i < hooks.length; i++) {
    if (res.indexOf(hooks[i]) === - 1) {
      res.push(hooks[i])
    }
  }
  return res
}
Copy the code

The lifecycle hook array is executed sequentially, so the hook functions in the parent option are executed first, followed by the hook functions in the child option.

4. Merge policies for resources (Components, directives, filters)

Component components, directive directives, and filters are called resources because they can be provided as third-party applications. The resource options are merged using the mergeAssets function and the logic is simple.

function mergeAssets (parentVal: ? Object, childVal: ? Object, vm? : Component, key: string) :Object {
  const res = Object.create(parentVal || null)
  if(childVal) { process.env.NODE_ENV ! = ='production' && assertObjectType(key, childVal, vm)
    return extend(res, childVal)
  } else {
    return res
  }
}
Copy the code

Define a null object after merging. If the parent option exists, the parent option is the prototype, otherwise there is no prototype. If the suboption is a pure object, the attributes on the suboption are copied to the merged option object. The Vue. Options property looks like this:

Vue.options = {
    components: {
        KeepAlive,
        Transition,
        TransitionGroup
    },
    directives: {
        model,
        show
     },
    filters: Object.create(null),
    _base: Vue
}
Copy the code

KeepAlive, Transition, and TransitionGroup are built-in components. Model and show are built-in commands. They can be used without registration.

5. Select the merger strategy of Watch

The watch option is an object, but the attributes of an object can take many forms: strings, functions, objects, and arrays.

// work around Firefox's Object.prototype.watch...
if (parentVal === nativeWatch) parentVal = undefined
if (childVal === nativeWatch) childVal = undefined
/* istanbul ignore if */
if(! childVal)return Object.create(parentVal || null)
if(process.env.NODE_ENV ! = ='production') {
  assertObjectType(key, childVal, vm)
}
if(! parentVal)return childVal
const ret = {}
extend(ret, parentVal)
for (const key in childVal) {
  let parent = ret[key]
  const child = childVal[key]
  if (parent && !Array.isArray(parent)) {
    parent = [parent]
  }
  ret[key] = parent
    ? parent.concat(child)
    : Array.isArray(child) ? child : [child]
}
return ret
Copy the code

Because the Object prototype Object of Firefox browser has the watch attribute, it is necessary to check whether the options set has the watch attribute added by the developer before merging. If not, the merging will not be processed. If the child option does not exist, an empty object modeled after the parent option is returned. If the parent does not exist, check that the child is a pure object and return the child. If both parent and child options exist, the attributes of the parent option are first copied to the merged object, and then the attributes of the child option are examined. Properties that are on the child option but not the parent option are arrays that are added directly to the merged object. If it is not an array, it is populated into a new array, which is added to the merge object. Attributes that exist on both parent and child options, and then add the corresponding attributes on the child options to the array.

6. Options props, methods, Inject, computed merge strategy

The options props, Methods, Inject, and computed use the same merge strategy. Methods and computed accept only object forms, while props and Inject are pure object forms after the previous standardization.

if(childVal && process.env.NODE_ENV ! = ='production') {
  assertObjectType(key, childVal, vm)
}
if(! parentVal)return childVal
const ret = Object.create(null)
extend(ret, parentVal)
if (childVal) extend(ret, childVal)
return ret
Copy the code

First check if the suboption is pure. If it is not, an error is reported in a non-production environment. If the parent option does not exist, the child option is returned directly. If the parent option exists, create an empty object with no prototype as the merge option object, and copy the attributes from the parent option to the merge option object. If a suboption exists, all attributes on the suboption object are copied to the merged object, so the value of that attribute on the suboption is taken if the parent option has the same attribute. Finally, the merge option object is returned.

7. Summary of option combination strategy

1. El, propsData, and options merged using the default strategy: choose the value of the child if there is one, or the value of the parent if there is one. 2, data, provide: returns a function that returns the merged object. Based on the child option object, if there is an attribute that does not exist on the child option but exists on the parent option, the attribute is converted into a responsive attribute and added to the child option object. 3. Lifecycle hook options: Merge into an array of functions, with parent options placed before child options, and execute in sequence. Components, directives, filters: Defines an empty merge object without a stereotype, and copies the properties on the suboptions to the merge object if they exist; If the parent option exists, the parent option object is modeled. 5. Watch: if no child option exists, return an empty object modeled after the parent option; Parent option does not exist, return child option; If both parent and child options exist, it is similar to the lifecycle merge strategy. If the parent option also exists, the parent option is pushed into the array. 6. Options props, methods, Inject, computed: Add attributes on the parent option to an empty object with no prototype, and take the value of the child option if the parent option has the same attributes. Extends and mixins of child options: The values of the two items are combined with the parent option as a child option, and the combination rules are combined according to the above rules. Finally, the items are combined with the properties of the same name of the child option according to the above rules.

Four,

Before merging options, standardize the inject, props, and directives of the options. The extends and mixins in the child option collection are then merged with the parent option recursively by calling the merge function as the child option. Finally, the options are combined using the policy pattern. If you want to reprint, please indicate the source: www.cnblogs.com/lidengfeng/…