组件

组件

概念

html,css和js的集合体,可以提高代码的复用性

分类

根组件:new Vue()生成的组件

局部组件:let 组件名={} , {}中填写vue语法,并需要在根组件中注册使用

全局组件:Vue.component('组件名',{}), {}内部使用vue语法

特点

  • 每一个组件都是一个vue实例

  • 每一个组件都具有自身的模板template,根组件的模板就是挂载点的html,

  • template有且只有一个根标签

  • 子组件的数据需要隔离(数据组件化,每个组件拥有自己的名称空间)

  • 组件中出现的所有变量(模板,变量),由该组件自己提供.而变量名

  • 父组件中出现的所有变量,由父组件提供

根组件

<body>
    <div id="app">
        {{ msg }}
    </div>
</body>
<script src="js/vue.js"></script>
<script>new Vue({
        el: '#app',  // 被组件 template 模块进行替换的占位符
        data: {
            msg: '组件信息'
        },
        template: '<p>{{ msg }}</p>'
    })
</script>

// 总结:根组件,可以不明确template,template默认采用挂载点页面结构;如果设置的template,挂载点内部的内容无效,因为会被替换
// 解释:html,body标签不能被替换,所以不能作为挂载点

局部组件

<body>
<div id="app">
    <div class="wrap">
        <local-tag></local-tag>
    </div>
</div>
</body>
<script src="js/vue.js"></script>
<script>
    let localTag = {
        template:`
        <div class="box" @click="fn">
            <img src="img/001.jpg" alt="">
            <h2>beauty</h2>
        </div>
        `,
        methods:{
            fn(){
                console.log(this)
            }
        }
    }
    new Vue({
        el:"#app",
        data:{},
        //注册局部组件
        components:{
            //相当于localTag=localTag,
            //当变量名=变量值的时候,可以只写一个来简化表示
            localTag,
        }
    })
</script>

全局组件

<body>
<div id="app">
    <div class="wrap">
        <local-tag></local-tag>
        <global-tag></global-tag> 
    </div>
</div>
</body>
<script src="js/vue.js"></script>
<script>
    Vue.component("global-tag",{
        template:`
        <div class="box" @click="fn">
            <img src="img/002.jpg" alt="">
            <h2>leg</h2>
        </div>
        `,
        methods:{
            fn(){
                console.log(this)
            }
        }
            
    });
    new Vue({
        el:"#app",
        data:{},
        //注册局部组件
        components:{
            //相当于localTag=localTag,
            //当变量名=变量值的时候,可以只写一个来简化表示
            localTag,
        }
    })
</script>

组件传参

父传子

<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <style>
        body,h2 {
            margin: 0;
        }
        .wrap {
             880px;
            margin: 0 auto;
        }
        .wrap:after {
            content: '';
            display: block;
            clear: both;
        }
        .box {
             200px;
            border-radius: 10px;
            overflow: hidden;
            background-color: #eee;
            float: left;
            margin: 10px;
        }
        .box img {
             200px;
            height: 240px;
        }
        .box h2 {
            text-align: center;
            font-size: 20px;
            font-weight: normal;
        }
    </style>
</head>
<body>
<div id="app">
    <div class="wrap">
        //v-for="dog in dogs"在这里控制了渲染local-tag组件的次数
        <local-tag v-for="dog in dogs" :dog="dog"></local-tag>
    </div>
</div>
</body>
<script src="js/vue.js"></script>
<script>
    let dogs = [
        {
            name: '二哈1号',
            img: 'img/100.jpg',
        },
        {
            name: '二哈2号',
            img: 'img/200.jpg',
        },
        {
            name: '二哈3号',
            img: 'img/300.jpg',
        },
        {
            name: '二哈4号',
            img: 'img/400.jpg',
        },
        {
            name: '二哈1号',
            img: 'img/100.jpg',
        },
        {
            name: '二哈2号',
            img: 'img/200.jpg',
        },
        {
            name: '二哈3号',
            img: 'img/300.jpg',
        },
        {
            name: '二哈4号',
            img: 'img/400.jpg',
        }
    ];
    let localTag = {
        props:['dog'],
        template:`
            <div class="box" @click="fn">
                <img :src="dog.img" alt="">
                <h2>打{{dog.name}}{{count}}次</h2>
            </div>
        `,
        data (){
            return{
                count:0,
            }
        },
        methods:{
            fn(){
                this.count++;
            }
        },
    };
    new Vue({
        el:"#app",
        data:{
            dogs,
        },
        components:{
            localTag,
        }
    })
</script>

父组件中使用自定义方法传入变量:dog="dog"

子组件中就可以通过template接收,方法fn同理.

父传给子的属性值,只可以用,不可以改

子传父

父类中自定义方法无法自动触发,必须手动触发

<body>
    <div id="app">
        <h1>{{ h1 }}</h1>
        <h3>{{ h3 }}</h3>


        <!--自定义组件标签的事件
            自定义事件是属于子组件的,子组件在父组件中渲染并绑定事件方法,所以事件方法由父组件来实现
            子组件如何触发自定义事件:this.$emit('自定义事件名', 触发事件回调的参数们)

            子组件触发自定义事件,携带出子组件的内容,在父组件中实现自定义事件的方法,拿到子组件传递给父组件的消息
        -->
        <tag @action="actionFn"></tag>
        <hr>
        <tag2 @h1a="aFn1" @h3a="aFn3"></tag2>
    </div>
</body>
<script src="js/vue.js"></script>
<script>
    let tag = {
        template: `
        <div>
            <input type="text" v-model="t1">
            <input type="text" v-model="t2">
            <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="s1" @input="t1Fn">
            子标题内容:<input type="text" v-model="s2">
        </div>
        `,
        data() {
            return {
                s1: '',
                s2: '',
            }
        },
        methods: {
            t1Fn() {
                this.$emit('h1a', this.s1);
            }
        },
        watch: {
            s2 () {
                this.$emit('h3a', this.s2);
            }
        }
    };
    new Vue({
        el: '#app',
        data: {
            h1: '主标题',
            h3: '子标题'
        },
        components: {
            tag,
            tag2,
        },
        methods: {
            actionFn(a, b, c) {
                // console.log('触发了', a, b, c);
                this.h1 = a;
                this.h3 = b;
            },
            aFn1(a) {
                if (!a) {
                    this.h1 = '主标题';
                    return;
                }
                this.h1 = a;
            },
            aFn3(a) {
                if (!a) {
                    this.h3 = '子标题';
                    return;
                }
                this.h3 = a;
            },
        }
    })
</script>

let的子组件中的自定义事件(changeTitle)是属于子组件的,子组件在父组件中渲染并绑定事件方法(action),通过父组件定义方法来实现.

子组件触发自定义事件:this.$emit("自定义事件名",触发事件需要传入的参数们)

子组件触发自定义事件,将参数传给父组件方法,通过父组件自定义事件的方法,就可以拿到子组件传递给父组件的数据.

原文地址:https://www.cnblogs.com/agsol/p/12064592.html