组件其他

跟组件和组件 一些问题
    new Vue() --->管理div ----》根组件
    自己再定义的全局, 局部是组件
    组件有自己的html,css,js ---》数据,事件。。。。。
    在组件中,this代指当前组件
    父子组件的data是无法分享的
    data是1个函数,需要有返回值(return)

组件间通信之父传子

组件间数据不共享-----》需要进行数据传递

父传子 :使用自定义属性方式

通过自定义属性--->不能用驼峰,不要跟子组件中变量冲突
'''
关键字 : props
'''
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../vue_day05/js/vue.js"></script>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
          integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous">

</head>
<body>
<div class="app">
    <h1>父子通信之父传子,通过自定义属性--->不能用驼峰,不要跟子组件中变量冲突</h1>
<!--    <h2>字符串的age&ndash;&gt;age="age"</h2>-->
<!--    <child age="age"></child>-->
<!--    <h2>:age="19"</h2>-->
<!--    <child :age="19"></child>-->
<!--    <h2>age="19"</h2>-->
<!--    <child age="19"></child>-->
    <h2>:age="age"</h2>
    <child :age="age" myname="彭于晏"></child>

    <h1>属性验证---》传入的必须是xx类型</h1>
<!--    <h2>字符串的age :age="age"</h2>-->
<!--    <child age="age"></child>-->
    <!--    <h2>:age="19"</h2>-->
    <!--    <child :age="19"></child>-->
    <!--    <h2>age="19"</h2>-->
    <!--    <child age="19"></child>-->
    <!--    <h2>:age="age"</h2>-->
    <!--    <child :age="age"></child>-->

</div>
</body>
<script>
    // 父中有age 子child 只有name,没有age,现在把父中的age传到child中 显示
    var child = {
        template: `
          <div>
          <button>后退</button>
          <span>
          首页---名字:{{ myname }}---->年龄:{{ age }}
        </span>
          <button>前进</button>
          </div>
        `,
        data() {
            return {
                myname: '彭于晏',
            }
        },
        // props:['age'],
        props: {age: Number, myname: String,},
    }


    var vm = new Vue({
        el: '.app',
        data: {
            age: 19,
        },
        components: {
            child
        }


    })
</script>
</html>

组件间通信之子传父$emit

通过自定义事件

methods:{
            handleSend(){
                //alert(this.mytext)
                // 子组件,触发自定义事件的执行,会执行父组件自定义事件绑定的函数,有几个参数,就传几个参数
                this.$emit('myevent',this.mytext)
            }

image

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../vue_day05/js/vue.js"></script>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
          integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous">

</head>
<body>
<div class="app">
    <p>子组件传递过来的数据:{{mytext}}</p>
    <hr>
    <child @myevent="handleEvent"></child>
</div>
</body>
<script>
    // 父中有age 子child 只有name,没有age,现在把父中的age传到child中 显示
    var child = {
        template: `
          <div>
          <input type="text" v-model="mytext">
          <button @click="handleSend">点我传递</button>
          </div>
        `,
        data() {
            return {
                mytext:''
            }
        },
       methods:{
            handleSend(){
                //alert(this.mytext)
                // 子组件,触发自定义事件的执行,会执行父组件自定义事件绑定的函数,有几个参数,就传几个参数
                this.$emit('myevent',this.mytext)
            }
       }
    }
    var vm = new Vue({
        el: '.app',
        data: {
            mytext:''
        },
        methods: {
           handleEvent(mytext){
               this.mytext = mytext
           }
        },
        components:{
            child
        }
    })
</script>
</html>

ref属性$refs

自定义属性和自定义事件   可以实现父子传值

ref属性 可以更方便的实现父子通信

ref属性放在普通标签上  拿到的是原生的dom对象,通过原生dom修改 标签

ref属性放在组件上    <child ref="mychild"></child>
     通过this.$refs.mychild  拿到的是组件对象,既然拿到了组件对象,组件对象中的 变量,方法,都能直接通过 . 的方式调用
     因此不需要关注是子传父,还是父传子,直接通过组件对象,使用即可 

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../vue_day05/js/vue.js"></script>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
          integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous">

</head>
<body>
<div class="app">
    <button @click="handleClick">点我</button>------》{{age}}
    <br>

    <input type="text" ref="myinput">
    <div ref="mydiv">我是div</div>
    <hr>
    <child ref="mychild"></child>
    <hr>


</div>
</body>
<script>
    // 父中有age 子child 只有name,没有age,现在把父中的age传到child中 显示
    var child = {
        template: `
          <div>

          <h1>名字:{{name}} ----->年龄:{{age}}</h1>
          <button @click="handleClick">点我弹出名字</button>
          </div>
        `,
        data() {
            return {
                name:'lqz',
                age:19
            }
        },
       methods:{
            handleClick(){
                alert(this.name)

            }
       }
    }


    var vm = new Vue({
        el: '.app',
        data: {
            age:999,
        },
        methods: {
           handleClick(){
              // 1 ref 属性放在普通标签上,拿到标签的dom对象,
               // 通过this.$refs可以拿到所有标签上写了ref属性的标签,对象类型,key值是ref对应的value值,value值是原生dom对象
               // console.log(this,$refs)
               // 直接修改原生dom对象的value属性,input就能看到有值了
               // this.$refs.myinput.value = 'lqz is handsome'
                // 2 ref 属性放在组件上,拿到的是组件对象,就可以使用组件对象的属性和方法
               console.log(this.$refs)  // 对象中有3个值,两个普通标签,一个组件
               // this.$refs.mychild  就是组件对象可以属性 方法
               this.age = this.$refs.mychild.age   //把局部组件的age赋值给全局
               // this.$refs.mychild.age=this.age     // 把全局的age赋值给局部

               // 重点:以后就不需要关注是子传父还是父传子了,直接通过对象取值赋值即可,而且可以主动调用子组件中的函数
               this.$refs.mychild.handleClick()
           }
        },
        components:{
            child
        }
    })
</script>
</html>

动态组件

不使用动态组件

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../vue_day05/js/vue.js"></script>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
          integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous">

</head>
<body>
<div class="app">
    <span @click="handleClick('home')">首页</span>| <span @click="handleClick('order')">订单</span>|
    <span @click="handleClick('goods')">商品</span>

    <home v-if="chooseType=='home'"></home>
    <order v-else-if="chooseType=='order'"></order>
    <goods v-else></goods>


</div>
</body>
<script>
    var home= {
        template:` <div>
 <h1>home页面</h1>
</div>`,
    }
    var order= {
        template:` <div>
 <h1>order页面</h1>
</div>`,
    }
    var goods= {
        template:` <div>
 <h1>商品页面</h1>
</div>`,
    }

    var vm = new Vue({
        el: '.app',
        data: {
            chooseType:'home'
        },
        methods: {
           handleClick(type){
              this.chooseType = type
           }
        },
        components:{home,
            order,goods

        }


    })


</script>
</html>

keep_alive保持组件不销毁

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../vue_day05/js/vue.js"></script>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
          integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous">

</head>
<body>
<div class="app">
    <span @click="handleClick('home')">首页</span>| <span @click="handleClick('order')">订单</span>|
    <span @click="handleClick('goods')">商品</span>


    <keep_alive>
        <component :is="who"></component>
    </keep_alive>


</div>
</body>
<script>
    var home= {
        template:` <div>
 <h1>home页面</h1>
</div>`,
    }
    var order= {
        template:` <div>
 <h1>order页面</h1>
</div>`,
    }
    var goods= {
        template:` <div>
 <h1>商品页面</h1>
 <input type="text"> <button>搜索</button>
</div>`,
    }

    var vm = new Vue({
        el: '.app',
        data: {
            who:'home'
        },
        methods: {
           handleClick(type){
              this.who = type
           }
        },
        components:{home,
            order,goods

        }


    })

</script>
</html>

插槽slot

一般情况下,编写完一个组件之后,组件的内容都是写死的,需要加数据   只能去组件中修改,扩展性很差
然后就出现了插槽这个概念,只需在组件中添加<slot></slot> 就可以在body的组件标签中添加内容

匿名插槽

<slot></slot>
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../vue_day05/js/vue.js"></script>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
          integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous">

</head>
<body>
<div class="app">
    <hr>
    <home>
        <div>
            <img src="./img.png" alt="">
        </div>
    </home>

    <hr>

</div>
</body>
<script>
    var home= {
        template:` <div>
 <h1>home页面</h1>
 <slot></slot>
 <h1>结束了</h1>
</div>`,
    }

    var vm = new Vue({
        el: '.app',
        data: {
          
        },


        components:{home,


        }


    })


</script>
</html>

具名插槽

<div slot="a">
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../vue_day05/js/vue.js"></script>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
          integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous">

</head>
<body>
<div class="app">
    <hr>
    <home>
        <div slot="a">
            <img src="./img.png" alt="" width="100px" height="100px">
        </div>
        <div slot="b">
            我是div
        </div>
    </home>

    <hr>

</div>
</body>
<script>
    var home = {
        template: ` <div>
         <h1>home页面</h1>
         <slot name="a"></slot>
         <h1>结束了</h1>
         <slot name="b"></slot>
         <slot name="a"></slot>
        </div>`,
    }

    var vm = new Vue({
        el: '.app',
        data: {},
        components: {
            home,
        }
    })


</script>
</html>

计算属性computed

计算属性只有使用的变量发送变化时,才重新运算
计算属性就像python中的property 可以把方法/函数伪装成属性

计算属性基本使用

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="../vue_day05/js/vue.js"></script>
    <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/3.4.1/css/bootstrap.min.css"
          integrity="sha384-HSMxcRTRxnN+Bdg0JdbxYKrThecOKuH5zCYotlSAcp1+c8xmyTe9GYg1l9a69psu" crossorigin="anonymous">

</head>
<body>
<div class="app">
<!--   <h1>input输入单词 首字母转成大写展示</h1>-->

<!--    <input type="text" v-model="mytext">{{mytext.slice(0,1).toUpperCase()+ mytext.slice(1)}}-&ndash;&gt;-->

    <h1>input输入单词 首字母转成大写展示----函数方式---只要页面刷新,无论跟它有没有关,都会重新运算</h1>

<!--    <input type="text" v-model="mytext">&#45;&#45;&#45;&#45;&#45;&#45;{{getUpper()}}-->

    <input type="text" v-model="mytext"> ---->{{newText}}

    <br>
    <input type="text" v-model="age">  ------>{{age}}
</div>
</body>
<script>

    var vm = new Vue({
        el: '.app',
        data: {
            mytext:'',
            age:10
        },
        methods:{
            getUpper(){
                console.log('函数===执行了')
                return this.mytext.slice(0,1).toUpperCase()+ this.mytext.slice(1)
            }
        },
        // 计算属性 ----》computed 里面写方法,以后,方法当属性用,一定要有return
        computed:{
            newText(){
                console.log('计算属性----执行了')
                return this.mytext.slice(0,1).toUpperCase() + this.mytext.slice(1)
            }
        }

    })


</script>
</html>

通过计算属性重写过滤案例

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="./js/vue.js"></script>
</head>
<body>
<div>
    <div class="app">
        <h1>过滤案例</h1>
        <p>请输入要搜索的内容:<input type="text" v-model="myText"></p>
        <ul>
            <li v-for="item in newDateList">{{item}}</li>
        </ul>

    </div>

</div>
</body>
<script>
    var vm = new Vue({
        el: '.app',
        data: {
            myText: '',
            dataList: ['a', 'at', 'atom', 'be', 'beyond', 'cs', 'csrf'],
        },
        computed: {
            newDateList() {
                return this.dataList.filter(
                    item => item.indexOf(this.myText) >= 0
                )
            }
        }

    })
</script>
</html>

监听属性

在data中定义了一些变量 只要变量发生变化 我们就执行一个函数
watch:{
    属性名(){
    }
}

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="./js/vue.js"></script>
</head>
<body>
<div>
    <div class="app">
        <!--        <span @click="handleClick(1)">Python</span>|    <span @click="handleClick(2)">Linux</span>-->
        <span @click="course_type=1">Python</span>| <span @click="course_type=2">Linux</span>
        <div>
            假设有很多课程,点击上面的标签可以完成过滤
        </div>

    </div>

</div>
</body>
<script>
    var vm = new Vue({
        el: '.app',
        data: {
            course_type: '0'
        },
        created() {
            this.getData()
        },
        methods: {
            getData() {
                // 发送ajax ,获取所有课程,通过course过滤
                // http://127.0.0.1:8080/api/v1/courses?course_type=0
            },
            // handleClick(type){
            //     this.course_type=type
            //     this.getData()
            // }
        },
        watch: {
            course_type() {
                console.log('我变化了')
                this.getData()
            }
        }

    })


</script>
</html>

node环境搭建

Vue-CLI项目搭建
    vue脚手架  可以创建vue

vue脚手架必须要按照 node js 解释型语言

     node js 是一门后端语言

     JavaScript只能运行在里浏览中,因为浏览器中有他的解释器环境
     基于谷歌浏览器的v8引擎(js解释器) 使它能够运行在操作系统上
      文件操作
      网络操作
      数据库操作  模块

   nodejs 解释器环境
   http://nodejs.cn/  下载对应平台的nodejs解释器
   安装完成会有两个可执行问题
        python    node 
        pip       npm
   打开cmd
     node  进入到了node环境
     npm install 装模块