持续创作,加速成长!这是我参与「掘金日新计划 · 6 月更文挑战」的第6天,点击查看活动详情
方式一
首先安装 axios:npm i axios
修改 App.vue
<template>
<div>
<button @click="getBookInfo">获取书籍信息</button>
</div>
</template>
<script>
import axios from "axios"
export default {
name: 'App',
methods:{
getBookInfo(){
axios.get("http://localhost:8080/downloads/data.json").then(
response =>{
console.log("请求成功了",response.data);
},
error => {"请求失败了",error.message}
)
}
}
}
</script>
点击按钮请求书籍的信息,其中http://localhost:8080/downloads/data.json返回了书籍信息:
当我们点击按钮时,却发现报错了:
这是跨域请求的错误,当协议名(http)、主机名(localhost)、端口号(8080) 中有一个不一致会提示跨域请求。解决办法有:1、cors;2、jsonp;3、代理服务器
我们来学习第三个,可以先查看下官方文档 Vue Cli 配置参考:devServer.proxy
流程:8081 向 8080 请求数据,我们开启一个代理服务器,端口号和我们的 8081 相同
修改
1、 vue.config.js,直接复制官网代码,修改端口号为请求数据的端口8080
const { defineConfig } = require('@vue/cli-service')
module.exports = defineConfig({
transpileDependencies: true,
lintOnSave:false,
//开启代理服务器
devServer: {
proxy: 'http://localhost:8080'
}
})
2、请求数据端口修改为本地端口 8081
3、由于修改了脚手架配置,所以先停掉端口,然后重新运行
可以看到请求成功了
方式二
方式一有两个不完美的地方:1、只能配置一个代理服务器。2、当本地也就是 public 中有相同数据,就不请求代理服务器了
1、修改 vue.config.js,注释掉方式一的代码,改为
devServer: {
proxy: {
'/api': {
target: 'http://localhost:8080',
pathRewrite:{'^/api':''},
ws: true,
changeOrigin: true
}
}
}
其中/api为请求前缀,可以改成其他名字
pathRewrite为路径重写,匹配api开头的字符串,并把api替换为空字符串,这样才能保证代理服务器转发给 8080 的不带前边的前缀
ws 用于支持websocket
changeOrigin请求来自于。即控制请求头中host数据。默认为true说谎:来自8080;false如实回答来自代理服务器8081
2、请求路径增加前缀
3、由于修改了脚手架配置,所以先停掉端口,然后重新运行
vue 脚手架配置代理总结
方法一
在vue.config.js中添加如下配置:
devServer: {
proxy: 'http://localhost:8080'
}
说明:
1.优点:配置简单,请求资源时直接发给前端(8080)即可。
2.缺点:不能配置多个代理,不能灵活的控制请求是否走代理
3.工作方式:若按照上述配置代理,当请求了前端不存在的资源时,那么该请求会转发给服务器(优先匹配前端资源)
方法二
devServer: {
proxy: {
'/api': {//匹配所有以/api开头的路径
target: 'http://localhost:8080',//代理目标的基础路径
pathRewrite:{'^/api':''},//路径重写,匹配api开头的字符串,并把api替换为空字符串
ws: true,//用于支持websocket
changeOrigin: true//请求来自于。即控制请求头中host数据。默认为true说谎:来自8080;false如实回答来自代理服务器8081
}
}
}
说明:
1.优点:可以配置多个代理,且可以灵活的控制请求是否走代理。
2.缺点:配置略微繁琐,请求资源时必须加前缀。
TodoList案例增加编辑功能
App.vue 中增加样式
.btn-edit {
color: #ffffff;
background-color: skyblue;
border: 1px solid skyblue;
margin-right: 10px;
}
同时增加修改 todo 的方法,同时绑定
//更新todo
updateTodo(id,title) {
this.todos.forEach((todo) => {
if (todo.id === id) todo.title = title
})
}
mounted() {
......
this.$bus.$on("updateTodo",this.updateTodo)
},
destroyed() {
......
this.$bus.$off("updateTodo")
}
Item.vue 中增加编辑按钮,使用增加的样式
增加点击事件 handleEdit,使用isEdit来判断是否处于编辑状态。这个方法中先判断有没有 isEdit 属性,如果没有则追加,如果有则直接改为 true
增加 input ,值和 span 中显示的值一样,如果是编辑状态则展示。而 span 则相反
同时 input 增加失去焦点事件 handleBlur,当失去焦点时,修改 todo 的值
<template>
<li>
<label>
<input type="checkbox" :checked="todo.done" @change="handleCheck(todo.id)">
<span v-show="!todo.isEdit">{{todo.title}}</span>
<input type="text"
v-show="todo.isEdit"
:value="todo.title"
@blur="handleBlur(todo,$event)"
ref="inputTitle">
</label>
<button class="btn btn-danger" @click="handleDelete(todo.id)">删除</button>
<button v-show="!todo.isEdit" class="btn btn-edit" @click="handleEdit(todo)">编辑</button>
</li>
</template>
<script>
import pubsub from "pubsub-js"
export default {
......
//编辑
handleEdit(todo){
if(Object.prototype.hasOwnProperty.call(todo, 'isEdit')){
todo.isEdit = true;
}else{
this.$set(todo,"isEdit",true)
}
//同时获取焦点,
this.$nextTick(function () {
this.$refs.inputTitle.focus()
})
},
//失去焦点处理
handleBlur(todo,e){
todo.isEdit = false
if(!e.target.value.trim()) return alert("输入内容不能为空")
this.$bus.$emit("updateTodo",todo.id,e.target.value)
}
}
}
</script>
nextTick
1.语法this.$nextTick(回调函数)
2.作用:在下一次DOM更新结束后执行其指定的回调
3.什么时候用:当改变数据后,要基于更新后的新DOM进行某些操作时,要在 nextTick 所指定的回调函数中执行