vue 基础学习 第四天

2018-05-31  本文已影响51人  那就远走

32 组件

<div id="app">
    <!-- 定义好全局组件之后, -->
    <test1></test1>
    <test2></test2>
    <test3></test3>
    <test4></test4>
</div>


<script>
    // 我们在这里定义 “全局组件”
    Vue.component('test1', { // Vue.component('标签', { //... 定义 })
        template: '<h1> 这是全局组件的模板test1 </h1>', //模板
    });

    // 为了方便我们还可以定义一个变量存储局部组件然后在根组件中载入 (这个定义也必须声明在前面)
    var test4= {
        template: '<h2> 这是局部组件的模板test4 </h2>',
    }

    // 我们之前一直定义的其实是 “根组件”
    var app = new Vue({
        el: '#app',
        data: {
            
        },
        // 我们还可以定义 “局部组件”
        components: {
            test3: {
                template: '<h2> 这是局部组件的模板test3 </h2>',
            },
            // test4: test4,
            // 可以使用 ES6 语法,由于 key: value 是一样的,直接:
            test4
        }
    });

    // “全局组件” 必须在根组件之前定义
    Vue.component('test2', {
        template: '<h1> 这是全局组件的模板test2 </h1>',
    });
</script>

33 组件中定义 data 数据

 <div id="app">
    <test></test>
</div>

<!-- 在 Vue 中可以使用这样的方式定义模板 -->
<script type="text/x-template" id="myComponent">
    <ul>
        <li v-for="v in news">{{ v.id }} - {{ v.title }}</li>
    </ul>
</script>

<script>
    var myComponent = {
        template: "#myComponent",
        // data: {} //子组件中不能把data定义成属性。会报错:
        // The "data" option should be a function that returns a per-instance value in component definitions.
        data() { //必须这样定义 data() { return {json对象} }
            return {
                news: [
                    {id:1, title:'测试1'},
                    {id:2, title:'测试2'},
                ],
            };
        }
    };
    var app = new Vue({
        el: '#app',
        data: {

        },
        components: {
            // 有个坑: ES6 方式注册子组件不能使用 驼峰 风格命令,Vue会自动转换成全小写。
            test: myComponent,
        }
    });
</script>

34 父组件给子组件传递参数

<div id="app">
    <!-- 在标签里面传递 :子组件.props中声明的名字 = "根组件.data中的变量名" -->
    <!-- 如果不写 :属性="value" 的话,会解析为字符串 -->
    <students :students="students" flag1="false" :flag2="true"></students>    
</div>

<!-- 模板 -->
<script type="text/x-template" id="students">
    <ul>
        <li v-for="student in students"> {{ student.id }} - {{ student.name }} </li>

        <span v-if="flag1"> 这其实是字符串 {{ flag1 }} </span> <!-- 这里之所以flag = "false" 会显示是因为 字符串"false" = 布尔true -->
        <span v-if="flag2"> :这才是布尔值 {{ flag2 }} </span> <!-- 因此这里如果传递 flag2 = false 则不会显示 -->
    </ul>
</script>

<script>
    // 子组件
    var students= {
        template: "#students",
        // 在接收时需要在 props 属性中声明
        props: ['students', 'flag1', 'flag2'],
    };
    
    // 根组件
    var app = new Vue({
        el: '#app',
        // 定义父组件的数据
        data: {
            students: [
                {id: 1, name: 'liuihaoyu'},
                {id: 2, name: 'lidaye'},
                {id: 3, name: 'linainai'},
            ],
        },
        // 注册子组件
        components: {
            students,
        }
    });
</script>

35 props 数据验证

<div id="app">
    <!-- 在标签里面传递 :子组件.props中声明的名字 = "根组件.data中的变量名" -->
    <!-- 如果不写 :属性="value" 的话,会解析为字符串 -->
    <students></students>    
</div>

<!-- 模板 -->
<script type="text/x-template" id="students">
    <ul>
        <li v-for="student in students"> {{ student.id }} - {{ student.name }} </li>
    </ul>
</script>

<script>
    // 子组件
    var students= {
        template: "#students",
        // 在 props 中进行数据验证
        props: { // 1、要求props定义成一个对象
            students: { // 2、用 【属性名: { //...相关配置 }】 进行数据接收
                type: [Array, Object], // 指定数据类型
                // required: true, // 设置数据是否必填
                default() { // 设置默认值
                    return [
                        {id: 1, name: 'liuhaoyu'},
                    ];
                },  
                validator(value) { // 设置数据验证规则
                    return value[0].id > 0;
                }
            },
        }
    };
    
    // 根组件
    var app = new Vue({
        el: '#app',
        // 定义父组件的数据
        data: {
        },
        // 注册子组件
        components: {
            students,
        }
    });
</script>

36 通过子组件呼叫父组件实现简单的购物车

<div id="app">
    <cart :goods="goods" @refresh="totalPrice"></cart>
    <span>
        总计:¥ {{ total }} 元
    </span>
</div>

<!-- 模板 -->
<script type="text/x-template" id="cart">
    <table border="1">
        <thead>
            <tr>
                <th>商品名称</th>
                <th>价格</th>
                <th>数量</th>
            </tr>
        </thead>
        <tbody>
            <tr v-for="good in goods">
                <td>{{ good.name }}</td>
                <td>{{ good.price }}</td>
                <td>
                    <input type="text" v-model="good.number" @blur="refresh">
                </td>
            </tr>
        </tbody>
    </table>
</script>

<script>
    //子组件
    var cart = {
        template: "#cart",
        props: {
            goods: {
                type: Array,
            }
        },
        methods: {
            refresh() {
                this.$emit('refresh')
            }
        },
    }
    
    // 根组件
    var app = new Vue({
        el: '#app',
        data: {
            // 定义商品信息
            goods: [
                {name: "macbookPro 2018", price: "20000", number:1},
                {name: "iphone 8", price: "6888", number:1},
                {name: "iphone8 Plus", price: "8888", number:1},
            ],
            // 总价初始化
            total: 0,
        },
        // 注册子组件
        components: {
            cart,
        },
        methods: {
            // 计算总价
            totalPrice() {
                this.total = 0;
                this.goods.forEach((good) => {
                    this.total += good.price * good.number;
                });
            }
        },
        // 挂载钩子程序
        mounted() { // 类似于 初始化程序
            this.totalPrice(); // 直接调用计算总价的方法
        },
    });
</script>

37 更优写法实现36购物车功能

<div id="app">
    <!-- 这里使用 :绑定属性.sync同步数据="父组件的goods" => 达到了当子组件的goods发生变化时,父组件的goods也会变化 -->
    <cart :goods.sync="goods"></cart>
    <span>
        总计:¥ {{ totalPrice }} 元
    </span>
</div>

<!-- 模板 -->
<script type="text/x-template" id="cart">
    <table border="1">
        <thead>
            <tr>
                <th>商品名称</th>
                <th>价格</th>
                <th>数量</th>
            </tr>
        </thead>
        <tbody>
            <tr v-for="good in goods">
                <td>{{ good.name }}</td>
                <td>{{ good.price }}</td>
                <td>
                    <input type="text" v-model="good.number">
                </td>
            </tr>
        </tbody>
    </table>
</script>

<script>
    //子组件
    var cart = {
        template: "#cart",
        props: {
            goods: {
                type: Array,
            }
        },
    }
    
    // 根组件
    var app = new Vue({
        el: '#app',
        data: {
            // 定义商品信息
            goods: [
                {name: "macbookPro 2018", price: "20000", number:1},
                {name: "iphone 8", price: "6888", number:1},
                {name: "iphone8 Plus", price: "8888", number:1},
            ],
            // 总价初始化
            total: 0,
        },
        // 注册子组件
        components: {
            cart,
        },
        // 计算总价
        computed: {
            totalPrice() {
                var sum = 0;
                this.goods.forEach((good) => {
                    sum += good.price * good.number;
                });
                return sum;
            }
        }
    });
</script>

38 子组件 slot 内容分发

<div id="app">
    <test>
        <h1 slot="title">这是标题</h1>
        <p slot="content">这是内容</p>
        <test1 slot="myinput" type="email" title="邮箱" placeholder="username@example.com"></test1>
        <test1 slot="myinput" type="text" title="用户名" placeholder="yourNickName"></test1>
        <test1 slot="myinput" type="password" title="密码" placeholder="yourBirthday"></test1>
    </test>
</div>

<!-- 子组件模板 test -->
<script type="text/x-template" id="test">
    <div>
        <slot name="title"></slot>
        <slot name="content"></slot>
        <slot name="myinput"></slot>
    </div>
</script>

<!-- 子组件模板 test1 -->
<script type="text/x-template" id="test1">
    <div>
        <span>{{ title }}</span>
        <input :type="type" :placeholder="placeholder">
    </div>
</script>

<script>
    // 子组件 test
    var test = {
        template: "#test",
    };
    // 子组件 test1
    var test1 = {
        template: "#test1",
        props: ['type', 'title', 'placeholder']
    }
    // 根组件
    var app = new Vue({
        el: '#app',
        data: {
        },
        components: {
            test,
            test1,
        }
    });
</script>

39 scope 的使用

<div id="app">
    <users :users="users" scope="v">
        <!-- 这里 **必须** 使用 template 标签 -->
        <!-- 并且使用 scope="任意变量名" 接收 slot抛出的数据 -->
        <template scope="data">
            <li>
                {{ data.user.id }} - {{ data.user.name }}
            </li>
        </template>
    </users>
</div>

<script type="text/x-template" id="users">
    <ul>
        <!-- <li v-for="user in users"> {{ user.id }} - {{ user.name }}</li> -->
        <!-- 这里用slot定义,但是在后面用 :user="user" 抛出数据 -->
        <slot v-for="user in users" :user="user"></slot>
    </ul>
</script>

<script>
    var users = {
        template: "#users",
        props: {
            users: {
                type: Array,
            },
        }
    };
    var app = new Vue({
        el: '#app',
        data: {
            users: [
                {id: 1, name: 'liuhaoyu'},
                {id: 2, name: 'lidaye'},
                {id: 3, name: 'linainai'},
            ]
        },
        components: {
            users,
        }
    });
</script>
<子组件>
    <template scope="任意变量名这里暂时用data">
        <合适的标签>
            {{ data.data.属性 }} =>第一个data对应scope里的,第二个data对应子组件中使用 :data 抛出的。
        </合适的标签>
    </template>
</子组件>

40 动态组件

<div id="app">
    <input type="radio" v-model="formType" value="myInput"> 使用 input
    <input type="radio" v-model="formType" value="myTextarea"> 使用 textarea

    <div :is="formType"></div>
</div>

<script>
    var myInput = {
        template: "<div> <input> </div>",
    };
    var myTextarea = {
        template: "<div> <textarea></textarea> </div>",
    }
    var app = new Vue({
        el: '#app',
        data: {
            formType: "myInput",
        },
        components: {
            myInput,
            myTextarea
        }
    });
</script>
上一篇 下一篇

猜你喜欢

热点阅读