美文网首页
Vue重新起航(四)

Vue重新起航(四)

作者: 阿龙哟 | 来源:发表于2018-10-26 15:58 被阅读0次

    事件处理

    监听事件:用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"
    >
    

    相关文章

      网友评论

          本文标题:Vue重新起航(四)

          本文链接:https://www.haomeiwen.com/subject/wduktqtx.html