vue组件

组件概念

html、css、js 的集合体,为该集合体命名,用该名字复用html、css与js组成的集合体 复用性

组件分类:

  • 根组件:new Vue() 生成的组件
  • 局部组件: 组件名={} 内部采用的是vue语法
  • 全局组件: vue.component('组件名', {}, {})

组件的特点:

  • 组件都有管理组件html 页面结果 的 template 实例成员,template中有且只有一个根标签
  • 根组件都是作为最顶层父组件, 局部与全局组件作为子组件,也可以成为其他局部与全局的父组件
  • 子组件的数据需要隔离(数据组件化,每一个组件拥有自己的数据的独立名称空间)
  • 局部组件必须注册后才能使用,全局组件不需要注册,提倡使用局部组件
  • 组件中出现的所有的变量,都由该组件自己提供管理
  • 局部全局和根组件都是一个vue实例,一个实例对应一套html、css、js结构,所以实例就是组件

根组件

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<div id="app">
    <h1>{{ msg }}</h1>    
</div>
</body>
<script src="vue/vue.min.js"></script>
<script>
    new Vue({
        el: '#app', // 挂载点不是虚拟DOM,而是要被虚拟DOM 替换的对象
        			// 被组件 template 模块尽行替换的占位符
        data: {
            msg: '根组件'
        },

        template: '<i>显示模板</i>'
    })
</script>
</html>

挂载点 本质上是被组件 template 模块进行替换的占位符

总结:

  • 根组件 可以不明确template,template默认采用挂载点 页面结构,如果设置的template,挂载点内部的内容无效

局部组件

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
<div id="app">
    <local-tag></local-tag>
    <local-tag></local-tag>
</div>
<script>
    let localTag = {
        data () {
            return {
                count: 0
            }
        },
        template: '<button @click="btnAction">局部{{ count }}</button>',
        methods: {
            btnAction () {
                this.count ++
            }
        }
    }
    new Vue({
        el: "#app",
        components: {
            'local-tag': localTag
        }
    })
</script>

全局组件

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<div id="app">
    <global-tag></global-tag>
    <global-tag></global-tag>
</div>
<script>
	Vue.component('global-tag', {
		data () {
			return {
				count: 0
			}
		},
		template: '<button @click="btnAction">全局{{ count }}</button>',
		methods: {
			btnAction () {
				this.count ++
			}
		}
	})
    new Vue({
        el: "#app"
    })
</script>
</html>

数据的组件化

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <style>
        body, h2 {
            margin: 0;
        }
        .wrap {
            width: 880px;
            margin: 0 auto;
        }
        .wrap:after {
            content: '';
            display: block;
            clear: both;
        }
        .box {
            width: 200px;
            border-radius: 10px;
            overflow: hidden;
            background-color: darkgrey;
            float: left;
            margin: 10px;
        }
        .box img {
            width: 100%;
        }
        .box h2 {
            text-align: center;
            font-weight: normal;
            font-size: 20px;
        }
    </style>
</head>
<body>
<div id="app">
    <div class="wrap">
        <local-tag></local-tag>
        <local-tag></local-tag>
        <local-tag></local-tag>
        <local-tag></local-tag>

        <global-tag></global-tag>
        <global-tag></global-tag>
        <global-tag></global-tag>
        <global-tag></global-tag>


     </div>
</div>
</body>
<script src="vue/vue.min.js"></script>
<script>
    let localTag = {
        // 声明局部组件,局部组件要在其父组件中注册才能使用
        template: `
        <div class="box" @click="fn">
            <img src="img/img123.jpg" alt="">
        <h2>阿斯弗 {{ count }}</h2>
        </div>
        `,

        data () {   // 局部或全局组件,一个模板可能会被复用多次,每个组件都应该有自己独立的变量名称空间
          return {
              count: 0
          }
        }, // 数据需要组件化,作为方法的返回值(方法执行后会产生一个局部作用域)

        methods: {
            fn() {
                console.log(this);
                this.count ++
            }
        }

    };

    Vue.component('global-tag', {
        template: `
        <div class="box" @click="fn">
            <img src="img/img2.jpg" alt="">
        <h2>潜水</h2>
        `,

        methods: {
            fn() {
                console.log(this)
            }
        }

    });

    new Vue({
        el: '#app',
        data: {},
        components: {
            localTag: localTag
        }
    })
</script>
</html>

父组件传递数据给子组件

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <style>
        body, h2 {
            margin: 0;
        }
        .wrap {
            width: 880px;
            margin: 0 auto;
        }
        .wrap:after {
            content: '';
            display: block;
            clear: both;
        }
        .box {
            width: 200px;
            border-radius: 10px;
            overflow: hidden;
            background-color: darkgrey;
            float: left;
            margin: 10px;
        }
        .box img {
            width: 200px;
        }
        .box h2 {
            text-align: center;
            font-weight: normal;
            font-size: 20px;
        }
    </style>
</head>
<body>
<div id="app">
    <div class="wrap">
        <local-tag v-for="view in views" :view="view"></local-tag>


     </div>
</div>
</body>
<script src="vue/vue.min.js"></script>
<script>

    let views = [
        {
            name: '1号',
            img: 'img/100.jpg'
        },
        {
            name: '2号',
            img: 'img/200.jpg'
        },
        {
            name: '3号',
            img: 'img/300.jpg'
        },
        {
            name: '4号',
            img: 'img/400.jpg'
        }
    ];

    // 子组件可以通过props自定义组件属性  (采用反射机制,需要填写字符串,但是使用时可以直接作为变量)
    // 子组件会在父组件中渲染,渲染时将父组件的变量绑定给子组件的自定义属性,就可以将变量值传递给子组件
    let localTag = {
        props: ['view'],

        // 声明局部组件,局部组件要在其父组件中注册才能使用
        template: `
        <div class="box" @click="fn">
            <img :src="view.img" alt="">
        <h2>{{ view.name }} {{ count }}</h2>

        </div>
        `,

        data () {   // 局部或全局组件,一个模板可能会被复用多次,每个组件都应该有自己独立的变量名称空间
          return {
              count: 0
          }
        }, // 数据需要组件化,作为方法的返回值(方法执行后会产生一个局部作用域)

        methods: {
            fn() {
                console.log(this.view);
                this.count ++
            }
        }

    };


    new Vue({
        el: '#app',
        data: {
            views
        },
        components: {
            localTag: localTag
        }
    })
</script>
</html>

总结:

  • 子组件可以通过props自定义组件属性 (采用反射机制,需要填写字符串,但是使用时可以直接作为变量)
  • 子组件会在父组件中渲染,渲染时,将父组件的变量绑定给子组件的自定义属性,就可以将变量值传递给子组件。

子组件传递数据给父组件

自定义组件标签的事件:

  • 事件是属于子组件的,子组件在父组件中渲染并绑定事件方法,所以事件方法由父组件来实现

子组件如何触发自定义事件:this.$emit('自定义事件名', '触发事件回调的参数')

子组件触发自定义事件,携带出子组件的内容,在父组件中实现自定义事件的方法,拿到子组件传递给父组件的消息

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<div id="app">
    <h1>{{ h1 }}</h1>
    <h2>{{ h2 }}</h2>
    <tag @action="actionFn"></tag>
    <hr>
    <tag2 @h1a="aFn1" @h2a="aFn2"></tag2>

</div>
</body>
<script src="vue/vue.min.js"></script>
<script>
    let tag = {

        template: `
      <div>
        <input type="text" v-model="t1">
        <input type="text" v-model="t2">
        <br>
        <button @click="changeTitle">修改标题</button>

      </div>
      `,

        data() {
            return {
                t1: '',
                t2: '',
            }
        },

        methods: {
            changeTitle() {

                if (this.t1 && this.t2) {
                    // console.log(this.t1, this.t2);
                    this.$emit('action', this.t1, this.t2);
                    this.t1 = '';
                    this.t2 = '';
                }
            }
        }

    };

    let tag2 = {
      template: `
      <div>
        主标题<input type="text" v-model="t3" @input="t1Fn">
        子标题<input type="text" v-model="t4" @input="t2Fn">
       </div>
      `,

        data() {
          return {
              t3: '',
              t4: '',
          }
        },

        methods: {
            t1Fn() {
                this.$emit('h1a', this.t3)
            },
            t2Fn() {
                this.$emit('h2a', this.t4)
            }
        }

    };


    new Vue({
        el: '#app',
        data: {
            h1: '主标题',
            h2: '子标题'
        },
        components: {
            tag,
            tag2
        },
        methods: {
            actionFn(a, b, c) {
                // console.log('触发了', a, b, c)
                this.h1 = a;
                this.h2 = b;
            },

            aFn1(a) {
                if (!a) {
                    this.h1 = '主标题';
                    return
                }
                this.h1 = a
            },
            aFn2(a) {
                if (!a) {
                    this.h2= '子标题';
                    return
                }
                this.h2 = a
            }
        }
    })
</script>
</html>

Vue项目环境搭建

1、安装node

官网下载安装包,傻瓜式安装:https://nodejs.org/zh-cn/

2、换源安装cnpm

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

3、安装vue项目脚手架

cnpm install -g @vue/cli

4、清空npm缓存

2或3终端安装失败时,可以清空 npm缓存 再重复执行失败的步骤

npm cache clean --force
posted @ 2019-12-18 22:43  SetCreed  阅读(318)  评论(0编辑  收藏  举报