vue 前端數據庫 組件 組件間信息交互


一、todolist留言板案例

# 前端數據庫

懸浮小手  首增尾增  前端分割切分      

"""
1) 留言就是往留言數組中添加數據,刪除留言就是從留言數組中移除數據

2) 前台數據庫:localStorage 和 sessionStorage
    localStorage永久保存數據
    sessionStorage臨時保存數據(當所屬頁面標簽被關閉,數據被清空)
    
3) 前台localStorage 和 sessionStorage數據庫存儲的值是字符串類型,所以要存放arr、dic等復雜數據需要JSON參與
"""
<!DOCTYPE html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <title></title>
    <style>
        li:hover {
            color: red;
            cursor: pointer; 
        } 
    </style>
</head>
<body>
    <div id="app">
        <p>
            <input type="text" v-model="userMsg">
            <button type="button" @click="sendMsg">留言</button>
        </p>

        <ul>
            <li v-for="(msg, index) in msgs" @click="deleteMsg(index)">  
                {{ msg }}
            </li>
        </ul>

    </div>
</body>
<script src="js/vue.js"></script>
<script>
    new Vue({
        el: '#app',
        data: {
            msgs: localStorage.msgs ? JSON.parse(localStorage.msgs) : [],  // 這是所有留言,頁面需要用的時候循環msgs,如果數據庫有值則渲染,無值為空列表
            userMsg: '',  // 留言框用戶留言,默認為空
        },
        methods: {
            sendMsg() {  // 留言事件
                // 尾增
                // this.msgs.push(this.userMsg);
                // 首增
                // this.msgs.unshift(this.userMsg);

                let userMsg = this.userMsg;  //獲取輸入框中留言內容
                if (userMsg) {  //判斷留言框是否為空,有內容才會繼續
                    this.msgs.unshift(userMsg);  // 塞入所有留言列表
 localStorage.msgs = JSON.stringify(this.msgs); // 同步到數據庫
                    this.userMsg = '';  // 清空留言框
                }
            },
            deleteMsg(index) {
                // 三個參數:開始索引  操作長度  操作的結果們(例如:(0,0,1,2,3)就是說從零位操作零位,結果變成1,2,3,點一下增加三個數)
                //這里兩個參數的意思是從索引開始操作一位結果變成空,就是刪除一個數的意思
                this.msgs.splice(index, 1)
            }
        }
    })
</script>

<script>

   //測試前端數據庫  // localStorage['num'] = 10; // sessionStorage.num = 888; // console.log(localStorage.num); // localStorage.msgs = JSON.stringify(['1111', '2222']); // console.log(JSON.parse(localStorage.msgs)); // console.log(JSON.parse(localStorage.msgs)[0]); </script> </html>

二、組件

1.組件認識

// 1) 組件:一個包含html、css、js獨立的集合體,這樣的集合體可以完成頁面解構的代碼復用,后端沒有組件概念,前端才有
// 2) 分組分為根組件、全局組件與局部組件
1.根組件:所有被new Vue()產生的組件,在項目開發階段,一個項目只會出現一個根組件,因為根組件和根組件之間不能直接進行信息交互
2.全局組件:不用注冊,就可以成為任何一個組件的子組件
3.局部組件:必須注冊,才可以成為注冊該局部組件的子組件 // 3) 每一個組件都有自身的html結構,css樣式,js邏輯
1.每一個組件其實都有自己的template,就是用來標識自己html結構的
2.template模板中有且只有一個根標簽
3.根組件一般不提供template,就由掛載點的真實DOM提供html結構 // 4) 除根組件的其他組件,數據要有局部作用域,保證組件復用時,各組件間數據的獨立性 // 5) 在多組件共處時,在哪個組件模板中出現的變量,由當前組件組件提供
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <style>
        span{color: yellowgreen;}
    </style>
</head>
<body>
<!--組件是有html模板,有CSS樣式,有js邏輯的集合體,這里先這樣寫,后期會分成三部分,分開寫-->
<!--掛載點不能用html,也不能用body,因為會被組件會被template替換掉,也就是說如果你用了body或者html他們會被替換掉,也就是說他們不再存在,這顯然不合理-->
<!--掛載點必須有,但是里面的p標簽可有可無,因為后期會被替換,掛載點div類似占位符,先用該div渲染出真實DOM后面用template內部的虛擬DOM把這里的真實DOM替換掉-->
<div id="app01"> 
    <p><h1>組件的概念</h1></p>
</div>
    
</body>
<script src="js/vue.js"></script>
<script>
    new Vue({
        el:'#app01',
        template:`
        <div style="color: red">     css樣式
            <h1>組件渲染的模板</h1>
            <span @click="lll">點我</span>
        </div>
        `,
        data:{},
        methods:{
            lll:function () {  //js邏輯
                alert(123)
            }
        }
    })
</script>
</html>

這就是組件的最終格式,有自己的HTML模板,有css樣式,還有自己的js邏輯的代碼塊,然后我們在實際開發中肯定會有好多好多的組件,這就需要用到了全局子組件和局部子組件,然后我們的template只在局部組件中使用,根組件直接使用掛載點的模板就好

2、局部組件

注意在使用局部組件的時候一定要注冊,在components中

 每個組件模板只能擁有一個根標簽

// 1) 創建局部組件
// 2) 在父組件中注冊該局部組件
// 3) 在父組件的template模板中渲染該局部組件

1.創建局部組件

 let localTag = {
        template: `
        <div class="box">
            <img src="img/666.jpg" alt="">
            <h3>鳳哥</h3>
            <p>馬叉蟲❤馬叉蟲</p>
        </div>
        `
    };
 2. 在父組件中注冊該局部組件
new Vue({
        el: '#app',
        components: {
            // mcc: localTag,  #這里冒號前什么名字,渲染的時候就用什么
            // localTag, 'local-tag': localTag,
        }
    })
3. 在父組件的template模板中渲染該局部組件
<div id="app">
    <!--<mcc></mcc>-->
    <local-tag></local-tag> #這里渲染的時候會將大寫轉換成小寫,就會報錯,要不就名字直接寫小寫,如果采取省略寫法這里就加下划線也可以等價於駝峰體
    <local-tag></local-tag>
</div>
<!DOCTYPE html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <title></title>
    <style>
        .box {
            box-shadow: 0 3px 5px 0 #666;
            width: 240px;
            height: 300px;
            text-align: center;
            padding: 20px 0;
            float: left;
            margin: 5px;
        }
        .box img {
            width: 200px;
        }
    </style>
</head>
<body>
    <div id="app">
        <!--<mcc></mcc>-->
        // 3) 在父組件的template模板中渲染該局部組件
        <local-tag></local-tag>
        <local-tag></local-tag>
    </div>
</body>
<script src="js/vue.js"></script>
<script>
    // 創建一個局部組件

    // 1) 創建局部組件
    let localTag = {
        template: `
        <div class="box">
            <img src="img/666.jpg" alt="">
            <h3>鳳哥</h3>
            <p>馬叉蟲❤馬叉蟲</p>
        </div>
        `
    };

 // 2) 在父組件中注冊該局部組件
    new Vue({
        el: '#app',
        components: {
            // mcc: localTag,
            // localTag,
            'local-tag': localTag,
        }
    })
</script>
</html>
View Code

3、全局組件

 Vue.component('全局組件名',{
        在該字典中書寫代碼塊
    })

全局組件與局部組件的區別是全局組件不需要注冊,局部組件需要注冊,我們自己在使用的時候,用全局組件會多一些,但是在一個項目 開發中使用局部組件會多!

<!DOCTYPE html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <title></title>
    <style>
        .box {
            box-shadow: 0 3px 5px 0 #666;
            width: 240px;
            height: 300px;
            text-align: center;
            padding: 20px 0;
            float: left;
            margin: 5px;
        }
        .box img {
            width: 200px;
        }
    </style>
</head>
<body>
    <div id="app">
        <global-tag></global-tag>
        <global-tag></global-tag>
        <global-tag></global-tag>
    </div>
</body>
<script src="js/vue.js"></script>
<script>
    // 1) 創建全局組件
    // 2) 在父組件的template模板中直接渲染該全局組件
    Vue.component('global-tag', {
        template: `
        <div class="box" @click="action">
            <img src="img/666.jpg" alt="">
            <h3>鳳哥</h3>
            <p>馬叉蟲❤{{ num }}</p>
        </div>
        `,
        data () {
            return {
                num: 0
            }
        },
        methods: {
            action() {
                this.num++;
            }
        }
    });

    // 數據局部化分析導入
    //如果這里直接是 a=10|b1=a|b2=a|b3=a  那么當b1或者b2或b3任意一個改變,a會改變,也就是說所有的數都改變
    
    //采用下面的函數,每一次實例化出一個a,有自己的名稱空間,互不干擾
    // a = function () {
    //     return {num: 10}
    // };
    // b1 = a();
    // b2 = a();
    // b3 = a();

    new Vue({
        el: '#app',
    })
</script>
</html>
點擊哪個哪個加一

三、組件間信息交互

知道了組件之后,我們實際中會有多個組件存在,多個組件中拿到的數據有可能是從根組件提供的,如何從根組件那里拿到數據,或者說從子組件那里拿到數據給根組件,這就是組件之間的信息交互,和我們學過的進程間通信是比較像的,我們下面來學習書寫

1.父傳子

父傳子重點就是找對信息交互的點,就是子組件的渲染標簽,

該標簽是子組件的,屬性也就是子組件的,但是屬性值由父組件提供,

子組件中通過props反射將這個值取出就拿到了數據,渲染成功

// 數據交互 - 父傳子 - 通過綁定屬性的方式
// 1) 子組件的標簽,父組件提供數據
// 2) 在父組件模板中,為子組件標簽設置自定義屬性,綁定的值由父組件提供
// 3) 在子組件實例中,通過props反射實例成員獲得自定義屬性

 

 

 

<!DOCTYPE html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <title></title>
    <style>
        .info {
            text-align: center;
            width: 200px;
            padding: 3px;
            box-shadow: 0 3px 5px 0 pink;
            float: left;
            margin: 5px;
        }
        .info img {
            width: 200px;
        }
    </style>
</head>
<body>
    <div id="app">
        <info v-for="info in infos" :key="info.image" :myinfo="info"></info>
    </div>
</body>
<script src="js/vue.js"></script>
<script>
    // 偽代碼:模擬數據從后台請求
    /*
    let infos = '';
    document.onload = function () {
        $.ajax({
            url: '/images/',
            type: 'get',
            success (response) {
                infos = response.data
            }
        })
    };
     */

    let infos = [
        {
            image: 'img/001.png',
            title: '小貓'
        },
        {
            image: 'img/002.png',
            title: '蛋糕'
        },
        {
            image: 'img/003.png',
            title: '藍糕'
        },
        {
            image: 'img/004.png',
            title: '惡犬'
        },
    ];

    let info = {
        template: `
        <div class="info">
            <img :src="myinfo.image" alt="">
            <p><b>{{ myinfo.title }}</b></p>
        </div>
        `,
        // 3) 在子組件實例中,通過props實例成員獲得自定義屬性
        props: ['myinfo']
    };

    // 數據交互 - 父傳子 - 通過綁定屬性的方式
    // 1) 父組件提供數據
    // 2) 在父組件模板中,為子組件標簽設置自定義屬性,綁定的值有父組件提供
    // 3) 在子組件實例中,通過props實例成員獲得自定義屬性
    new Vue({
        el: '#app',
        components: {
            info,
        },
        data: {
            infos,
        }
    })
</script>
</html>
View Code

2.子傳父

父級先產生,然后加載子級,父級產生的信息傳給子級,子級完成了一系列的數據操作之后傳給父級,那就是鈎子思想的運用,在子級通過emit定義一個事件,只要觸發了就傳給父級
// 組件交互-子傳父
// 1) 數據由子組件提供
// 2) 子組件內部通過觸發系統事件,發送一個自定義事件,將數據攜帶出來
// 3) 父組件位子組件標簽的自定義屬性通過方法實現,就可以通過參數拿到子組件傳遞處理的參數

<!DOCTYPE html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <title></title>
    <style>
        .close:hover {
            cursor: pointer;
            color: red;
        }
    </style>
</head>
<body>
    <div id="app">
        <p>
            <input type="text" v-model="userMsg">
            <button @click="sendMsg">留言</button>
        </p>
        <ul>
            <msg-li @remove_msg="removeAction" v-for="(msg, i) in msgs" :msg="msg" :index="i" :key="msg"></msg-li>
        </ul>
    </div>
</body>
<script src="js/vue.js"></script>
<script>
    let msgLi = {
        template: `
        <li>
            <span class="close" @click="deleteMsg(index)">x </span>
            <span>第{{ index + 1 }}條:</span>
            <span>{{ msg }}</span>
        </li>
        `,
        props: ['msg', 'index'],
        methods: {
            // 系統的click事件
            deleteMsg(i) {
                // $emit('自定義事件名', 參數們)
                this.$emit('remove_msg', i);
                this.$emit('myclick', 1, 2, 3, 4, 5)
            }
        }
    };
    // 組件交互-子傳父
    // 1) 數據由子組件提供
    // 2) 子組件內部通過觸發系統事件,發送一個自定義事件,將數據攜帶出來
    // 3) 父組件位子組件標簽的自定義屬性通過方法實現,就可以通過參數拿到子組件傳遞處理的參數

    new Vue({
        el: '#app',
        data: {
            msgs: [],
            userMsg: ''
        },
        methods: {
            sendMsg() {
                if (this.userMsg) {
                    this.msgs.push(this.userMsg);
                    this.userMsg = "";
                }
            },
            removeAction(i) {
                this.msgs.splice(i, 1)
            }
        },
        components: {
            msgLi
        }
    })
</script>
</html>
留言板案例 插件版

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM