Vue基础

JavaScript基础知识

  1. ES6语法规范(结构赋值、模板字符串、箭头函数)
  2. ES6模块化(默认、分别、统一暴露)
  3. 包管理器(npm、yarn)
  4. 原型、原型链
  5. 数组常用方法
  6. axios
  7. promise

1 Vue 简介

  • 渐进式JavaScript框架
  • 简化Dom操作
  • 响应式数据驱动

2 Vue 特点

2.1 组件化模式

采用组件化模式,提高代码复用率,易于维护。

2.2 声明式编码

使用Vue指令,无需直接操作DOM,提高开发效率。

2.3 虚拟DOM+Diff

数据->虚拟DOM->真实DOM。

通过Diff比较判断虚拟DOM和真实DOM的区别,从而更新真实DOM。

image-20220607105031042

3 第一个 Vue 程序

  • 导入开发版本Vue.js
  • 创建Vue实例对象,设置 eldata 属性
  • 使用简洁的模板语法把数据渲染到页面上
<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        {{ message }}
    </div>
    <script>
        var app = new Vue({
            el: '#app',
            data: {
                message: 'Hello Vue!'
            }
        })
    </script>
</body>
</html>

image-20210823084926166

4 el 挂载点

el(element:元素),设置挂载点。

  • 作用范围:Vue会管理el选项命中的元素及其内部的后代元素。

  • 选择器:可以使用其他选择器,但是建议使用ID选择器。

  • dom元素:可以使用其他的双标签,但不能使用 htmlbody

5 data 数据对象

  • Vue用到的数据定义在data中。
  • data中可以写复杂类型的数据。
  • 渲染复杂的数据类型时,遵循js语法即可。

6 Vue 指令

6.1 本地应用——计数器

6.1.1 本地应用 v-text 指令

设置标签的文本值(textContent)

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <h2 v-text="message+'!'">地址:</h2>
        <h2>地址:{{message+"!"}}</h2>
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                message: "深圳市南山区"
            }
        })
    </script>
</body>
</html>

image-20210823093443387

6.1.2 本地应用 v-html 指令

设置标签的innerHTML,内容中有html结构会被解析为标签。

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <h2 v-html="content"></h2>
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                content: "<a href='https://www.baidu.com'>百度</a>"
            }
        })
    </script>
</body>
</html>

image-20210823094646480

6.1.3 本地应用 v-on 指令

为元素绑定事件,事件名不需要写on,指令可以简写为@。

<div>
    <input type="button" value="事件绑定" v-on:click="doIt">
    <input type="button" value="事件绑定" v-on:mouseenter="doIt">
    <input type="button" value="事件绑定" v-on:dblclick="doIt">
    <input type="button" value="事件绑定" @dbclick="doIt">
</div>
var app = new Vue({
    el: "#app",
    methods:{
        doIt:function(){
            // 逻辑
        }
    }
})
<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="button" value="v-on指令" @click="doIt">
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            methods: {
                doIt:function(){
                    alert("Do It!")
                }
            }
        })
    </script>
</body>
</html>

image-20210823101736406

6.1.4 计数器

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="button" value="-" @click="sub">
        <span v-text="count"></span>
        <input type="button" value="+" @click="add">
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                count: 0
            },
            methods: {
                sub:function(){
                    if (this.count > 0) {
                        this.count--;
                    }else{
                        alert("不能再减了!")
                    }
                },
                add:function(){
                    if (this.count < 5) {
                        this.count++;
                    }else{
                        alert("不能再加了!")
                    }
                }
            }
        })
    </script>
</body>
</html>

6.2 本地应用——图片切换

6.2.1 本地应用 v-show 指令

根据布尔值切换显示状态(适用于频繁切换的元素)。

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="button" value="切换显示状态" @click="toggleShow">
        <img src="./Gitee.png" alt="jpg" v-show="isShow">
        <img src="./GitHub.png" alt="jpg" v-show="age>=18">
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                isShow: true,
                age: 17
            },
            methods: {
                toggleShow:function(){
                    this.isShow = !this.isShow;
                }
            }
        })
    </script>
</body>
</html>

image-20210823170604902

image-20210823170542542

6.2.2 本地应用 v-if 指令

根据表达式真假,切换元素的显示和隐藏(操作dom元素,对性能消耗大)。

本质是操作dom元素来切换显示状态,表达式为真时,元素存在于dom树中,为假时,从dom树中移除。

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="button" value="切换显示" @click="toggleShow">
        <p v-if="isShow">Hello Vue by v-if</p>
        <p v-show="isShow">Hello Vue by v-show</p>
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                isShow: true,
            },
            methods: {
                toggleShow:function(){
                    this.isShow = !this.isShow;
                }
            }
        })
    </script>
</body>
</html>

image-20210823170644164

image-20210823170711956

6.2.3 本地应用 v-bind 指令

设置元素的属性(比如src,title,class)。

v-bind:属性=表达式(v-bind可省略)。

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
    <style>
        .active {
            border: 1px solid red;
        }
    </style>
</head>
<body>
    <div id="app">
        <img :src="imgSrc" alt="">
        <br/>
        <img :src="imgSrc" alt="" :title="imgTitle+'!!!'">
        <br/>
        <img :src="imgSrc" alt="" :class="{active:isActive}" @click="toggleActive">
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                isActive: false,
                imgSrc: "./Gitee.png",
                imgTitle: "Gitee"
            },
            methods: {
                toggleActive:function(){
                    this.isActive = !this.isActive;
                }
            }
        })
    </script>
</body>
</html>

6.2.4 图片切换

<!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">
   <!-- 开发环境版本,包含了有帮助的命令行警告 -->
   <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
   <title>Vue</title>
   <style>
       .active {
           border: 1px solid red;
       }
   </style>
</head>
<body>
   <div id="app">
       <img :src="imgArr[index]" alt="">
       <a href="#" @click="prev" v-show="index!=0">上一张</a>
       <a href="#" @click="next" v-show="index<imgArr.length-1">下一张</a>
   </div>
   <script>
       var hello = new Vue({
           el: '#app',
           data: {
               isActive: false,
               imgArr: ['./Gitee.png', './GitHub.png', './weixin.png', './QQ.png'],
               index: 0
           },
           methods: {
               prev:function(){
                   if (this.index > 0){
                       this.index--;
                   }
               },
               next:function(){
                   if (this.index < this.imgArr.length-1){
                       this.index++;
                   }
               }
           }
       })
   </script>
</body>
</html>

image-20210824134421330

image-20210824134439473

image-20210824134459233

6.3 本地应用——小黑记事本

6.3.1 本地应用 v-for 指令

根据数据生成列表结构。

数组经常和 v-for 结合使用。

语法是( item, index) in 数据

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="button" value="添加数据" @click="add">
        <input type="button" value="移除数据" @click="remove">
        <ul>
            <li v-for="(item, index) in arr">
               {{index}} {{ item }}
            </li>
        </ul>
        <h3 v-for="item in vegetables" :title="item.name">
            {{ item.name }}
        </h3>
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                arr: ["北京", "上海", "广州", "深圳"],
                vegetables: [
                    {name: "西红柿"},
                    {name: "西兰花"}
                ]
            },
            methods: {
                add:function(){
                    this.vegetables.push({name: "西葫芦"});
                },
                remove:function(){
                    this.vegetables.shift();
                }
            }
        })
    </script>
</body>
</html>

6.3.2 本地应用 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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="button" value="点我" @click="doIt(666, 'adt')">
        <input type="text" @keyup.enter="saiHi">
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            methods: {
                doIt:function(p1, p2){
                    alert(p1+p2);
                },
                saiHi:function(){
                    alert("你好!");
                }
            }
        })
    </script>
</body>
</html>

6.3.3 本地应用 v-model 指令

获取和设置表单元素的值(双向数据绑定)。

绑定的数据会和表单元素值相关联。

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="button" value="设置message" @click="setMessage">
        <input type="text" v-model="message" @keyup.enter="getMessage">
        <h3>{{ message }}</h3>
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                message: "深圳市南山区"
            },
            methods: {
                getMessage:function(){
                    alert(this.message);
                },
                setMessage:function(){
                    this.message = "广东省";
                }
            }
        })
    </script>
</body>
</html>

6.3.4 小黑记事本

6.3.4.1 新增
  1. 生成列表结构(v-for 数组)。
  2. 获取用户输入(v-model)。
  3. 回车,新增数据(v-on,.enter,添加数据)。
6.3.4.2 删除
  1. 数据改变,和数据绑定的元素也会同步改变。
  2. 事件可接受自定义参数。
  3. 点击删除指定内容(v-on,splice).
6.3.4.3 统计
  1. 统计信息个数(v-text length)
  2. 基于数据的开发方式。
6.3.4.4 清空

点击清除所有信息(v-on,清空数组)。

6.3.4.5 隐藏

没有数据时,隐藏元素(v-show,v-if 数组非空)。

<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="text" v-model="items" placeholder="请输入待办事项" @keyup.enter="addlist">
        <ul>
            <li v-for="(item,index) in list">
                {{ index+1 }} {{item}}
                <input type="button" value="X" @click="remove(index)">
            </li>
        </ul>
        <span v-show="list.length!=0">
            <strong>
                {{ list.length }} 个待办
            </strong>
        </span>
        <input type="button" value="清空待办" v-show="list.length!=0" @click="removeAll">
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                list: ["写代码", "吃饭", "睡觉"],
                items: "",
            },
            methods: {
                addlist:function(){
                    this.list.push(this.items);
                },
                remove:function(index){
                    this.list.splice(index, 1);
                },
                removeAll:function(){
                    this.list = [];
                }
            }
        })
    </script>
</body>
</html>

6.4 网络应用

6.4.1 axios 基本使用

axios是一个功能强大的网络请求库。

  • axios 必须先导入再使用。
  • 使用 getpost 方法即可发送对应的请求。
  • then 方法中的回调函数会在请求成功或失败时触发。
  • 通过回调函数的形参可以获取响应内容,或错误信息。
<script src="https://unpkg.com/axios/dist/axios.min.js"></script>
// get 语法
axios.get(地址?key1=value1&key2=value2).then(function(reponse){},function(err){})
// post 语法
axios.post(地址,{key1:value1,key2:value2}).then(function(reponse){},function(err){})
<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <script src="https://unpkg.com/axios/dist/axios.min.js"></script>
    <title>Vue</title>
</head>
<body>
    <input type="button" value="get请求" class="get">
    <input type="button" value="post请求" class="post">
    <script>
        /*
            接口1:随机笑话
            请求地址:https://autumnfish.cn/api/joke/list
            请求方法:get
            请求参数:num(笑话条数)
            响应内容:随机笑话
        */
        document.querySelector(".get").onclick = function(){
            axios.get("https://autumnfish.cn/api/joke/list?num=3").then(function(response) {
                console.log(response);
            },function(err){
                console.log(err);
            })
        };
        /*
            接口2:用户注册
            请求地址:https://autumnfish.cn/api/user/reg
            请求方法:post
            请求参数:username(用户名 字符串)
            响应内容:注册成功或失败
        */
        document.querySelector(".post").onclick = function(){
            axios.post("https://autumnfish.cn/api/user/reg", {username: "jack1122654"}).then(function(response) {
                console.log(response);
            },function(err){
                console.log(err);
            })
        };
    </script>
</body>
</html>

6.4.2 axios 加 vue

  • axios 回调函数中的 this 已经改变,无法访问到 data 中数据。
  • this 保存起来,回调函数中直接使用保存的 this 即可。
  • 和本地应用最大的区别就是改变了数据来源。
<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <script src="https://unpkg.com/axios/dist/axios.min.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="button" value="获取笑话" @click="getJoke">
        <p>{{ jokes }}</p>
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                jokes: "笑话",
            },
            methods: {
                /*
                    接口1:随机笑话
                    请求地址:https://autumnfish.cn/api/joke
                    请求方法:get
                    请求参数:无
                    响应内容:获取一条随机笑话
                */
                getJoke:function(){
                    var that = this;
                    axios.get("https://autumnfish.cn/api/joke").then(function(response) {
                        that.jokes = response.data;
                        console.log(response.data);
                    },function(err){
                        console.log(err);
                    });
                }
            }
        })
    </script>
</body>
</html>

6.4.3 天知道

  • 应用的逻辑代码建议和页面分离,使用单独的 js 文件编写。
  • axios 回调函数中 this 指向改变了,需要额外保存一份。
  • 服务器返回的数据比较复杂时,获取的时候需要注意层级结构。
6.4.3.1 回车查询
  • 按下回车(v-on,keyup.enter)

  • 查询数据(axios 接口,v-model)

  • 渲染数据(v-for 数组 that)

/*  天气查询接口
	请求地址:http://wthrcdn.etouch.cn/weather_mini
	请求方法:get
	请求参数:city(查询的城市名)
	响应内容:天气信息
*/
6.4.3.2 点击查询
  • 点击城市(v-on 自定义参数)。
  • 查询数据(this.方法())。
  • 渲染数据。
  • 自定义参数让代码复用性更高。
  • methods 中定义的方法内部,可以通过 this 关键字点出其他的方法。
<!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">
    <!-- 开发环境版本,包含了有帮助的命令行警告 -->
    <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
    <script src="https://unpkg.com/axios/dist/axios.min.js"></script>
    <title>Vue</title>
</head>
<body>
    <div id="app">
        <input type="text" v-model="city" placeholder="请输入要查询的城市" @keyup.enter="getWeather">
        <input type="button" value="搜索" @click="getWeather">
        <a href="javascript:;" @click="changeCity('北京')">北京</a>
        <a href="javascript:;" @click="changeCity('上海')">上海</a>
        <a href="javascript:;" @click="changeCity('广州')">广州</a>
        <a href="javascript:;" @click="changeCity('深圳')">深圳</a>
        <ul>
            <li v-for="item in weatherList">{{item.date}} | {{item.fengxiang}} | {{item.high}} | {{item.low}} | {{item.type}}</li>
        </ul>
    </div>
    <script>
        var hello = new Vue({
            el: '#app',
            data: {
                city: "",
                weatherList: []
            },
            methods: {
                /*
                    请求地址:http://wthrcdn.etouch.cn/weather_mini
                    请求方法:get
                    请求参数:city(查询的城市名)
                    响应内容:天气信息
                */
                getWeather:function(){
                    var that = this;
                    axios.get("http://wthrcdn.etouch.cn/weather_mini?city="+this.city).then(function(response) {
                        that.weather = response.data.data.forecast;
                        that.weatherList = response.data.data.forecast;
                        console.log(response.data);
                    },function(err){
                        console.log(err);
                    });
                },
                changeCity:function(cityName){
                    this.city = cityName;
                    this.getWeather();
                }
            }
        })
    </script>
</body>
</html>

6.5 综合应用

6.5.1 音乐查询

/*	歌曲搜索接口
	请求地址:https://autumnfish.cn/search
	请求方法:get
	请求参数:keywords(查询的关键字)
	响应内容:歌曲搜索结果
*/
  • 按下回车(v-on keyup.enter)。
  • 查询数据(axios接口 v-model)。
  • 渲染数据(v-for 数组 that)。

6.5.2 音乐播放

/*	歌曲url获取接口
	请求地址:https://autumnfish.cn/song/url
	请求方法:get
	请求参数:id(歌曲id)
	响应内容:歌曲的url地址
*/
  • 点击播放(v-on)。
  • 歌曲地址获取(接口 歌曲id)。
  • 歌曲地址设置(v-bind)。
  • 歌曲id依赖于歌曲搜索的结果,对于不用的数据也需要关注。

6.5.3 歌曲封面

/*	歌曲详情获取接口
	请求地址:https://autumnfish.cn/song/detail
	请求方法:get
	请求参数:ids(歌曲id)
	响应内容:歌曲详情,包含封面信息
*/
  • 点击播放(v-on 增加逻辑)。
  • 歌曲封面获取(接口 歌曲id)。
  • 歌曲封面设置(v-bind)。

6.5.4 歌曲评论

/*	歌曲热门评论获取接口
	请求地址:https://autumnfish.cn/comment/hot?type=0
	请求方法:get
	请求参数:id(歌曲id,type固定为0)
	响应内容:歌曲的热门评论
*/
  • 点击播放。
  • 歌曲评论获取。
  • 歌曲评论渲染。

6.5.5 播放动画

  • 监听音乐播放(v-on play)。
  • 监听音乐暂停(v-on pause)。
  • 操纵类名(v-bind 对象)。
  • audio标签的play事件会在音频播放的时候触发。
  • audio标签的pause事件会在音频暂停的时候触发。
  • 通过对象的方式设置类名,类名生效与否取决于后面值得真假。

6.5.6 播放 mv

/*	
	请求地址:https://autumnfish.cn/mv/url
    请求方法:get
    请求参数:id(mvid,为0表示没有mv)
    响应内容:mv的地址
*/
6.5.6.1 index.css
body,
ul,
dl,
dd {
  margin: 0px;
  padding: 0px;
}

.wrap {
  position: fixed;
  left: 0;
  top: 0;
  width: 100%;
  height: 100%;
  background: url("../images/bg.jpg") no-repeat;
  background-size: 100% 100%;
}

.play_wrap {
  width: 800px;
  height: 544px;
  position: fixed;
  left: 50%;
  top: 50%;
  margin-left: -400px;
  margin-top: -272px;
  /* background-color: #f9f9f9; */
}

.search_bar {
  height: 60px;
  background-color: #1eacda;
  border-top-left-radius: 4px;
  border-top-right-radius: 4px;
  display: flex;
  align-items: center;
  justify-content: space-between;
  position: relative;
  z-index: 11;
}

.search_bar img {
  margin-left: 23px;
}

.search_bar input {
  margin-right: 23px;
  width: 296px;
  height: 34px;
  border-radius: 17px;
  border: 0px;
  background: url("../images/zoom.png") 265px center no-repeat
    rgba(255, 255, 255, 0.45);
  text-indent: 15px;
  outline: none;
}

.center_con {
  height: 435px;
  background-color: rgba(255, 255, 255, 0.5);
  display: flex;
  position: relative;
}

.song_wrapper {
  width: 200px;
  height: 435px;
  box-sizing: border-box;
  padding: 10px;
  list-style: none;
  position: absolute;
  left: 0px;
  top: 0px;
  z-index: 1;
}

.song_stretch {
  width: 600px;
}

.song_list {
  width: 100%;
  overflow-y: auto;
  overflow-x: hidden;
  height: 100%;
}
.song_list::-webkit-scrollbar {
  display: none;
}

.song_list li {
  font-size: 12px;
  color: #333;
  height: 40px;
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  width: 580px;
  padding-left: 10px;
}

.song_list li:nth-child(odd) {
  background-color: rgba(240, 240, 240, 0.3);
}

.song_list li a {
  display: block;
  width: 17px;
  height: 17px;
  background-image: url("../images/play.png");
  background-size: 100%;
  margin-right: 5px;
  box-sizing: border-box;
}

.song_list li b {
  font-weight: normal;
  width: 122px;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}

.song_stretch .song_list li b {
  width: 200px;
}

.song_stretch .song_list li em {
  width: 150px;
}

.song_list li span {
  width: 23px;
  height: 17px;
  margin-right: 50px;
}
.song_list li span i {
  display: block;
  width: 100%;
  height: 100%;
  cursor: pointer;
  background: url("../images/table.png") left -48px no-repeat;
}

.song_list li em,
.song_list li i {
  font-style: normal;
  width: 100px;
}

.player_con {
  width: 400px;
  height: 435px;
  position: absolute;
  left: 200px;
  top: 0px;
}

.player_con2 {
  width: 400px;
  height: 435px;
  position: absolute;
  left: 200px;
  top: 0px;
}

.player_con2 video {
  position: absolute;
  left: 20px;
  top: 30px;
  width: 355px;
  height: 265px;
}

.disc {
  position: absolute;
  left: 73px;
  top: 60px;
  z-index: 9;
}
.cover {
  position: absolute;
  left: 125px;
  top: 112px;
  width: 150px;
  height: 150px;
  border-radius: 75px;
  z-index: 8;
}
.comment_wrapper {
  width: 180px;
  height: 435px;
  list-style: none;
  position: absolute;
  left: 600px;
  top: 0px;
  padding: 25px 10px;
}
.comment_wrapper .title {
  position: absolute;
  top: 0;
  margin-top: 10px;
}
.comment_wrapper .comment_list {
  overflow: auto;
  height: 410px;
}
.comment_wrapper .comment_list::-webkit-scrollbar {
  display: none;
}
.comment_wrapper dl {
  padding-top: 10px;
  padding-left: 55px;
  position: relative;
  margin-bottom: 20px;
}

.comment_wrapper dt {
  position: absolute;
  left: 4px;
  top: 10px;
}

.comment_wrapper dt img {
  width: 40px;
  height: 40px;
  border-radius: 20px;
}

.comment_wrapper dd {
  font-size: 12px;
}

.comment_wrapper .name {
  font-weight: bold;
  color: #333;
  padding-top: 5px;
}

.comment_wrapper .detail {
  color: #666;
  margin-top: 5px;
  line-height: 18px;
}
.audio_con {
  height: 50px;
  background-color: #f1f3f4;
  border-bottom-left-radius: 4px;
  border-bottom-right-radius: 4px;
}
.myaudio {
  width: 800px;
  height: 40px;
  margin-top: 5px;
  outline: none;
  background-color: #f1f3f4;
}
/* 旋转的动画 */
@keyframes Rotate {
  from {
    transform: rotateZ(0);
  }
  to {
    transform: rotateZ(360deg);
  }
}
/* 旋转的类名 */
.autoRotate {
  animation-name: Rotate;
  animation-iteration-count: infinite;
  animation-play-state: paused;
  animation-timing-function: linear;
  animation-duration: 5s;
}
/* 是否正在播放 */
.player_con.playing .disc,
.player_con.playing .cover {
  animation-play-state: running;
}

.play_bar {
  position: absolute;
  left: 200px;
  top: -10px;
  z-index: 10;
  transform: rotate(-25deg);
  transform-origin: 12px 12px;
  transition: 1s;
}
/* 播放杆 转回去 */
.player_con.playing .play_bar {
  transform: rotate(0);
}
/* 搜索历史列表 */
.search_history {
  position: absolute;
  width: 296px;
  overflow: hidden;
  background-color: rgba(255, 255, 255, 0.3);
  list-style: none;
  right: 23px;
  top: 50px;
  box-sizing: border-box;
  padding: 10px 20px;
  border-radius: 17px;
}
.search_history li {
  line-height: 24px;
  font-size: 12px;
  cursor: pointer;
}
.switch_btn {
  position: absolute;
  right: 0;
  top: 0;
  cursor: pointer;
}
.right_line {
  position: absolute;
  left: 0;
  top: 0;
}
.video_con video {
  position: fixed;
  width: 800px;
  height: 546px;
  left: 50%;
  top: 50%;
  margin-top: -273px;
  transform: translateX(-50%);
  z-index: 990;
}
.video_con .mask {
  position: fixed;
  width: 100%;
  height: 100%;
  left: 0;
  top: 0;
  z-index: 980;
  background-color: rgba(0, 0, 0, 0.8);
}
.video_con .shutoff {
  position: fixed;
  width: 40px;
  height: 40px;
  background: url("../images/shutoff.png") no-repeat;
  left: 50%;
  margin-left: 400px;
  margin-top: -273px;
  top: 50%;
  z-index: 995;
}
6.5.6.2 main.js
/*
  1 歌曲搜索接口
    请求地址:https://autumnfish.cn/search
    请求方法:get
    请求参数:keywords(查询关键字)
    响应内容:歌曲搜索结果
  2 歌曲url获取接口
    请求地址:https://autumnfish.cn/song/url
    请求方法:get
    请求参数:id(歌曲id)
    响应内容:歌曲url地址
  3 歌曲详情获取
    请求地址:https://autumnfish.cn/song/detail
    请求方法:get
    请求参数:ids(歌曲id)
    响应内容:歌曲详情(包括封面信息)
  4 热门评论获取
    请求地址:https://autumnfish.cn/comment/hot?type=0
    请求方法:get
    请求参数:id(歌曲id,地址中的type固定为0)
    响应内容:歌曲的热门评论
  5 mv地址获取
    请求地址:https://autumnfish.cn/mv/url
    请求方法:get
    请求参数:id(mvid,为0表示没有mv)
    响应内容:mv的地址
*/
var app = new Vue({
    el: "#player",
    data: {
        qurey:"",
        musicList:[],
        musicUrl:"",
        musicDetail:"",
        hotComments:[],
        // 动画播放
        isPlaying:false,
        // 遮罩层显示状态
        isShow: false,
        // mv地址
        mvUrl:""
    },
    methods: {

        // 歌曲搜索
        searchMusic:function(){
            var that = this;
            axios.get("https://autumnfish.cn/search?keywords="+this.qurey)
              .then(function(response){
                  that.musicList = response.data.result.songs;
              }, function(err){console.log(err);});
          },

        // 歌曲播放
        musicPlayer:function(musicId){
            var that = this;
            
            // 获取播放链接
            axios.get("https://autumnfish.cn/song/url?id="+musicId)
            .then(function(response){
                that.musicUrl = response.data.data[0].url;
            }, function(err){console.log(err);});

            // 获取歌曲封面
            axios.get("https://autumnfish.cn/song/detail?ids="+musicId)
            .then(function(response){
                that.musicDetail = response.data.songs[0].al.picUrl;
            },function(err){console.log(err);});

            // 获取歌曲热门评论
            axios.get("https://autumnfish.cn/comment/hot?type=0&id="+musicId)
            .then(function(response){
                that.hotComments = response.data.hotComments;
            }, function(err){console.log(err);});
        },

        // 播放状态
        play:function(){
            this.isPlaying = true;
        },

        // 暂停状态
        pause: function(){
            this.isPlaying = false;
        },

        // 播放mv
        playMv:function(mvid){
            var that = this;
            axios.get("https://autumnfish.cn/mv/url?id="+mvid)
            .then(function(response){
                that.isShow = true;
                that.mvUrl = response.data.data.url;
                this.isPlaying = false;
            }, function(err){console.log(err);});
        },

        // 隐藏
        hide:function(){
            this.isShow = false;
            this.mvUrl = "";
        }
    }
});
6.5.6.3 music.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>悦听</title>
  <!-- 样式 -->
  <link rel="stylesheet" href="./css/index.css">
</head>

<body>
  <div class="wrap">
    <!-- 播放器主体区域 -->
    <div class="play_wrap" id="player">
      <div class="search_bar">
        <img src="images/player_title.png" alt="" />
        <!-- 搜索歌曲 -->
        <input type="text" autocomplete="off" v-model="qurey" @keyup.enter="searchMusic" />
      </div>
      <div class="center_con">
        <!-- 搜索歌曲列表 -->
        <div class='song_wrapper'>
          <ul class="song_list">
           <li v-for="item in musicList">
             <a href="javascript:;" @click="musicPlayer(item.id)"></a> 
             <b>{{item.name}}</b> 
             <span v-if="item.mvid!=0" @click="playMv(item.mvid)"><i></i></span></li>
          </li>
          </ul>
          <img src="images/line.png" class="switch_btn" alt="">
        </div>
        <!-- 歌曲信息容器 -->
        <div class="player_con" :class="{playing:isPlaying}" class="playing">
          <img src="images/player_bar.png" class="play_bar" />
          <!-- 黑胶碟片 -->
          <img src="images/disc.png" class="disc autoRotate" />
          <img :src="musicDetail"  class="cover autoRotate" />
        </div>
        <!-- 评论容器 -->
        <div class="comment_wrapper">
          <h5 class='title'>热门留言</h5>
          <div class='comment_list'>
            <dl v-for="item in hotComments">
              <dt><img :src="item.user.avatarUrl" alt=""></dt>
              <dd class="name">{{item.user.nickname}}</dd>
              <dd class="detail">
                {{item.content}}
              </dd>
            </dl>
          </div>
          <img src="images/line.png" class="right_line">
        </div>
      </div>
      <div class="audio_con">
        <audio ref='audio'  @play="play" @pause="pause" :src="musicUrl"  controls autoplay loop class="myaudio"></audio>
      </div>
      <div class="video_con" v-show="isShow"  style="display: none;">
        <video :src="mvUrl" controls="controls"></video>
        <div class="mask" @click="hide"></div>
      </div>
    </div>
  </div>
  <!-- 开发环境版本,包含了有帮助的命令行警告 -->
  <script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
  <!-- 官网提供的 axios 在线地址 -->
  <script src="https://unpkg.com/axios/dist/axios.min.js"></script>
  <!-- 自己的js -->
  <script src="./js/main.js"></script>
</body>
</html>

文章作者:GentleTK
原文链接:https://gentletk.gitee.io/Vue基础

posted @ 2022-07-31 11:12  GentleTK  阅读(17)  评论(0编辑  收藏  举报