一篇文章帶你了解網頁框架——Vue簡單入門

一篇文章帶你了解網頁框架——Vue簡單入門

這篇文章將會介紹我們前端入門級別的框架——Vue的簡單使用

如果你以後想從事後端程式設計師,又想要稍微了解前端框架知識,那麼這篇文章或許可以給你帶來幫助

溫馨提醒:在學習該文章前,請先學習HTML,CSS,JavaScript,Ajax等前端知識

Vue基礎

首先,我們從官方文檔中可以得到下述描述:

  • Vue是一套用於構建用戶介面的漸進式框架。
  • Vue的核心庫只關注視圖層,不僅易於上手,還便於與第三方庫或既有項目整合。
  • 另一方面,當與現代化的工具鏈以及各種支援類庫結合使用時,Vue 也完全能夠為複雜的單頁應用提供驅動。

其中我們需要格外注意的是:

  • Vue只負責前端頁面的設計,並不能完全實現前端的所有功能

Vue主要特點:

  • 採用JavaScript框架

  • 簡化Dom操作

  • 響應式數據驅動

VsCode插件推薦

在正式開始介紹Vue之前,我想向大家安利一個VsCode插件——Live Server

Live Server 插件可以同步程式碼和網頁的更新:

  • 當你保存程式碼時,你的頁面將會同步進行刷新,省略了手動點擊頁面的步驟,在開發中提供便利

Vue程式碼框架

首先我們需要學會Vue的基本使用

  1. 導入相關包
<!--Vue為我們提供了兩個版本,我們通常採用開發環境版本,具有檢錯功能便於學習-->

<!-- 開發環境版本,包含了有幫助的命令行警告 -->
<script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>

<!-- 生產環境版本,優化了尺寸和速度 -->
<script src="//cdn.jsdelivr.net/npm/vue"></script>
  1. 基本框架使用
<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <title>Vue基礎</title>
</head>

<body>
    
   <!--首先我們創建一個div,並綁定id為app--> 
  <div id="app">
      <!--我們採用{{}}來調用相關Vue中的data中存放的數據-->
    {{ message }}
  </div>
    
  <!-- 導入vue:開發環境版本,包含了有幫助的命令行警告 -->
  <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    
    <!--書寫js程式碼-->
  <script>
      <!-- 以下為vue基本框架 new Vue({el,data,method})-->
    var app = new Vue({
        <!--el實現掛載點-->
      el:"#app",
        <!--data存放相關數據-->
      data:{
          <!--數據屬性-->
        message:" 你好 小黑! "
      }
    })
  </script>
</body>

</html>

EL掛載點介紹

學習過Vue的基本使用後,我們先對EL掛載點做出最基本的介紹:

  • EL掛載點負責設置頁面中屬性與Vue屬性的連接
  • EL掛載點設置後,頁面屬性可以調用Vue中的數據(data)和方法(method)

EL掛載點注意點:

  • Vue的作用範圍在EL掛載點的本體元素以及後代元素中
  • Vue的EL掛載點可以依賴於各種選擇器,例如類選擇器等,但推薦使用ID選擇器
  • Vue的EL掛載點不可以作用在HTML和BODY兩個頁面最大元素上

我們給出簡單示例:

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <title>el:掛載點</title>
</head>

<body id="body">
    <!--這裡的{{ message }}不會有所顯示,因為未與Vue連接--> 
  {{ message }}
  <h2 id="app" class="app">
      <!-- 這裡的{{ message }}會顯示,因為屬於Vue連接本體--> 
    {{ message }}
      <!-- 這裡的{{ message }}會顯示,因為屬於Vue連接後代元素-->
    <span> {{ message }} </span>
  </h2>
  <!-- 開發環境版本,包含了有幫助的命令行警告 -->
  <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
  <script>
    var app = new Vue({
      // 以下均可使用,
      // el:"#app",
      // el:".app",
      // el:"div",
        
      // body無法連接
      el:"#body",
      data:{
        message:"秋落"
      }
    })
  </script>
</body>

</html>

Data數據對象介紹

學習過Vue的基本使用後,我們對Data做出最基本的介紹:

  • Data用於存儲頁面元素中使用的各類屬性
  • 屬性可以包括各種類型,例如文本,數組等複雜類型
  • 渲染複雜類型數據時,遵循JavaScript的語法基本都可以使用

我們給出簡單示例:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>data:數據對象</title>
</head>

<body>
    <div id="app">
        <!--使用{{}}調用data數據-->
        {{ message }}
        <!--對象採用.來調用內部元素-->
        <h2> {{ school.name }} {{ school.mobile }}</h2>
        <ul>
            <!--數組採用[index]來選擇內部元素-->
            <li>{{ campus[0] }}</li>
            <li>{{ campus[3] }}</li>
        </ul>
    </div>
    <!-- 開發環境版本,包含了有幫助的命令行警告 -->
    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    <script>
        var app = new Vue({
            el:"#app",
            data:{
                message:"你好!",
                school:{
                    name:"河南師範大學",
                    mobile:"0373-3325000"
                },
                campus:["東校區","西校區","新鄉校區","平原湖校區"]
            }
        })
    </script>
</body>

</html>

Vue本地應用

在介紹了Vue的基本使用後,我們針對Vue內部的各種屬性方法來做出一一介紹

每條屬性或方法我們都會給出解釋和相關案例

v-text

文本解釋:

v-text:設置標籤的文本值

程式碼解釋:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>v-text指令</title>
</head>

<body>
    <div id="app">
        <!--我們採用v-text來設置標籤內文本,注意會覆蓋掉內部文本,「深圳」將不再顯示-->
        <h2 v-text="message+'!'">深圳</h2>
        <h2 v-text="info+'!'">深圳</h2>
        <!--另一種文本書寫方式{{}}-->
        <h2>{{ message +'!'}}深圳</h2>
    </div>
    <!-- 開發環境版本,包含了有幫助的命令行警告 -->
    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    <script>
        var app = new Vue({
            el:"#app",
            data:{
                message:"河南師範大學",
                info:"軟體學院"
            }
        })
    </script>
</body>

</html>

v-html

文本解釋:

v-html:以html的格式來設置標籤的文本值

程式碼解釋:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>v-html指令</title>
</head>

<body>
    <!--v-html:設置元素的innerHTML-->
    
    <!--採用v-html後顯示的是河南師範大學的加粗版,但採用v-text後顯示的是<strong>河南師範大學</strong>-->
    <div id="app" v-html="context">
    	
    </div>
    <!-- 開發環境版本,包含了有幫助的命令行警告 -->
    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    <script>
        var app = new Vue({
            el:"#app",
            data:{
                content:"<strong>河南師範大學</strong>"
            }
        })
    </script>
</body>

</html>

v-on

文本解釋:

v-on:為元素綁定事件,可以採用@代替

程式碼解釋:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>v-on</title>
</head>
<body>
    <div id="app">
        <!--v-on:後面跟事件名 用於綁定事件 可以用@代替 後面用methods中的方法即可-->
        <input type="button" value="點擊" v-on:click="doIt">
        <input type="button" value="點擊" @click="doIt">
    </div>
    

    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    <script>
        var app = new Vue({
            el:"#app",
            methods:{
                doIt:function(){
                    alert("河南師範大學");
                }
            }
        })
    </script>
</body>
</html>

v-show

文本解釋:

v-show:根據表達值的真假,切換元素的顯示和隱藏(隱藏後源程式碼仍存在)

程式碼解釋:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>v-show指令</title>
  </head>
  <body>
    <div id="app">
        <!--點擊後改變isShow的值,將isShow變為true或false,以此控制img的顯示與隱藏-->
      <input type="button" value="切換顯示狀態" @click="changeIsShow">
        <!--點擊增加年齡值-->
      <input type="button" value="累加年齡" @click="addAge">
        <!--v-show用於控制元素的存在或隱藏,這裡採用isShow變數,根據isShow的值來判斷是否存在-->
      <img v-show="isShow" src="./img/monkey.gif" alt="">
        <!--根據年齡值大小控制元素的存在或隱藏,只是為了表示v-show的參數需要是一個true或false的判定-->
      <img v-show="age>=18" src="./img/monkey.gif" alt="">
    </div>
    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    <script>
      var app = new Vue({
        el:"#app",
        data:{
          isShow:false,
          age:17
        },
        methods: {
            // 進行 isShow 的true或false更改
          changeIsShow:function(){
            this.isShow = !this.isShow;
          },
          addAge:function(){
            this.age++;
          }
        },
      })
    </script>
  </body>
</html>

v-if

文本解釋:

v-if:根據表達值的真假,切換元素的顯示和隱藏(隱藏後,源程式碼不存在)

程式碼解釋:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>v-if指令</title>
</head>
<body>
    <div id="app">
        <!--以下操作實際效果同v-show相同,但隱藏後在頁面的展示欄中無法看到源程式碼,屬於徹底隱藏-->
        <input type="button" value="切換顯示" @click="toggleIsShow">
        <p v-if="isShow">河南師範大學</p>
        <p v-show="isShow">河南師範大學 - v-show修飾</p>
        <h2 v-if="temperature>=35">熱死啦</h2>
    </div>
    <!-- 開發環境版本,包含了有幫助的命令行警告 -->
    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    <script>
        var app = new Vue({
            el:"#app",
            data:{
                isShow:false,
                temperature:20
            },
            methods: {
                toggleIsShow:function(){
                    this.isShow = !this.isShow;
                }
            },
        })
    </script>
</body>

</html>

v-bind

文本解釋:

v-bind:設置元素的屬性(比如:src,title。class)

程式碼解釋:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>v-bind指令</title>
    <style>
        .active{
            border: 1px solid red;
        }
    </style>
</head>

<body>
    <!--v-bind通常可以隱藏,直接寫 :屬性 即可-->
    
    <div id="app">
        <!--這裡採用v-bind設置頁面圖片-->
        <img v-bind:src="imgSrc" alt="">
        <br>
        <!--這裡採用v-bind略寫方式設置頁面圖片,後面採用 data 變數控制圖片展示-->
        <!--同樣採用v-bind控制title,class等屬性,採用三元運算符來控制active-->
        <img :src="imgSrc" alt="" :title="imgTitle+'!!!'" :class="isActive?'active':''" @click="toggleActive">
        <br>
        <!--通過點擊事件來控制class-->
        <img :src="imgSrc" alt="" :title="imgTitle+'!!!'" :class="{active:isActive}" @click="toggleActive">
    </div>
    <!-- 開發環境版本,包含了有幫助的命令行警告 -->
    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    <script>
        var app = new Vue({
            el:"#app",
            data:{
                imgSrc:"//www.itheima.com/images/logo.png",
                imgTitle:"黑馬程式設計師",
                isActive:false
            },
            methods: {
                toggleActive:function(){
                    this.isActive = !this.isActive;
                }
            },
        })
    </script>
</body>

</html>

v-for

文本解釋:

v-for:根據數據生成列表結構

程式碼解釋:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>v-for指令</title>
</head>

<body>
    <div id="app">
        
        <!--簡單的添加刪除操作,針對h2的數據-->
        <input type="button" value="添加數據" @click="add">
        <input type="button" value="移除數據" @click="remove">

        <ul>
            <!--簡單v-for,參數為(數據名稱[任起],下標[index]) in data數組-->
            <li v-for="(it,index) in arr">
                <!--在內部,可以使用數據名稱和下標值-->
                {{ index+1 }}城市推薦:{{ it }}
            </li>
        </ul>
        
        <!--數組中裝有對象也是同樣的使用方法-->
        <h2 v-for="item in vegetables" v-bind:title="item.name">
            {{ item.name }}
        </h2>
        
    </div>

    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    
    <script>
        var app = new Vue({
            el:"#app",
            data:{
                arr:["北京","上海","廣州","深圳"],
                vegetables:[
                    {name:"魚"},
                    {name:"雞"}
                ]
            },
            methods: {
                add:function(){
                    this.vegetables.push({ name:"紅燒魚" });
                },
                remove:function(){
                    this.vegetables.shift();
                }
            },
        })
    </script>
</body>

</html>

v-on+

文本解釋:

v-on+:補充v-on的部分知識點

程式碼解釋:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>v-on補充</title>
</head>

<body>
    <div id="app">
        <!--方法中可以攜帶參數,正常調用即可-->
        <input type="button" value="點擊" @click="doIt(666,'老鐵')">
        <!--可以選取事件的部分事件做反應,例如@keyup.enter就是點擊enter鍵生效,我們通常採用"."來表示事件限制-->
        <input type="text" @keyup.enter="sayHi">
    </div>

    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    
    <script>
        var app = new Vue({
            el:"#app",
            methods: {
                doIt:function(p1,p2){
                    console.log("做it");
                    console.log(p1);
                    console.log(p2);
                },
                sayHi:function(){
                    alert("吃了沒");
                }
            },
        })
    </script>
</body>

</html>

v-model

文本解釋:

v-model:實現雙向綁定,便捷設置

程式碼解釋:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>v-model指令</title>
</head>

<body>
    <div id="app">
        <!--點擊按鈕實現setM方法-->
        <input type="button" value="修改message" @click="setM">
        <!--我們將文本內容與message數據雙向綁定-->
        <!--當我們修改文本的值時,VsCode中的程式碼不會發生變化,但實際上message已經發生變化,我們將message的值單獨列出來-->
        <input type="text" v-model="message" @keyup.enter="getM">
        
        <!--當上述message發生改變時,message本身發生變化,那麼調用它的顯示值也發生變化-->
        <h2>{{ message }}</h2>
    </div>

    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    
    <script>
        var app = new Vue({
            el:"#app",
            data:{
                message:"河南師範大學"
            },
            methods: {
                getM:function(){
                    alert(this.message);
                },
                setM:function(){
                    this.message ="軟體學院";
                }
            },
        })
    </script>
</body>

</html>

案例:計算器

下面我們通過一個案例來鞏固前面所學習的一些知識點

案例要求:

  • 我們希望通過”-“和”+”控制中間數字的大小(最小為0,最大為10)

知識點複習:

  • EL掛載點,Data數據,Methods方法
  • v-on,v-text方法

程式碼展示:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>計算器</title>
      
      <!--css配置-->
    <style>
      #app {
        width: 480px;
        height: 100px;
        margin: 200px auto;
      }
      .input-num {
        margin-top: 20px;
        height: 100%;
        display: flex;
        border-radius: 10px;
        overflow: hidden;
        box-shadow: 0 0 4px black;
      }
      .input-num button {
        width: 150px;
        height: 100%;
        font-size: 40px;
        color: gray;
        cursor: pointer;
        border: none;
        outline: none;
      }
      .input-num span {
        height: 100%;
        font-size: 40px;
        flex: 1;
        text-align: center;
        line-height: 100px;
      }
    </style>
      
  </head>
    
  <body>
      <!--index主頁-->
      
      <!--設置Vue的id綁定-->
    <div id="app">
      <img src="//www.itheima.com/images/logo.png" alt="" />
      <!-- 計數器 -->
      <div class="input-num">
          <!--綁定事件sub-->
        <button @click="sub"> - </button>
          <!--數字展示-->
        <span>{{ num }}</span>
          <!--綁定事件add-->
        <button @click="add"> + </button>
      </div>
    </div>
  </body>
    
</html>

<script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>

<!-- 編碼 -->
<script>
  /*
    1. data中定義num屬性,類型是數字,渲染到2個按鈕中間
    2. 減號綁定點擊事件,對應的方法名叫sub,大於0之前遞減
    3. 加號綁定點擊事件,對應的方法名叫add,小於0之前累加
  */
  // 創建Vue實例
  var app = new Vue({
    el: "#app",
    data: {
      // 修改的數字
      num:1
    },
    methods: {
      // 減
      sub:function(){
        // 遞減
        if(this.num>0){
          this.num--;
        }else{
          alert("數字最小為0");
        }
      },
      // 加
      add:function(){
        // 累加
        if(this.num<10){
          this.num++;
        }else{
          alert("數字最大為10");
        }
      }
    }
  });
</script>

案例:圖片切換

下面我們通過一個案例來鞏固前面所學習的一些知識點

案例要求:

  • 我們通過點擊左右兩邊的圖片切換符號來切換中間圖片

知識點複習:

  • EL掛載點,Data數據,Methods方法
  • v-on,v-show,v-bind方法

程式碼展示:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>Document</title>
      <!--css樣式在下面單列-->
    <link rel="stylesheet" href="./css/index.css" />
  </head>

  <body>
    <div id="mask">
      <div class="center">
        <h2 class="title"><img src="./images/logo.png" alt=""> 學校環境 </h2>
          
          <!--展示圖片,切換的中心點-->
        <img :src="imgList[index]" alt="" />
          
          <!--左轉圖片-->
          <!--添加綁定事件prev,用於圖片的切換-->
          <!--添加v-show,當左側無圖片時,隱藏-->
        <a
          href="javascript:void(0)"
          @click="prev"
          class="left"
          v-show="index>0"
        >
          <img src="./images/prev.png" alt="" />
        </a>
          
          <!--右轉圖片-->
          <!--添加綁定事件right,用於圖片的切換-->
          <!--添加v-show,當右側無圖片時,隱藏-->
        <a
          href="javascript:void(0)"
          @click="next"
          class="right"
          v-show="index<imgList.length-1"
        >
          <img src="./images/next.png" alt="" />
        </a>
          
      </div>
    </div>

    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
   
    <script>
      const app = new Vue({
        el: "#mask",
        data: {
            // 圖片元素
          imgList: [
            "./images/00.jpg",
            "./images/01.jpg",
            "./images/02.jpg",
            "./images/03.jpg",
            "./images/04.jpg",
            "./images/05.jpg",
            "./images/06.jpg",
            "./images/07.jpg",
            "./images/08.jpg",
            "./images/09.jpg",
            "./images/10.jpg",
          ],
            // 圖片序號
          index: 0
        },
        methods: {
          // 上一張
          prev() {
            this.index--;
          },
          // 下一張
          next() {
            this.index++;
          }
        }
      });
    </script>
  </body>
</html>
* {
  margin: 0;
  padding: 0;
}

html,
body,
#mask {
  width: 100%;
  height: 100%;
}

#mask {
  background-color: #c9c9c9;
  position: relative;
}

#mask .center {
  position: absolute;
  background-color: #fff;
  left: 50%;
  top: 50%;
  transform: translate(-50%, -50%);
  padding: 10px;
}
#mask .center .title {
  position: absolute;
  display: flex;
  align-items: center;
  height: 56px;
  top: -61px;
  left: 0;
  padding: 5px;
  padding-left: 10px;
  padding-bottom: 0;
  color: rgba(175, 47, 47, 0.8);
  font-size: 26px;
  font-weight: normal;
  background-color: white;
  padding-right: 50px;
  z-index: 2;
}
#mask .center .title img {
  height: 40px;
  margin-right: 10px;
}

#mask .center .title::before {
  content: "";
  position: absolute;
  width: 0;
  height: 0;
  border: 65px solid;
  border-color: transparent transparent white;
  top: -65px;
  right: -65px;
  z-index: 1;
}

#mask .center > img {
  display: block;
  width: 700px;
  height: 458px;
}

#mask .center a {
  text-decoration: none;
  width: 45px;
  height: 100px;
  position: absolute;
  top: 179px;
  vertical-align: middle;
  opacity: 0.5;
}
#mask .center a :hover {
  opacity: 0.8;
}

#mask .center .left {
  left: 15px;
  text-align: left;
  padding-right: 10px;
  border-top-right-radius: 10px;
  border-bottom-right-radius: 10px;
}

#mask .center .right {
  right: 15px;
  text-align: right;
  padding-left: 10px;
  border-top-left-radius: 10px;
  border-bottom-left-radius: 10px;
}

案例:記事本

下面我們通過一個案例來鞏固前面所學習的全部知識點

案例要求:

  • 新增:書寫內容,點擊Enter後,便利貼放於頁面
  • 刪除:點擊內容後方的”x”號後,該內容消失
  • 統計:統計內容數量,在左下角顯示
  • 清空:點擊右下角清空鍵,內容全部清空
  • 隱藏:當無內容時,下述記事本部分隱藏

程式碼展示:

<html>

<head>
  <meta http-equiv="content-type" content="text/html; charset=UTF-8" />
  <title>小黑記事本</title>
  <meta http-equiv="content-type" content="text/html; charset=UTF-8" />
  <meta name="robots" content="noindex, nofollow" />
  <meta name="googlebot" content="noindex, nofollow" />
  <meta name="viewport" content="width=device-width, initial-scale=1" />
    <!--css樣式在下列出-->
  <link rel="stylesheet" type="text/css" href="./css/index.css" />
</head>

<body>
  <!-- 主體區域 -->
  <section id="todoapp">
    <!-- 輸入框 -->
    <header class="header">
      <h1>小黑記事本</h1>
        <!--輸入欄,實現雙向綁定,當點擊時使用add方法-->
      <input v-model="inputValue" @keyup.enter="add" autofocus="autofocus" autocomplete="off" placeholder="請輸入任務"
        class="new-todo" />
    </header>
    <!-- 列表區域 -->
    <section class="main">
      <ul class="todo-list">
          <!--記事主題內容,採用v-for遍歷記事內容list的內容,輸出在頁面-->
        <li class="todo" v-for="(item,index) in list">
          <div class="view">
              <!--記事內容-->
            <span class="index">{{ index+1 }}.</span>
            <label>{{ item }}</label>
              <!--刪除按鈕,點擊時觸發remove功能,參數為當前index-->
            <button class="destroy" @click="remove(index)"></button>
          </div>
        </li>
      </ul>
    </section>
    <!-- 統計和清空 -->
      <!--記事本主體,當不存在記事內容時,隱藏-->
    <footer class="footer" v-show="list.length!=0">
      <span class="todo-count" v-if="list.length!=0">
          <!--統計值:直接採用list的長度代替-->
        <strong>{{ list.length }}</strong> items left
      </span>
        <!--記事本主體,當不存在記事內容時,隱藏-->
      <button v-show="list.length!=0" class="clear-completed" @click="clear">
        Clear
      </button>
    </footer>
  </section>
  <!-- 底部 -->
  <footer class="info">
    <p>
      <a href="//www.itheima.com/"><img src="./img/black.png" alt="" /></a>
    </p>
  </footer>
    
    
  <!-- 開發環境版本,包含了有幫助的命令行警告 -->
  <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    
    
  <script>
    var app = new Vue({
        // 實現Vue綁定
      el: "#todoapp",
      data: {
          // list表示記事內容
        list: ["寫程式碼", "吃飯飯", "睡覺覺"],
          // 表示目前輸入內容,雙向綁定
        inputValue: "好好學習,天天向上"
      },
      methods: {
          // 新添方法,將輸入值填入計時內容中
        add: function () {
          this.list.push(this.inputValue);
        },
          // 刪除方法,刪除當前下表為index的內容
        remove: function (index) {
          this.list.splice(index, 1);
        },
          // 清除方法,清除所有方法
        clear: function () {
          this.list = [];
        }
      },
    })
  </script>
    
</body>

</html>
html,
body {
  margin: 0;
  padding: 0;
}
body {
  background: #fff;
}
button {
  margin: 0;
  padding: 0;
  border: 0;
  background: none;
  font-size: 100%;
  vertical-align: baseline;
  font-family: inherit;
  font-weight: inherit;
  color: inherit;
  -webkit-appearance: none;
  appearance: none;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
}

body {
  font: 14px "Helvetica Neue", Helvetica, Arial, sans-serif;
  line-height: 1.4em;
  background: #f5f5f5;
  color: #4d4d4d;
  min-width: 230px;
  max-width: 550px;
  margin: 0 auto;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
  font-weight: 300;
}

:focus {
  outline: 0;
}

.hidden {
  display: none;
}

#todoapp {
  background: #fff;
  margin: 180px 0 40px 0;
  position: relative;
  box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.2), 0 25px 50px 0 rgba(0, 0, 0, 0.1);
}

#todoapp input::-webkit-input-placeholder {
  font-style: italic;
  font-weight: 300;
  color: #e6e6e6;
}

#todoapp input::-moz-placeholder {
  font-style: italic;
  font-weight: 300;
  color: #e6e6e6;
}

#todoapp input::input-placeholder {
  font-style: italic;
  font-weight: 300;
  color: gray;
}

#todoapp h1 {
  position: absolute;
  top: -160px;
  width: 100%;
  font-size: 60px;
  font-weight: 100;
  text-align: center;
  color: rgba(175, 47, 47, .8);
  -webkit-text-rendering: optimizeLegibility;
  -moz-text-rendering: optimizeLegibility;
  text-rendering: optimizeLegibility;
}

.new-todo,
.edit {
  position: relative;
  margin: 0;
  width: 100%;
  font-size: 24px;
  font-family: inherit;
  font-weight: inherit;
  line-height: 1.4em;
  border: 0;
  color: inherit;
  padding: 6px;
  border: 1px solid #999;
  box-shadow: inset 0 -1px 5px 0 rgba(0, 0, 0, 0.2);
  box-sizing: border-box;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
}

.new-todo {
  padding: 16px;
  border: none;
  background: rgba(0, 0, 0, 0.003);
  box-shadow: inset 0 -2px 1px rgba(0, 0, 0, 0.03);
}

.main {
  position: relative;
  z-index: 2;
  border-top: 1px solid #e6e6e6;
}

.toggle-all {
  width: 1px;
  height: 1px;
  border: none; /* Mobile Safari */
  opacity: 0;
  position: absolute;
  right: 100%;
  bottom: 100%;
}

.toggle-all + label {
  width: 60px;
  height: 34px;
  font-size: 0;
  position: absolute;
  top: -52px;
  left: -13px;
  -webkit-transform: rotate(90deg);
  transform: rotate(90deg);
}

.toggle-all + label:before {
  content: "❯";
  font-size: 22px;
  color: #e6e6e6;
  padding: 10px 27px 10px 27px;
}

.toggle-all:checked + label:before {
  color: #737373;
}

.todo-list {
  margin: 0;
  padding: 0;
  list-style: none;
  max-height: 420px;
  overflow: auto;
}

.todo-list li {
  position: relative;
  font-size: 24px;
  border-bottom: 1px solid #ededed;
  height: 60px;
  box-sizing: border-box;
}

.todo-list li:last-child {
  border-bottom: none;
}

.todo-list .view .index {
  position: absolute;
  color: gray;
  left: 10px;
  top: 20px;
  font-size: 16px;
}

.todo-list li .toggle {
  text-align: center;
  width: 40px;
  /* auto, since non-WebKit browsers doesn't support input styling */
  height: auto;
  position: absolute;
  top: 0;
  bottom: 0;
  margin: auto 0;
  border: none; /* Mobile Safari */
  -webkit-appearance: none;
  appearance: none;
}

.todo-list li .toggle {
  opacity: 0;
}

.todo-list li .toggle + label {
  /*
		Firefox requires `#` to be escaped - //bugzilla.mozilla.org/show_bug.cgi?id=922433
		IE and Edge requires *everything* to be escaped to render, so we do that instead of just the `#` - //developer.microsoft.com/en-us/microsoft-edge/platform/issues/7157459/
	*/
  background-image: url("data:image/svg+xml;utf8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%2240%22%20height%3D%2240%22%20viewBox%3D%22-10%20-18%20100%20135%22%3E%3Ccircle%20cx%3D%2250%22%20cy%3D%2250%22%20r%3D%2250%22%20fill%3D%22none%22%20stroke%3D%22%23ededed%22%20stroke-width%3D%223%22/%3E%3C/svg%3E");
  background-repeat: no-repeat;
  background-position: center left;
}

.todo-list li .toggle:checked + label {
  background-image: url("data:image/svg+xml;utf8,%3Csvg%20xmlns%3D%22http%3A//www.w3.org/2000/svg%22%20width%3D%2240%22%20height%3D%2240%22%20viewBox%3D%22-10%20-18%20100%20135%22%3E%3Ccircle%20cx%3D%2250%22%20cy%3D%2250%22%20r%3D%2250%22%20fill%3D%22none%22%20stroke%3D%22%23bddad5%22%20stroke-width%3D%223%22/%3E%3Cpath%20fill%3D%22%235dc2af%22%20d%3D%22M72%2025L42%2071%2027%2056l-4%204%2020%2020%2034-52z%22/%3E%3C/svg%3E");
}

.todo-list li label {
  word-break: break-all;
  padding: 15px 15px 15px 60px;
  display: block;
  line-height: 1.2;
  transition: color 0.4s;
}

.todo-list li.completed label {
  color: #d9d9d9;
  text-decoration: line-through;
}

.todo-list li .destroy {
  display: none;
  position: absolute;
  top: 0;
  right: 10px;
  bottom: 0;
  width: 40px;
  height: 40px;
  margin: auto 0;
  font-size: 30px;
  color: #cc9a9a;
  margin-bottom: 11px;
  transition: color 0.2s ease-out;
}

.todo-list li .destroy:hover {
  color: #af5b5e;
}

.todo-list li .destroy:after {
  content: "×";
}

.todo-list li:hover .destroy {
  display: block;
}

.todo-list li .edit {
  display: none;
}

.todo-list li.editing:last-child {
  margin-bottom: -1px;
}

.footer {
  color: #777;
  padding: 10px 15px;
  height: 20px;
  text-align: center;
  border-top: 1px solid #e6e6e6;
}

.footer:before {
  content: "";
  position: absolute;
  right: 0;
  bottom: 0;
  left: 0;
  height: 50px;
  overflow: hidden;
  box-shadow: 0 1px 1px rgba(0, 0, 0, 0.2), 0 8px 0 -3px #f6f6f6,
    0 9px 1px -3px rgba(0, 0, 0, 0.2), 0 16px 0 -6px #f6f6f6,
    0 17px 2px -6px rgba(0, 0, 0, 0.2);
}

.todo-count {
  float: left;
  text-align: left;
}

.todo-count strong {
  font-weight: 300;
}

.filters {
  margin: 0;
  padding: 0;
  list-style: none;
  position: absolute;
  right: 0;
  left: 0;
}

.filters li {
  display: inline;
}

.filters li a {
  color: inherit;
  margin: 3px;
  padding: 3px 7px;
  text-decoration: none;
  border: 1px solid transparent;
  border-radius: 3px;
}

.filters li a:hover {
  border-color: rgba(175, 47, 47, 0.1);
}

.filters li a.selected {
  border-color: rgba(175, 47, 47, 0.2);
}

.clear-completed,
html .clear-completed:active {
  float: right;
  position: relative;
  line-height: 20px;
  text-decoration: none;
  cursor: pointer;
}

.clear-completed:hover {
  text-decoration: underline;
}

.info {
  margin: 50px auto 0;
  color: #bfbfbf;
  font-size: 15px;
  text-shadow: 0 1px 0 rgba(255, 255, 255, 0.5);
  text-align: center;
}

.info p {
  line-height: 1;
}

.info a {
  color: inherit;
  text-decoration: none;
  font-weight: 400;
}

.info a:hover {
  text-decoration: underline;
}

/*
	Hack to remove background from Mobile Safari.
	Can't use it globally since it destroys checkboxes in Firefox
*/
@media screen and (-webkit-min-device-pixel-ratio: 0) {
  .toggle-all,
  .todo-list li .toggle {
    background: none;
  }

  .todo-list li .toggle {
    height: 40px;
  }
}

@media (max-width: 430px) {
  .footer {
    height: 50px;
  }

  .filters {
    bottom: 10px;
  }
}

Vue網路應用

我們在本篇開頭說過Vue的作用僅僅是用來完成靜態頁面

所以如果想要完成項目開發功能,還需要與後台交互的技術Ajax(主要採用Axios技術)

Axios技術

Axios技術是原生的Ajax進行封裝,簡化書寫

我們在之前的Ajax專題中有完整的介紹過Ajax和Axios技術,在這裡我們僅做簡單回憶:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>axios基本使用</title>
</head>

<body>
    
    <!--兩個按鈕分別用來實現兩種數據獲取-->
    <input type="button" value="get請求" class="get">
    <input type="button" value="post請求" class="post">
    
    <!-- 官網提供的 axios 在線地址 -->
    <script src="//unpkg.com/axios/dist/axios.min.js"></script>
    <script>
        /*
            介面1:隨機笑話
            請求地址://autumnfish.cn/api/joke/list
            請求方法:get
            請求參數:num(笑話條數,數字)
            響應內容:隨機笑話
        */
        
        // 事件綁定
        document.querySelector(".get").onclick = function () {
            // axios的get格式:get後跟url,response為返回體,err為錯誤
            axios.get("//autumnfish.cn/api/joke/list?num=6")
            .then(function (response) {
                console.log(response);
              },function(err){
                  console.log(err);
              })
        }
        /*
             介面2:用戶註冊
             請求地址://autumnfish.cn/api/user/reg
             請求方法:post
             請求參數:username(用戶名,字元串)
             響應內容:註冊成功或失敗
         */
        
        // 事件綁定
        document.querySelector(".post").onclick = function () {
            // axios的post格式:post後跟url和請求體,response為返回體,err為錯誤
            axios.post("//autumnfish.cn/api/user/reg",{username:"鹽焗西蘭花"})
            .then(function(response){
                console.log(response);
                console.log(this.skill);
            },function (err) {
                console.log(err);
              })
          }

    </script>
</body>

</html>

Axios+Vue技術

我們常常用Vue作為頁面的設計框架,同時採用Axios作為前後端交互的技術

兩者的結合其實並沒有互相摻雜,大致上還保留原本的形式,最大的改變只有數據來源發生變化

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>axios+vue</title>
</head>

<body>
    
    <div id="app">
        <input type="button" value="獲取笑話" @click="getJoke">
        <p> {{ joke }}</p>
    </div>
    
    <!-- 官網提供的 axios 在線地址 -->
    <script src="//unpkg.com/axios/dist/axios.min.js"></script>
    <!-- 開發環境版本,包含了有幫助的命令行警告 -->
    <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
    
    <script>
        /*
            介面:隨機獲取一條笑話
            請求地址://autumnfish.cn/api/joke
            請求方法:get
            請求參數:無
            響應內容:隨機笑話
        */
        
        // 採用Vue開啟框架頁面
        var app = new Vue({
            el:"#app",
            data:{
                // 頁面數據展示
                joke:"很好笑的笑話"
            },
            methods: {
                // 獲得笑話的方法,採用axios技術進行數據請求
                getJoke:function(){
                    var that = this;
                    axios.get("//autumnfish.cn/api/joke").then(function(response){
                        console.log(response.data);
                        that.joke = response.data;
                    },function (err) {  })
                }
            },
        })
    </script>
    
</body>

</html>

案例:天氣預報

我們同樣採用一個簡單的案例來鞏固Vue網路應用

案例需求:

  • 我們可以手動查找輸入框內的城市的天氣情況
  • 我們可以點擊頁面內含有的城市的天氣情況

程式碼展示:

<!--主頁面展示-->

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8" />
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
  <meta http-equiv="X-UA-Compatible" content="ie=edge" />
  <title>天知道</title>
    <!--CSS鏈接-->
  <link rel="stylesheet" href="css/index.css" />
</head>

<body>
    <!--Vue綁定-->
  <div class="wrap" id="app">
    <div class="search_form">
      <div class="logo"><img src="img/logo.png" alt="logo" /></div>
      <div class="form_group">
          <!--雙向綁定city,添加事件@keyup.enter="queryWeather"-->
        <input type="text" class="input_txt" placeholder="請輸入查詢的天氣" v-model="city" @keyup.enter="queryWeather" />
          <!--添加事件@keyup.enter="queryWeather"-->
        <button class="input_sub" @click="queryWeather">
          搜 索
        </button>
      </div>
        
        <!--展示列出的城市,點擊觸發事件-->
      <div class="hotkey">
        <a href="javascript:;" v-for="city in hotCitys" @click="clickSearch(city)">{{ city }}</a>
      </div>
        
        <!--展示返回的天氣數據-->
    </div>
    <ul class="weather_list">
        <!---->
      <li v-for="(item,index) in forecastList" :key="item.date" :style="{transitionDelay:index*100+'ms'}">
        <div class="info_type">
          <span class="iconfont">{{ item.type }}</span>
        </div>
        <div class="info_temp">
          <b>{{ item.low}}</b>
          ~
          <b>{{ item.high}}</b>

        </div>
          
        <div class="info_date">
          <span>{{ item.date }}</span>
        </div>
          
      </li>
    </ul>
  </div>
    
  <!-- 開發環境版本,包含了有幫助的命令行警告 -->
  <script src="//cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
  <!-- 官網提供的 axios 在線地址 -->
  <script src="//unpkg.com/axios/dist/axios.min.js"></script>
    
  <script>
    new Vue({
      el: "#app",
      data: {
          // 輸入框城市,雙向綁定
        city: "武漢",
          // 天氣情況
        forecastList: [],
          // 城市展示
        hotCitys: ["北京", "上海", "廣州", "深圳"]
      },
      methods: {
        queryWeather() {
            // 將天氣情況清零
          this.forecastList = [];
            // axios獲得天氣狀況
          axios
            .get(`//wthrcdn.etouch.cn/weather_mini?city=${this.city}`)
            .then(res => {
              console.log(res);
              this.forecastList = res.data.data.forecast;
            })
            .catch(err => {
              console.log(err);
            })
            .finally(() => { });
        },
          // 點擊城市觸發queryWeather方法,獲得天氣情況
        clickSearch(city) {
          this.city = city;
          this.queryWeather();
        }
      }
    });
  </script>
</body>

</html>
/*
css展示
頁面前置css修改
*/

body,ul,h1,h2,h3,h4,h5,h6{
    margin: 0;
    padding: 0;
}
h1,h2,h3,h4,h5,h6{
    font-size:100%;
    font-weight:normal;
}
a{
    text-decoration:none;
}
ul{
    list-style:none;
}
img{
    border:0px;
}

/* 清除浮動,解決margin-top塌陷 */
.clearfix:before,.clearfix:after{
    content:'';
    display:table;    
}
.clearfix:after{
    clear:both;
}
.clearfix{
    zoom:1;
}

.fl{
    float:left;
}
.fr{
    float:right;
}

/*
css展示
頁面內部css修改
*/

body{
    font-family:'Microsoft YaHei';   
}
.wrap{
    position: fixed;
    left:0;
    top:0;
    width:100%;
    height:100%;
    /* background: radial-gradient(#f3fbfe, #e4f5fd, #8fd5f4); */
    /* background:#8fd5f4; */
    /* background: linear-gradient(#6bc6ee, #fff); */
    background:#fff;

}
.search_form{
    width:640px;
    margin:100px auto 0;
}
.logo img{
    display:block;
    margin:0 auto;
}
.form_group{
    width:640px;
    height:40px;
    margin-top:45px;
}
.input_txt{
   width:538px;
   height:38px;
   padding:0px;
   float:left;
   border:1px solid #41a1cb;
   outline:none;
   text-indent:10px;
}

.input_sub{
    width:100px;
    height:40px;
    border:0px;
    float: left;
    background-color: #41a1cb;
    color:#fff;
    font-size:16px;
    outline:none;
    cursor: pointer;
    position: relative;
}
.input_sub.loading::before{
    content:'';
    position: absolute;
    left: 0;
    top: 0;
    width: 100%;
    height: 100%;
    background: url('../img/loading.gif');
}

.hotkey{
    margin:3px 0 0 2px;
}

.hotkey a{
    font-size:14px;
    color:#666;
    padding-right:15px;
}
.weather_list{
    height:200px;
    text-align:center;
    margin-top:50px;
    font-size:0px;
}
.weather_list li{
    display:inline-block;
    width:140px;
    height:200px;
    padding:0 10px;
    overflow: hidden;
    position: relative;
    background:url('../img/line.png') right center no-repeat;
    background-size: 1px 130px;
}

.weather_list li:last-child{
    background:none;
}

/* .weather_list .col02{
    background-color: rgba(65, 165, 158, 0.8);
}
.weather_list .col03{
    background-color: rgba(94, 194, 237, 0.8);
}
.weather_list .col04{
    background-color: rgba(69, 137, 176, 0.8);
}
.weather_list .col05{
    background-color: rgba(118, 113, 223, 0.8);
} */


.info_date{
    width:100%;
    height:40px;
    line-height:40px;
    color:#999;
    font-size:14px;
    left:0px;    
    bottom:0px;    
    margin-top: 15px;
}
.info_date b{
    float: left;
    margin-left:15px;
}

.info_type span{
    color:#fda252;
    font-size:30px;
    line-height:80px;
}
.info_temp{
    font-size:14px;  
    color:#fda252;
}
.info_temp b{
    font-size:13px;
}
.tem .iconfont {
    font-size: 50px;
  }

結束語

好的,關於Vue入門案例的內容就介紹到這裡,希望能為你帶來幫助!

附錄

該文章屬於學習內容,具體參考B站黑馬程式設計師vue前端基本教程

這裡附上鏈接:01.課程介紹_嗶哩嗶哩_bilibili