Vue笔记

一:环境搭建

    1.安装node.js

    2.cnpm 下载包

      地址:http://npm.taobao.org/

      安装cnpm:

      npm install -g cnpm --registry=https://registry.npm.taobao.org

    3.搭建vue的开发环境:

      https://cn.vuejs.org/v2/guide/installation.html

      1、必须要安装nodejs

      2、搭建vue的开发环境 ,安装vue的脚手架工具  官方命令行工具

        npm install --global vue-cli  /  cnpm install --global vue-cli (此命令只需要执行一次)

      3、创建项目  必须cd到对应的一个项目里面

        vue init webpack vue-demo01

        cd  vue-demo01

        cnpm install  /  npm install          如果创建项目的时候没有报错,这一步可以省略。如果报错了  cd到项目里面运行  cnpm install  /  npm install

        npm run dev

      4、另一种创建项目的方式  (推荐)  安装文件比较简洁

        vue init webpack-simple vuedemo02

        cd  vuedemo02

        cnpm install  /  npm install       

        npm run dev


二:运行说明

    1、cd 到demo里面

    如:cd vuedemo

    2、安装依赖:

    cnpm install 

    3、运行项目

    npm run dev

三:基本代码

    {{msg}}---双大括号是插值表达式,小胡子语法。

    el:'#app'---表示Vue实例控制页面上那个区域。

    data:{ }---存放el中需要用到的数据

    methods:{}---方法,定义当前实例所有可用的方法

四:Vue指令

    1:v-cloak---解决在网络延迟时插值表达式闪烁问题

      用法:

      <p v-cloak>{{msg}}</p>

      <style>[v-cloak]{display: none;}</style>

    2: v-text---默认没有闪烁问题,会覆盖元素中原本的内容

      用法:

      <p v-text="msg">被v-text覆盖的内容</p>

    3: v-html---双大括号的方式会将数据解释为纯文本,而非HTML。为了输出真正的HTML,可以用v-html指令。它等同于JS的innerHtml属性 

      <p v-html="msg2">1212112</p>

    4: v-bind---绑定属性的指令,简写为 :要绑定的属性。只能实现单向绑定,从M自动绑定到V

      <input type="button" value="按钮" v-bind:title="mytitle + '123'">

    5:v-on---事件绑定机制,简写为 @

      <input type="button" value="按钮" v-on:click="show">

      <input type="button" value="按钮" @:click="show">

    6:v-model---实现表单元素和Model数据双向绑定,只能运用在表单元素中。input(radio, text, address, email....)  select    checkbox  textarea

      <input type="text" v-model="msg"> 

    7:v-for---循环数组

      循环普通数组

        <p v-for="item in list">{{item}}</p>

        <li v-for="(item, i) in list">索引:{{i}} --- 姓名:{{item.name}} --- 年龄:{{item.age}}</li>

        循环对象数组

        <p v-for="(user, i) in list">Id:{{ user.id }} --- 名字:{{ user.name }} --- 索引:{{i}}</p>

        循环遍历对象身上的属性

        <div v-for="(val, key, i) in userInfo">{{val}} --- {{key}} --- {{i}}</div>

        迭代数字

        <p v-for="count in 10">这是第 {{ count }} 次循环</p>

        Key属性使用---循环时,key属性只能使用number和string,须使用 v-bind 属性绑定的形式,指定 key 的值

        <p v-for="item in list" :key="item.id">

          <input type="checkbox">{{item.id}} --- {{item.name}}

        </p>

    8. v-if---每次都会重新删除或者创建元素,有更高的切换消耗

        <h3 v-if="flag">这是用v-if控制的元素</h3>

    9. v-show---每次不会重新进行DOM的删除和创建操作,只是切换了元素的 display:none 样式

        <h3 v-show="flag">这是用v-show控制的元素</h3>

        如果元素涉及到频繁的切换,最好不要使用 v-if, 而是推荐使用 v-show

        如果元素可能永远也不会被显示出来被用户看到,则推荐使用 v-if

五:事件修饰符

    .stop --停止--阻止冒泡

    .prevent --阻止--阻止默认行为

    .capture --捕获--实现捕获触发事件机制   

    .self--自己--实现只有点击当前元素时候,才会触发事件处理函数

    .once--一次--只触发一次事件函数

六:Vue中样式

    1.普通样式

      <p class="red thin">一</p>

    2.传递一个数组,注意: 这里的 class 需要使用  v-bind 做数据绑定

      <p :class="['thin', 'italic']">传递一个数组</p>

    3.数组中使用三元表达式

      <p :class="['thin','italic',flag?'active':'']">数组中使用三元表达式</p>

    4.数组中嵌套对象

      <p :class="['red', 'thin', {'active': isactive}]">数组中嵌套对象</p>

    5.直接使用对象

      <p :class="classObj">直接使用对象</p>

      <p :class="red:true, italic:true, active:true,thin:true">直接使用对象</p>

    6.使用内联样式

      直接在元素上通过 `:style` 的形式,书写样式对象

      <p :style="{color: 'red', 'font-size': '40px'}">这是一个善良的p</p>

      <p :style="[ styleObj1, styleObj2, styleObj3]">这是一个p</p>

      <script>

        var vm = new Vue({

          el: '#app',

          data: {

            flag: true,

            classObj: { red: true, thin: true, italic: false, active: false }

          },

          methods: {}

        });

        </script>

        <style>

        .red {color: red;  }

        .thin {font-weight: 200;}

        .italic {font-style: italic;  }

        .active {letter-spacing: 0.5em;  }

        </style>

七:键盘修饰符以及自定义键盘修饰符

      通过Vue.config.keyCodes.名称 = 按键值来自定义案件修饰符的别名:

      Vue.config.keyCodes.f2 = 113;

      <input type="text" v-model="name" @keyup.f2="add">

八:过滤器定义


    Vue.filter('过滤器的名称', function(){})

    过滤器中的 function ,第一个参数,已经被规定死了,永远都是 过滤器 管道符前面 传递过来的数据

    Vue.filter('过滤器的名称', function (data) {

      return data + '123'

      })

    过滤器调用时候的格式    {{ name | 过滤器的名称 }}

    私有过滤器:

      1. HTML元素:

      ```

      <td>{{item.ctime | dataFormat('yyyy-mm-dd')}}</td>

      ```

      2. 私有 `filters` 定义方式:

      ```

      filters: { // 私有局部过滤器,只能在 当前 VM 对象所控制的 View 区域进行使用


          dataFormat(input, pattern = "") { // 在参数列表中 通过 pattern="" 来指定形参默认值,防止报错


            var dt = new Date(input);


            // 获取年月日


            var y = dt.getFullYear();


            var m = (dt.getMonth() + 1).toString().padStart(2, '0');


            var d = dt.getDate().toString().padStart(2, '0');




            // 如果 传递进来的字符串类型,转为小写之后,等于 yyyy-mm-dd,那么就返回 年-月-日


            // 否则,就返回  年-月-日 时:分:秒


            if (pattern.toLowerCase() === 'yyyy-mm-dd') {


              return `${y}-${m}-${d}`;


            } else {


              // 获取时分秒


              var hh = dt.getHours().toString().padStart(2, '0');


              var mm = dt.getMinutes().toString().padStart(2, '0');


              var ss = dt.getSeconds().toString().padStart(2, '0');




              return `${y}-${m}-${d} ${hh}:${mm}:${ss}`;


            }


          }


        }


      ```


      ### 全局过滤器

      ```


      // 定义一个全局过滤器


      Vue.filter('dataFormat', function (input, pattern = '') {


        var dt = new Date(input);


        // 获取年月日


        var y = dt.getFullYear();


        var m = (dt.getMonth() + 1).toString().padStart(2, '0');

        var d = dt.getDate().toString().padStart(2, '0');

        // 如果 传递进来的字符串类型,转为小写之后,等于 yyyy-mm-dd,那么就返回 年-月-日

        // 否则,就返回  年-月-日 时:分:秒


        if (pattern.toLowerCase() === 'yyyy-mm-dd') {


          return `${y}-${m}-${d}`;


        } else {


          // 获取时分秒


          var hh = dt.getHours().toString().padStart(2, '0');


          var mm = dt.getMinutes().toString().padStart(2, '0');


          var ss = dt.getSeconds().toString().padStart(2, '0');

          return `${y}-${m}-${d} ${hh}:${mm}:${ss}`;

        }

      });

      ```

      > 注意:当有局部和全局两个名称相同的过滤器时候,会以就近原则进行调用,即:局部过滤器优先于全局过滤器被调用!

九:生命周期函数

    <script>

      // 开始创建 Vue 实例,得到 ViewModel

      var vm = new Vue({

        el: '#app',

        data: {

          msg: 'ok'

        },

        methods: {

          show() {

            console.log('执行了show方法')

          }

        },

        beforeCreate() { 创建前//  这是我们遇到的第一个生命周期函数,表示实例完全被创建出来之前,会执行它

          // console.log(this.msg)

          // this.show()

          //刚初始化一个vue空的实例对象,这时候,这个对象身上,只有默认的一些生命周期函数和默认事件,其他东西都未创建

          // 注意: 在 beforeCreate 生命周期函数执行的时候,data 和 methods 中的 数据都还没有没初始化

        },

        created() { 创建// 这是遇到的第二个生命周期函数

          // console.log(this.msg)

          // this.show()

          //  在 created 中,data 和 methods 都已经被初始化好了!

          // 如果要调用 methods 中的方法,或者操作 data 中的数据,最早,只能在 created 中操作

        },

        beforeMount() { 即将挂载// 这是遇到的第3个生命周期函数,表示模板已经在内存中编译完成了,但是尚未把模板渲染到页面中,此时,页面还是旧的。

          // console.log(document.getElementById('h3').innerText)

          // 在 beforeMount 执行的时候,页面中的元素,还没有被真正替换过来,只是之前写的一些模板字符串

        },

        mounted() { 挂载// 这是遇到的第4个生命周期函数,表示,内存中的模板,已经真实的挂载到了页面中,用户已经可以看到渲染好的页面了

          // console.log(document.getElementById('h3').innerText)

          // 注意: mounted 是 实例创建期间的最后一个生命周期函数,当执行完 mounted 就表示,实例已经被完全创建好了,此时,如果没有其它操作的话,这个实例,就静静的 躺在我们的内存中,一动不动

          //肉果要通过某些插件操作页面上的DOM节点,最早要在mounted中进行。只要执行力mounted,就表示整个Vue实例已经初始化完毕,

          //此时,组件已经脱离了创建阶段,进入到运行阶段。

        },

        // 接下来的是运行中的两个事件

        beforeUpdate() { 更新前// 这时候,表示 我们的界面还没有被更新【数据被更新了吗?  数据肯定被更新了】

          /* console.log('界面上元素的内容:' + document.getElementById('h3').innerText)

          console.log('data 中的 msg 数据是:' + this.msg) */

          // 得出结论: 当执行 beforeUpdate 的时候,页面中的显示的数据,还是旧的,此时 data 数据是最新的,页面尚未和 最新的数据保持同步

        },

        updated() { 更新

          console.log('界面上元素的内容:' + document.getElementById('h3').innerText)

          console.log('data 中的 msg 数据是:' + this.msg)

          // updated 事件执行的时候,页面和 data 数据已经保持同步了,都是最新的

        },

        beforeDestroy(){  销毁前

        /*  当执行beforeDest钩子函数的时候,Vue实例就已经从运行阶段,进入到销毁阶段;

        当执行beforeDe的时候,实例身上所有data和所有methods,以及过滤器,指令...都处于可用状态,此时,还没有真正执行销毁过程 */

        },

        destroyed(){ 完全销毁

          /* 当执行到destroyed函数的时候,组件已经完全销毁了,此时组件中所有的数据,方法,指令,过滤器,都已经不可用了。 */

        }

      });

    </script>

十:使用vue-resource请求数据的步骤

        1.安装vue-resource模块,加上--save

          npm install vue-resource --save /  cnpm install vue-resource --save

        2.在main.js引入vue-resource 

          import VueResource from 'vue-resource';

        3.在main.js引入Vue.use(VueResource);

          Vue.use(VueResource);

        4.在组件里面直接使用 

          this.$http.get(地址).then(function(){

          })

          <template>

            <!-- 所有的内容要被根节点包含起来 -->

            <div id="home">

                首页组件

                <button @click="getData()">请求数据</button>

                    <hr>

                <br>

                <ul>

                    <li v-for="item in list">

                        {{item.title}}

                    </li>

                </ul>

            </div>

        </template>

          <script>

                export default{

                    data(){

                        return {

                            msg:'我是一个首页组件msg',

                            flag:true,

                            list:[]

                        }

                    },

                    methods:{

                        getData(){

                                //请求数据

                                var api='http://www.phonegap100.com/appapi.php?a=getPortalList&catid=20&page=1';

                                this.$http.get(api).then((response)=>{

                                    console.log(response);

                                    //注意this指向

                                    this.list=response.body.result;

                                },function(err){

                                        console.log(err);

                                })

                        }

                    },

                    mounted(){  /*生命周期函数*/

                            this.getData();

                    }

                }

            </script>

十一:使用vue-axios请求数据

      1.安装vue-axios       

        cnpm install axios --save 

      2.哪里使用哪里引入

        import Axios from 'axios'

        <script>

            import Axios from 'axios';

            export default{

                data(){

                    return {

                        list:[]

                    }

                },

                methods:{

                    getData(){

                        var api='http://www.phonegap100.com/appapi.php?a=getPortalList&catid=20&page=1';

                        Axios.get(api).then((response)=>{

                            this.list=response.data.result;

                        }).catch((error)=>{

                            console.log(error);

                        })

                    }

                },

                mounted(){  /*生命周期函数*/


                    this.getData();

                }

            }

        </script>

十二:组件使用

      1.在script标签内引入组件

      2. 在components内挂载组件 

      3.在模板中使用组件

      <template>

          <div id="app">

            <v-home></v-home><!--在模板中使用组件 -->

            <v-news></v-news>

          </div>

        </template>

        <script>

          //引入组件

          import Home from './components/Home.vue';

          import News from './components/News.vue';

          export default {

              data () {

                return {


                msg:'你好vue'

                }

              },

              components:{  /*前面的组件名称不能和html标签一样*/

                'v-home':Home, //挂载组件

                'v-news':News

              }

            }

        </script>

十三:父组件给子组件传值 父组件给子组件传方法

      1. 父组件调用字组件时候,绑定动态属性

        <v-header :title="title"></v-header>

      2.在子组件里面通过props接收父组件传过来的数据

        props:['title']     

      父组件代码: 

                <template>

                    <div id="home">

                        <v-header :title="title" :homemsg='msg'  :run="run"  :home="this"></v-header>

                        <hr>

                        首页组件

                    </div>

                </template>

                <script>

                    import Header from './Header.vue';

                    export default{

                        data(){

                            return {

                              msg:'我是一个home组件',

                              title:'首页111'

                            }

                        },

                        components:{

                            'v-header':Header

                        },

                        methods:{

                            run(data){

                              alert('我是Home组件的run方法'+data);

                            }

                        }

                    }

                </script>

      子组件代码:

                <template>

                    <div>

                        <h2>我是头部组件--{{title}}---{{homemsg}}</h2>

                        <button @click="run('123')">执行父组件的方法</button>

                        <button @click="getParent()">获取父组件的数据和方法</button>

                    </div>

                </template>

                <script>

                export default{

                    data(){

                        return{

                            msg:'子组件的msg'

                        }

                    },

                    methods:{

                        getParent(){

                            this.home.run()

                        }

                    },

                    props:['title','homemsg','run','home']

                }

                </script> 

十四:路由使用

        1.安装

          npm install vue-router  --save  / cnpm install vue-router  --save

        2、引入并 Vue.use(VueRouter)  (main.js)

          import VueRouter from 'vue-router'

          Vue.use(VueRouter)

        3、配置路由  下面四个在main.js里面

          1、创建组件 引入组件

          2、定义路由  (建议复制s)

              const routes = [

              { path: '/home', component: Home },

              { path: '/news', component: News },

              <!-- 属性1 是 path, 表示监听 哪个路由链接地址;

                  属性2 是 component, 表示,如果 路由是前面匹配到的 path ,则展示 component 属性对应的那个组件

                  注意: component 的属性值,必须是一个 组件的模板对象, 不能是 组件的引用名称

              -->

              { path: '*', redirect: '/home' }  <!-- 默认跳转路由 -->

            ]

          3、实例化VueRouter

              const router = new VueRouter({

                routes // (缩写)相当于 routes: routes

              })

          4、挂载

              new Vue({

                el: '#app',

                router,

                render: h => h(App)

              })

          5 、根组件的模板里面放上这句话 

          <!-- vue-router提供的占位符 将来,路由规则,匹配到的组件,就会展示到这个 router-view 中去-->

          <router-view></router-view>     

          <!-- <router-view></router-view> 外层可以包裹 <transition>设置动画-->

          6、路由跳转

            <!--router默认渲染为a标签,使用tag属性可以更改默默认a标签 -->

            <router-link to="/foo" tag="span">Go to Foo</router-link>

            <router-link to="/bar">Go to Bar</router-link>

          7.1.this.$router.go(val) => 在history记录中前进或者后退val步,当val为0时刷新当前页面。

          2.this.$router.push(path) => 在history栈中添加一条新的记录。 

十五:动画

      1.使用transition元素,把需要被动画控制的元素包裹起来

        <transition mode="out-in"> <!-- mode="out-in是动画先出后进" -->

          <p>被控制元素</p>

        </transition>

      2.自定义两组样式,来控制 transition 内部的元素实现动画

        <style>

            /* v-enter 【这是一个时间点】 是进入之前,元素的起始状态,此时还没有开始进入 */

            /* v-leave-to 【这是一个时间点】 是动画离开之后,离开的终止状态,此时,元素 动画已经结束了 */

            .v-enter,

            .v-leave-to {

              opacity: 0;

              transform: translateX(150px);

            }

            /* v-enter-active 【入场动画的时间段】 */

            /* v-leave-active 【离场动画的时间段】 */

            .v-enter-active,

            .v-leave-active{

              transition: all 0.8s ease;

            }

          </style>

        3.修改动画v-前缀    transition加入name属性

          <transition name="my"> 

            <p>被控制元素</p>

          </transition>

          样式里面v-改完自定义名字

          <style>

            .my-enter,.my-leave-to{opacity: 0;}

            .my-enter-active,.my-leave-active{transition: all 0.8s ease;}

          </style>

        4.使用animate库实现动画,用npm安装看api

          <transition enter-active-class="animated bounceIn"

                    leave-active-class="animated bounceOut">

            <h3 v-if="flag">这是一个H3</h3>

          </transition>

        5.动画钩子函数 

          +1:定义 transition 组件以及三个钩子函数:

              <div id="app">

                  <input type="button" value="切换动画" @click="isshow = !isshow">

                  <transition

                  @before-enter="beforeEnter"

                  @enter="enter"

                  @after-enter="afterEnter">

                    <div v-if="isshow" class="show">OK</div>

                  </transition>

                </div>

            +2. 定义三个 methods 钩子方法

                <script>

                methods: {

                  // 注意: 动画钩子函数的第一个参数:el,表示 要执行动画的那个DOM元素,是个原生的 JS DOM对象

                  // 大家可以认为 , el 是通过 document.getElementById('') 方式获取到的原生JS DOM对象

                  beforeEnter(el){

                    // beforeEnter 表示动画入场之前,此时,动画尚未开始,可以 在 beforeEnter 中,设置元素开始动画之前的起始样式

                    // 设置小球开始动画之前的,起始位置

                    el.style.transform = "translate(0, 0)"

                  },

                  enter(el, done){

                    // 这句话,没有实际的作用,但是,如果不写,出不来动画效果;

                    // 可以认为 el.offsetWidth 会强制动画刷新

                    el.offsetWidth

                    // enter 表示动画 开始之后的样式,这里,可以设置小球完成动画之后的,结束状态

                    el.style.transform = "translate(150px, 450px)"

                    el.style.transition = 'all 1s ease'


                    // 这里的 done, 起始就是 afterEnter 这个函数,也就是说:done 是 afterEnter 函数的引用

                    done()

                  },

                  afterEnter(el){

                    // 动画完成之后,会调用 afterEnter

                    // console.log('ok')

                    this.flag = !this.flag

                  }

                }

                </script>

十六:使用ref获取DOM元素和组件

      <h3 id="myh3" ref="myh3">哈哈哈, 今天天气太好了!!!</h3>     

      <script>

      methods:{

        console.log(this.$refs.myh3.innerText

      }

      </script>

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