Designed by 77
加载资源 ......
感谢 ♥ 作者
先不感谢了

vue2.x学习笔记(四)

接着前面的内容:https://www.cnblogs.com/yanggb/p/12563162.html

模板语法

vue使用了基于html的模板语法,允许开发者声明式地将dom绑定到底层vue实例的数据。所有vue的模板都是合法的html,因此能被遵循规范的浏览器或html解析器解析。在底层的实现上,vue将模板编译成虚拟dom渲染函数,并结合响应系统,智能地计算出最少需要重新渲染多少组件,来将dom操作的次数减到最少。

插值-文本

数据绑定最常见的形式就是使用mustache语法(双大括号)的文本插值。

<span>i love {{ someone }}.</span>

mustache标签(双大括号)中的内容将会被替代为对应数据对象上someone的值,比如如果【someone】的值是【yanggb】,那么页面上渲染的内容就会是【i love yanggb.】。无论任何时候,只要绑定的数据对象上someone的属性值发生了改变,插值处的内容也会被同步更新。另外,如果你想要插值中的数据只能被替换一次的话,可以使用【v-once】指令,这个指令的作用是限制插值的内容只会被更新一次,之后无论多少次改变someone的值,插值处的内容都一直是【i love yanggb forever.】。

<span v-once>i love {{ someone }} forever.</span>

使用【v-once】指令的时候要格外注意不要影响到该节点上的其他数据绑定。

插值-原始html

双大括号会将数据解释为普通文本,而非html代码。为了输出真正的html,你会需要使用【v-html】指令。

<p>使用mustache语法: {{ rawHtml }}</p>
<p>使用v-html指令: <span v-html="rawHtml"></span></p>

这个时候,如果rawHtml的值是'<span>我是原始的html代码</span>'的话,使用mustache语法会将此字符串原样输出,而使用【v-html】指令,则会被解析成html渲染。要注意的是,不应该选择使用【v-html】指令来复合局部模板,因为vue不是基于字符串的模板引擎,这样做可能会造成意料之外的问题。再说了,vue还提供了组件这样的特性,组件在用户界面(UI)中更适合作为可重用和可组合的基本单位。

你的站点上动态渲染的任意HTML可能会非常危险,因为它很容易导致XSS 攻击。请只对可信内容使用HTML插值,绝不要对用户提供的内容使用插值。

官方文档上特别注明了使用【v-html】指令进行原始html插值是一项可能造成危险的行为,因此建议在开发中禁止使用这个语法,除非有非用不可的理由,将危险扼杀在摇篮里,只要知道有这个语法就可以了。

插值-attribute(属性)

因为mustache语法并不能作用在html的attribute上,因此vue提供了【v-bind】指令来满足这一场景。

<div v-bind:id="dynamicId"></div>

而对于布尔attribute来说,它们只要存在就意味着值为true,比如disabled属性,因此使用【v-bind】指令的时候会与常规的使用javascript操作原生html的方式有所不同。

<button v-bind:disabled="isButtonDisabled">Button</button>

在上面这个例子中,如果isButtonDisabled的值是null、undefined或false的话,disabled这个attribute是不会被包含在渲染出来的<button>元素中的。

使用javascript表达式

在前面的例子中,在模板中都是绑定的简单的属性键值。而实际上,对于所有的数据绑定,vue都提供了完全的javascript表达式支持。

{{ number + 1 }}

{{ ok ? 'yes' : 'no' }}

{{ message.split('').reverse().join('') }}

<div v-bind:id="'list-' + id"></div>

这些表达式会在所属vue实例的数据作用域下作为javascript被解析。但是有个限制是,每个绑定都只能包含单个表达式,非单个表达式的话,数据绑定会失效,可能导致表达式被原样输出,甚至导致直接报错。

<!-- 这是语句,不是表达式 -->
{{ var a = 1 }}

<!-- 流控制也不会生效,请使用三元表达式 -->
{{ if (ok) { return message } }}

此外,模板表达式都被放在沙盒中,只能访问全局变量的一个白名单,如Math和Date,开发者不应该在模板表达式中试图访问用户定义的全局变量。

/* not type checking this file because flow doesn't play well with Proxy */

import config from 'core/config'
import { warn, makeMap, isNative } from '../util/index'

let initProxy

if (process.env.NODE_ENV !== 'production') {
  const allowedGlobals = makeMap(
    'Infinity,undefined,NaN,isFinite,isNaN,' +
    'parseFloat,parseInt,decodeURI,decodeURIComponent,encodeURI,encodeURIComponent,' +
    'Math,Number,Date,Array,Object,Boolean,String,RegExp,Map,Set,JSON,Intl,' +
    'require' // for Webpack/Browserify
  )

  const warnNonPresent = (target, key) => {
    warn(
      `Property or method "${key}" is not defined on the instance but ` +
      'referenced during render. Make sure that this property is reactive, ' +
      'either in the data option, or for class-based components, by ' +
      'initializing the property. ' +
      'See: https://vuejs.org/v2/guide/reactivity.html#Declaring-Reactive-Properties.',
      target
    )
  }

  const warnReservedPrefix = (target, key) => {
    warn(
      `Property "${key}" must be accessed with "$data.${key}" because ` +
      'properties starting with "$" or "_" are not proxied in the Vue instance to ' +
      'prevent conflicts with Vue internals' +
      'See: https://vuejs.org/v2/api/#data',
      target
    )
  }

  const hasProxy =
    typeof Proxy !== 'undefined' && isNative(Proxy)

  if (hasProxy) {
    const isBuiltInModifier = makeMap('stop,prevent,self,ctrl,shift,alt,meta,exact')
    config.keyCodes = new Proxy(config.keyCodes, {
      set (target, key, value) {
        if (isBuiltInModifier(key)) {
          warn(`Avoid overwriting built-in modifier in config.keyCodes: .${key}`)
          return false
        } else {
          target[key] = value
          return true
        }
      }
    })
  }

  const hasHandler = {
    has (target, key) {
      const has = key in target
      const isAllowed = allowedGlobals(key) ||
        (typeof key === 'string' && key.charAt(0) === '_' && !(key in target.$data))
      if (!has && !isAllowed) {
        if (key in target.$data) warnReservedPrefix(target, key)
        else warnNonPresent(target, key)
      }
      return has || !isAllowed
    }
  }

  const getHandler = {
    get (target, key) {
      if (typeof key === 'string' && !(key in target)) {
        if (key in target.$data) warnReservedPrefix(target, key)
        else warnNonPresent(target, key)
      }
      return target[key]
    }
  }

  initProxy = function initProxy (vm) {
    if (hasProxy) {
      // determine which proxy handler to use
      const options = vm.$options
      const handlers = options.render && options.render._withStripped
        ? getHandler
        : hasHandler
      vm._renderProxy = new Proxy(vm, handlers)
    } else {
      vm._renderProxy = vm
    }
  }
}

export { initProxy }

上面这段代码是vue中定义全局变量的访问白名单,因为用户自定义的全局变量不在白名单内,在模板表达式中访问将会出现问题。

指令

指令(directives)是带有【v-】前缀的特殊attribute。指令attribute的值预期是单个javascript表达式(v-for是个例外)。指令的职责是,当表达式的值发生改变的时候,就将其产生的连带影响,响应式地作用到dom中。

参数

一些指令是能够接收一个参数的,这个参数跟在指令的后面,以冒号表示。

比如,可以使用【v-bind】指令响应式地更新html的attribute。

<a v-bind:href="url">...</a>

在这里,href是参数,作用是告知【v-bind】指令将该元素的href属性与表达式url的值进行绑定。

又比如,可以使用【v-on】指令监听dom事件。

<a v-on:click="doSomething">...</a>

在这个例子中,这样写就给这个<a>元素绑定了click点击事件。

动态参数

从2.6.0开始,可以用方括号括起来的javascript表达式作为一个指令的参数。

<!-- 注意,参数表达式的写法存在一些约束,如之后的“对动态参数表达式的约束”章节所述。-->
<a v-bind:[attributeName]="url"> ... </a>

这里的attributeName会被作为一个javascript表达式进行动态求值,求得的值将会作为最终的参数来使用。例如,如果你的vue实例有一个data属性attributeName,其值为href的话,那么这个绑定将等价于【v-bind:href】。

同样的,你可以使用动态参数为一个动态的事件名绑定处理函数。

<a v-on:[eventName]="doSomething"> ... </a>

在上面的这个例子中,当eventName的值为【focus】的时候,这个绑定将等价于【v-on:focus】。

对动态参数值的约束

动态参数预期会求出一个字符串,异常情况下值为null,这个特殊的null值可以被显性地用于移除绑定,任何其他非字符串类型的值都将会触发一个警告。

<a v-on:[eventName]="doSomething"> ... </a>

比如在上面这个例子中,如果eventName的值为null,那么这个<a>标签就不会被绑定任何dom事件了。

对动态参数表达式的约束

动态参数表达式有一些语法约束,因为某些字符,比如空格和引号,放在html的属性名里是无效的。

<!-- 这会触发一个编译警告 -->
<a v-bind:['foo' + bar]="value"> ... </a>

变通的办法是使用没有空格或者引号的表达式,或使用计算属性替代这种复杂的表达式。

另外,在dom中使用模板的时候(直接在一个html文件中编写模板),还需要避免使用大写字符来命名键名,因为浏览器会把attribute名全部强制转换为小写。

<!-- 在 DOM 中使用模板时这段代码会被转换为 `v-bind:[someattr]`。
除非在实例中有一个名为“someattr”的 property,否则代码不会工作。-->
<a v-bind:[someAttr]="value"> ... </a>

动态参数的使用场景在现有的开发场景中并没有使用到,大概只有特殊的场景才会用得上,有所了解就好了。

修饰符

修饰符(modifier)是以半角句号【.】指明的特殊后缀,用于指出一个指令应该以特殊的方式绑定。例如【.prevent】修饰符的作用是告诉【v-on】指令对于触发的事件调用【event.preventDefallut()】方法。

<form v-on:submit.prevent="onSubmit">...</form>

这个语法在一些特定的场景中十分有用,后面会详细了解。

缩写

【v-】前缀作为一种视觉提示,可以用于识别模板中vue特定的attribute。虽然在使用vue为现有的标签添加动态行为(dynamic behavior)的时候【v-】前缀会很有帮助,但是对于一些频繁使用到的指令来说,反复编写相同的代码就会让开发者觉得厌烦。同时,在构建由vue管理所有模板的单页面应用程序(SPA,single page application)的时候,【v-】前缀也变得没有那么重要了。因此,vue为【v-bind】和【v-on】这两个最常用的指令提供了特定的缩写,方便开发者编写vue应用程序。

【v-bind】指令的缩写

<!-- 完整语法 -->
<a v-bind:href="url">...</a>

<!-- 缩写 -->
<a :href="url">...</a>

<!-- 动态参数的缩写 (2.6.0+) -->
<a :[key]="url"> ... </a>

【v-on】指令的缩写

<!-- 完整语法 -->
<a v-on:click="doSomething">...</a>

<!-- 缩写 -->
<a @click="doSomething">...</a>

<!-- 动态参数的缩写 (2.6.0+) -->
<a @[event]="doSomething"> ... </a>

虽然它们看起来可能和普通的html略有不同,但是【:】和【@】对于attribute名来说都是合法字符,在所有支持vue的浏览器中都能被正确解析,而且它们并不会出现在最终渲染的标记中。缩写语法是完全可选的,你在编写vue应用的时候,完全可以选择完整的语法或使用缩写的语法,但是随着你深入的使用缩写语法和了解它们的作用,你会像我一样喜欢上缩写的语法。

 

"我还是很喜欢你,像日月轮回交替,不问朝夕。"

posted @ 2020-04-04 10:10  yanggb  阅读(615)  评论(0编辑  收藏  举报