Vue重新起航(四)

事件处理

监听事件:用v-on 来监听DOM事件,并在触发时执行一些JS语句
然而有很多操作逻辑很复杂,就不能把大量语句写在v-on指令中了,需要在v-on后面接一个方法名

直接接js操作
<div id="demo">
    <p>已经按了{{counter}}次</p>
    <button v-on:click="counter++">点击</button>
  </div>

接方法名
  <div id="demo2">
    <button v-on:click="great">Greet</button>
  </div>

除了直接绑定到一个方法,也可以在内联 JavaScript 语句中调用方法:

<button v-on:click="say('hi')">Say hi</button>


事件修饰符

在事件处理程序中调用 event.preventDefault() 或 event.stopPropagation() 是非常常见的需求。尽管我们可以在方法中轻松实现这点,但更好的方式是:方法只有纯粹的数据逻辑,而不是去处理 DOM 事件细节。

.stop
.prevent
.capture
.self
.once
.passive

<!-- 阻止单击事件继续传播 -->
<a v-on:click.stop="doThis"></a>
<!-- 提交事件不再重载页面 -->
<form v-on:submit.prevent="onSubmit"></form>
<!-- 修饰符可以串联 -->
<a v-on:click.stop.prevent="doThat"></a>
<!-- 只有修饰符 -->
<form v-on:submit.prevent></form>

<!-- 点击事件将只会触发一次 -->
<a v-on:click.once="doThis"></a>

添加事件监听器时使用事件捕获模式
即元素自身触发的事件先在此处理,然后才交由内部元素进行处理
<div v-on:click.capture="doThis">...</div>

只当在 event.target 是当前元素自身时触发处理函数
即事件不是从内部元素触发的
<div v-on:click.self="doThat">...</div>

按键修饰符

按键别名
.enter
.tab
.delete (捕获“删除”和“退格”键)
.esc
.space
.up
.down
.left
.right


使用 v-on 有几个好处:

扫一眼 HTML 模板便能轻松定位在 JavaScript 代码里对应的方法。

因为你无须在 JavaScript 里手动绑定事件,你的 ViewModel 代码可以是非常纯粹的逻辑,和 DOM 完全解耦,更易于测试。

当一个 ViewModel 被销毁时,所有的事件处理器都会自动被删除。你无须担心如何清理它们。




表单输入绑定

用 v-model 指令在表单 <input>、<textarea> 及 <select> 元素上创建双向数据绑定。其实是语法糖!
表单输入绑定用的时候再看


组件基础

简单的组件案例

<div id="demo">
      <item></item>>
    </div>


    <script>
      Vue.component('item',{
        data:function(){
          return {
            count:0
          }
        },
        template:`<button v-on:click="count++">你已经点了{{count}}次</button>`
      })

      new Vue({
        el:'#demo'
      })
    </script>

因为组件是可以复用的Vue实例,所以与new Vue接收相同的选项,例如data、computed、watch、methods以及生命周期钩子等,仅有的例外是像el只有根实例才有

组件的复用

<div id="demo">
    <item></item>>
    <item></item>>
    <item></item>>
</div>
image.png

点击按钮时,每个组件都会维护自己的count,因为每用一次组件就会有一个新的实例被创建


为什么组件的data必须是一个函数?

data: function () {
return {
count: 0
}
}
因为这样返回的是不同的对象,每个实例可以维护一份被返回对象的独立的拷贝,不这样的话,他们其实还是指向的同一个对象,牵一发动全身


通过prop向子组件传递数据

Vue.component('blog-post', {
  props: ['title'],
  template: '<h3>{{ title }}</h3>'
})

一个组件默认可以拥有任意数量的 prop,任何值都可以传递给任何 prop。在上述模板中,你会发现我们能够在组件实例中访问这个值,就像访问 data 中的值一样。

   <uesprops title="hehe"></uesprops>
    <uesprops title="xixi"></uesprops>
    <uesprops title="haha"></uesprops>

 Vue.component('uesprops', {
      props: ['title'],
      template: `<h2>{{title}}</h2>`
    })
image.png

好像组件命名不允许有大写字母

v-bind 来动态传递 prop。一开始不清楚要渲染的具体内容,比如从一个 API 获取博文列表的时候,是非常有用的

  
  <li is="uesprops"
       v-for="item in posts"
       v-bind:title="item.title"   
   >
  </li>

 Vue.component('uesprops', {
      props: ['title'],
      template: `<h2>{{title}}</h2>`
    })



    new Vue({
      el: '#demo',
      data: {
        posts: [{
            id: 1,
            title: 'My journey with Vue'
          },
          {
            id: 2,
            title: 'Blogging with Vue'
          },
          {
            id: 3,
            title: 'Why Vue is so fun'
          }
          ]
        }
    })

单个根元素

当你遇到这个问题的时候
every component must have a single root element (每个组件必须只有一个根元素)。你可以将模板的内容包裹在一个父元素内,来修复这个问题

<div class="blog-post">
<h3>{{ post.title }}</h3>
<div v-html="post.content"></div>
</div>


有的时候组件愈来愈复杂,为每个相关的信息定义一个 prop 会变得很麻烦:像这样

<blog-post
  v-for="post in posts"
  v-bind:key="post.id"
  v-bind:title="post.title"
  v-bind:content="post.content"
  v-bind:publishedAt="post.publishedAt"
  v-bind:comments="post.comments"
></blog-post>

需要重构一下这个组件,props直接接收一个对象

<blog-post
  v-for="post in posts"
  v-bind:key="post.id"
  v-bind:post="post"
></blog-post>
Vue.component('blog-post', {
  props: ['post'],
  template: `
    <div class="blog-post">
      <h3>{{ post.title }}</h3>
      <div v-html="post.content"></div>
    </div>
  `
})

通过事件向父组件发送消息

    <div v-bind:style="{fontSize:fontsize+'em'}">
       <blog-post 
      v-for="post in posts" 
      v-bind:key="post.id" 
      v-bind:post="post"
      v-on:enlarge="fontsize+=0.1"
      监听enlarge事件,如果触发了,就执行后面的js语句或者方法
      来处理父组件里的fontsize属性
      >
    </blog-post>

  Vue.component('blog-post', {
      props: ['post'],
      template: `
      <div class="blog-post">
        <h3>{{ post.title }}</h3>
        <button v-on:click="$emit('enlarge')" >变大吧</button>
      监听点击事件,如果监听到了点击,就用$emit()触发enlarge事件
      </div>
      `
    })

  new Vue({
      el: '#demo',
      data: {
        fontsize:1,
        posts: [{
            id: 1,
            title: 'My journey with Vue',
          },
          {
            id: 2,
            title: 'Blogging with Vue',
          },
          {
            id: 3,
            title: 'Why Vue is so fun',
          }
        ]
      }
    })

使用事件抛出值

即$emit()的第二个参数可以用$event访问到这个被抛出的值

        <button v-on:click="$emit('enlarge',0.1)" >变大吧</button>
        
      <div v-bind:style="{fontSize:fontsize+'em'}">
       <blog-post v-on:enlarge="fontsize+=$event">
    </blog-post>

如果事件处理函数是一个方法,那么$event 将成为函数的第一个参数

    <blog-post  v-for="post in posts" 
                  v-bind:key="post.id" 
                  v-bind:post="post" 
                  v-on:enlarge='enlargeText'>
      </blog-post>

      methods: {
        enlargeText:function(x){
          this.fontsize +=x
        }
      }

组件上使用v-model

<input v-model="searchText">
等价于:

<input
  v-bind:value="searchText"
  v-on:input="searchText = $event.target.value"
>
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容