本项目为客户端部分,后台管理系统部分详见 👉 vue-cars-admin
教程官网:http://www.web-jshtml.cn/#/ 视频地址:https://www.bilibili.com/video/BV1Gz411z75G
学习内容:头脑风暴分析项目,项目功能延伸思维,程序逻辑分析;高德地图基础级API学习;Vue+Vuex+Router+ElementUi+Webpack全家桶,ElementUI组件,VUE各项指令,component、props、propsData、computed,watch,组件化开发,生命周期,路由权限,Sass,Axios拦截器,缓存,项目部署,Nginx,域名,服务器,GIT,原型,接口联调,性能,缓存,检测toKen等;
学习成效:掌握对项目的扩展延伸思维,项目在业务上的关联性。完全自主搭建后台管理系统,高德地图API应用;
主要为项目开发流程介绍,包含需求文档、功能分析、原型绘制、测试用例等内容。
npm install vue-amap --save
引入vue-amap
// main.js
import Vue from 'vue';
import VueAMap from 'vue-amap';
import App from './App.vue';
Vue.use(VueAMap);
VueAMap.initAMapApiLoader({
key: 'your amap key', // 官网申请
plugin: ['AMap.Autocomplete', 'AMap.PlaceSearch', 'AMap.Scale', 'AMap.OverView', 'AMap.ToolBar', 'AMap.MapType', 'AMap.PolyEditor', 'AMap.CircleEditor'], // 插件(按需引入)
v: '1.4.4' // 高德 sdk 版本
});
new Vue({
el: '#app',
render: h => h(App)
})
使用vue-amap
// APP.vue
<template>
<div id="app">
<h3 class="title">{{ msg }}</h3>
<div class="amap-wrapper">
<el-amap class="amap-box" :vid="'amap-vue'"></el-amap>
</div>
</div>
</template>
<script>
export default {
data () {
return {
msg: 'vue-amap向你问好!'
}
}
}
</script>
<style>
.amap-wrapper {
width: 500px;
height: 500px;
}
</style>
在定制化程度较高的项目中,开发者可能只想通过 vue-amap 引入高德地图,而部分实例化的操作直接基于高德地图的 sdk 完成。这个时候就需要 lazyAMapApiLoaderInstance
。
<template>
<div class="amap-wrapper">
<el-amap :vid="'amapVue'" :center="center" :zoom="zoom"></el-amap>
</div>
</template>
<script>
import { lazyAMapApiLoaderInstance } from "vue-amap";
export default {
name: "Index",
data() {
return {
map: null,
zoom: 13,
center: [116.404765, 39.918052]
};
},
mounted() {
lazyAMapApiLoaderInstance.load().then(() => {
// eslint-disable-next-line no-undef
this.map = new AMap.Map("amapVue", {
center: this.center,
zoom: this.zoom
});
});
}
};
</script>
<style lang="scss" scoped>
.amap-wrapper {
width: 100vw;
height: 100vh;
}
</style>
@mixin clearfix {
&:after {
clear: both;
content: '.';
display: block;
height: 0;
line-height: 0;
overflow: hidden;
}
*height: 1%; // 貌似兼容IE7的意思?
}
.wrap {
@include clearfix; // 使用方法
}
如上图所示的需求,会显示一格电至十格电,这个时候就可以设置一些的通用的css样式,例如active-1代表前1个格子为蓝色,active-4代表前4个格子为蓝色,这时候我们就需要利用SCSS利用代码的思维来生成10个active-i,而不是我们自己手动重复写10个css样式。
css选择器小知识:
尤其要注意:nth-child(-n+length)代表前length个元素!!!
:nth-child() 代表选择某个的一个或多个特定的子元素
1、:nth-child(number);/*参数是具体数字*/
2、:nth-child(n);/*参数是n,n从0开始计算*/
3、:nth-child(n*length)/*n的倍数选择,n从0开始算*/
4、:nth-child(n+length);/*选择大于length后面的元素*/
5、:nth-child(-n+length)/*选择小于length前面的元素*/
6、:nth-child(n*length+1);/*表示隔几选一*/
需求实现代码:
@for $i from 1 through 10 {
.active-li-#{$i} {
li:nth-child(-n + #{$i}) {
// ...
}
}
}
Swiper是纯javascript打造的滑动特效插件,swiper6.0目前仅支持vue3.0,Vue2.0可以结合vue-awesome-swiper。
npm install swiper vue-awesome-swiper --save
具体使用详见官方文档API
略。
需求描述:点击user icon则右侧滑动出现,点击其他区域则右侧滑动消失。
实现方案:在页面挂载一个mouseup监听函数,当右侧栏显示的时候,点击鼠标时判断鼠标点击的位置是否是右侧栏的区域(通过document.getElementById等方法获取),如果不是则更新使右侧滑动消失,反之,则不处理。
mounted() {
document.addEventListener("mouseup", e => {
const sider = document.getElementById("sider-wrap");
if (sider && sider.contains(e.target)) {
this.$router.push({ name: "Home" });
}
});
}
全局注册后即可直接使用,
<MyComponent></MyComponent>
import MyComponent from '@/components/MyComponent.vue'
Vue.component("MyComponent", MyComponent); // 全局自定义组件
完成后台管理系统登录/注册功能,注意跨域访问、请求拦截器、路由守卫、验证码流程等相关内容可详见vue3.0后台管理系统项目实战,本项目仅简单实现。
略。
父组件向子组件传数据的时候,这是一个单向数据流,当我们在子组件中想要修改通过props绑定的父组件里的数据时,Vue不支持我们直接在子组件中进行修改,子组件没有修改父组件数据的权力,其中一种方法就是通过修饰符.sync
// 父组件
<template>
<!-- 完整写法 <Child :number="total" @update:number="total = $event"/> -->
<Child :number.sync="total"></Child>
</template>
<script>
import Child from 'components/Child.vue'
export default {
name: 'Parent',
components: {
Child
}
data() {
return {
total: 1000
}
}
}
</script>
// 子组件
<template>
{{number}}
<button @click="change" >change</button>
</template>
<script>
export default {
name: 'Child',
props: {
number: {
type: Number,
default: 0
}
},
methods:{
change() {
this.$emit('update:number',this.number + 1)
}
}
}
</script>
略。
npm i wangeditor --save
本文中使用wangEditor富文本编辑器插件,具体使用详见说明文档。
<template>
<div class="home">
<h3>wangEditor with vue</h3>
<div id="demo1"></div>
<button type="button" class="btn" @click="getEditorData">获取当前内容</button>
<h3>内容预览</h3>
<textarea name="" id="" cols="170" rows="20" readonly v-model="editorData"></textarea>
</div>
</template>
<script>
// 引入 wangEditor
import wangEditor from 'wangeditor'
export default {
data() {
return {
editor: null,
editorData: ''
}
},
mounted() {
const editor = new wangEditor(`#demo1`)
// 配置 onchange 回调函数,将数据同步到 vue 中
editor.config.onchange = (newHtml) => {
this.editorData = newHtml
}
// 创建编辑器
editor.create()
this.editor = editor
},
methods: {
getEditorData() {
// 通过代码获取编辑器内容
let data = this.editor.txt.html()
alert(data)
}
},
beforeDestroy() {
// 调用销毁 API 对当前编辑器实例进行销毁
this.editor.destroy()
this.editor = null
}
}
</script>
<style lang="scss">
.home {
width: 1200px;
margin: auto;
position: relative;
.btn {
position: absolute;
right: 0;
top: 0;
padding: 5px 10px;
cursor: pointer;
}
h3 {
margin: 30px 0 15px;
}
}
</style>
略。
传统的一些写法就是根据if
来不断判断条件进行设置:
// 省
if (level === 0) {
requestData.type = "province";
}
// 市
if (level === 1) {
requestData.type = "city";
requestData.province_code = node.value;
}
// 区
if (level === 2) {
requestData.type = "area";
requestData.city_code = node.value;
}
我们其实可以根据JSON配置来简化上述代码,这样更加利于配置:
const config = {
0: {
type: "province"
},
1: {
type: "city",
code: "province"
},
2: {
type: "area",
code: "city"
}
};
requestData.type = config[level].type;
if (config[level].code) {
requestData[`${config[level].code}_code`] = node.value;
}
这个方法很实用,也很方便,要掌握哦!!!
如下所示,组件回调的常规做法设置一堆对应的函数来进行调用。
<template>
<myComponent @a="functionA" @b="functionB" @c="functionC"></myComponent>
</template>
其实可以做一个简化的操作,通过返回一个JSON包括函数名和执行参数来处理
<template>
<myComponent @callback="callback"></myComponent>
</template>
<script>
export default {
methods: {
callback(params) {
if (params.funcName) {
this[params.funcName](params.data)
}
}
}
}
</script>
this.$emit("callback", {
funcName: a,
data: {}
})
略。
在多处使用的内容,可以抽离出来统一定义,避免了修改时需要修改多处的情况,例如本项目中停车场类型有室内室外,表单和列表中多个地方需要显示和选择,则可以在一个地方统一管理这些数据。可通过两种方法来管理:一种是通过Vuex来管理,就仅使用Vuex的state属性,然后全局引入即可使用;第二种是建立一个JS文件,在使用的地方引入即可。(也许更建议使用Vuex来管理吧?)
Vue 实现响应式并不是数据发生变化之后 DOM 立即变化,而是按一定的策略进行 DOM 的更新。$nextTick 是在下次 DOM 更新结束之后执行延迟回调,在修改数据之后使用 $nextTick,则可以在回调中获取更新后的 DOM。当项目中你想在改变DOM元素的数据后基于新的dom做点什么,对新DOM一系列的js操作都需要放进Vue.nextTick()的回调函数中。
注意,Vue生命周期的created()钩子函数进行的DOM操作一定要放在Vue.nextTick()的回调函数中,原因是在created()钩子函数执行的时候DOM 其实并未进行任何渲染,而此时进行DOM操作无异于徒劳,所以此处一定要将DOM操作的js代码放进Vue.nextTick()的回调函数中。
<script>
export default {
methods: {
myfunc(){
this.$nextTick(()=>{
// ....
})
}
}
}
</script>
例如修改input框的默认颜色:
input::-webkit-input-placeholder{
color:#585c89;
}
input::-moz-placeholder{ /* Mozilla Firefox 19+ */
color:#585c89;
}
input:-moz-placeholder{ /* Mozilla Firefox 4 to 18 */
color:#585c89;
}
input:-ms-input-placeholder{ /* Internet Explorer 10-11 */
color:#585c89;
}
Vue.js 允许你自定义过滤器,可被用于一些常见的文本格式化。过滤器可以用在两个地方:双花括号插值和 v-bind
表达式 (后者从 2.1.0+ 开始支持)。
<!-- 在双花括号中 -->
{{ message | capitalize }}
<!-- 在 `v-bind` 中 -->
<div v-bind:id="message | capitalize"></div>
filters: {
capitalize: function (value) {
if (!value) return ''
value = value.toString()
return value.charAt(0).toUpperCase() + value.slice(1)
}
}
在表单的实现中,实际的数据可能需要文本格式化,例如1对应“室内”,2对应“室外”,此时就需要用到过滤器。
- 传统方法直接通过函数来格式化文本:
<template slot-scope="scoped">
{{getType(scoped.row.type)}}
</template>
<script>
export default{
data() {
return {}
},
methods: {
getType(value){
if(value===1){
return "室内";
}else{
return "室外";
}
}
}
}
</script>
- 也可以通过过滤器来格式化文本:
<template slot-scope="scoped">
{{scoped.row.type | getType}}
</template>
<script>
export default{
data() {
return {}
},
filters: {
getType(value){
if(value===1){
return "室内";
}else{
return "室外";
}
}
}
}
</script>
如果在过滤器里使用this则会报错,对于过滤器里面无法使用this获取data里面的数据的问题?
<template slot-scope="scoped">
{{getType(scoped.row.type)}}
</template>
<script>
let _this = this;
export default{
data() {
_this = this
return {
num: 1
}
},
filters: {
getType(value){
if(value=== _this.num){ // 通过外部设置_this来使得过滤器内也能访问到this
return "室内";
}else{
return "室外";
}
}
}
}
</script>
略。
具名插槽:
// 父组件
<Child>
<template v-slot:header>
<h1>Here might be a page title</h1>
</template>
</Child>
// 子组件
<div class="container">
<header>
<slot name="header"></slot>
</header>
</div>
如何让父组件中的插槽内容能够访问子组件中才有的数据?绑定在 <slot>
元素上的 attribute 被称为插槽 prop,现在在父级作用域中,我们可以使用带值的 v-slot
来定义我们提供的插槽 prop 的名字。
// 父组件
<Child>
<template v-slot:header="slotProps">
{{ slotProps.myProps.firstName }}
</template>
</Child>
// 子组件
<span>
<slot name="header" :myProps="user"></slot>
</span>
车辆品牌新增/列表功能 + 表单组件封装 + 车辆新增/列表功能 + 搜索表单封装,略。
首先,通过.env.development
文件预先设置不同的地址:
# 管理端
VUE_APP_API_LOGIN = "/apiLogin"
VUE_API_DEV_LOGIN_TARGET = "http://www.web-jshtml.cn/api/cars"
# web端
VUE_APP_API_WEB = "/apiWeb"
VUE_API_DEV_WEB_TARGET = "http://www.web-jshtml.cn/api/cars/web"
然后,调用接口时使用不同的接口前缀以作区分,.env.development
文件中的内容通过process.env.名字
调用:
export function GetCode(data) {
return service.request({
method: "post",
url: `${process.env.VUE_APP_API_LOGIN}/getCode/`,
data
});
}
最后,在vue配置文件vue.config.js
中重定向至真实地址即可:
module.exports = {
// webpack-dev-server 相关配置
devServer: {
open: false, // 编译完成是否打开网页
host: "0.0.0.0", // 指定使用地址,默认localhost,0.0.0.0代表可以被外界访问
port: 8080, // 访问端口
https: false, // 编译失败时刷新页面
hot: true, // 开启热加载
hotOnly: false,
proxy: {
// 后端的接口:http://www.web-jshtml/api/cars 接口:/getCode/
[process.env.VUE_APP_API_LOGIN]: {
target: "http://www.web-jshtml.cn/api/cars", //真实服务器的地址
changeOrigin: true,
pathRewrite: {
[`^${process.env.VUE_APP_API_LOGIN}`]: ""
}
},
// 前端的接口:http://www.web-jshtml/api/cars/web
[process.env.VUE_APP_API_WEB]: {
target: process.env.VUE_API_DEV_WEB_TARGET, //真实服务器的地址
changeOrigin: true,
pathRewrite: {
[`^${process.env.VUE_APP_API_WEB}`]: ""
}
}
}
}
}
略。
地图上存在一个按钮,点击其使得地图定位到当前位置,可以通过Vuex来实现。
首先,Vuex中定义一个状态值selfLocation,当某组件点击按钮时触发mutations,使得selfLocation的值发生改变。
const state = {
selfLocation: true
};
const mutations = {
SELF_LOCATION() {
state.selfLocation = !state.selfLocation; // 在map中监听值改变从而进行处理
}
};
this.$store.commit("location/SELF_LOCATION");
然后,在地图组件中监听状态值selfLocation,当该值发生变化时,就触发响应的方法,使得地图定位到当前位置。
watch: {
"$store.state.location.selfLocation": {
handler() {
this.setLocation();
}
}
}
略。
当两个接口请求之间存在顺序调用的关系时,除了可以在第一个接口的完成函数里面调用第二个接口请求之外还可以采用异步操作async/await的形式。
// 内部调用的方式
demo1(){
func1().then(()=>{
// func2需要使用func1请求得到的数据
func2().then(()=>{
})
})
}
// 异步请求
outFunc1(){
// ...请求数据
func1().then(()=>{})
}
async demo2(){ // 执行
await outFunc1(); // 等待执行完才继续
func2().then(()=>{
})
}
<template>
<h1 class="price">{{priceToFixedByComputed}}</h1>
<h1 class="price">{{priceToFixedByMethods()}}</h1>
<h1 class="price">{{price | priceToFixedByFilters}}</h1>
</template>
<script>
export default {
name: "",
data() {
return {
price: 27.999999999
}
},
computed: {
priceToFixedByComputed() {
console.log('computed')
return this.price.toFixed(2)
}
},
methods: {
priceToFixedByMethods() {
console.log('methods')
return this.price.toFixed(2)
}
},
filters: {
priceToFixedByFilters(val) {
console.log('filters')
return val.toFixed(2)
}
},
watch: {
price(val) {
console.log('watch')
}
},
};
</script>
- computed,一般用于计算,可直接在模板中使用有缓存机制,性能比methods更高。
- methods, 一般用于函数方法,在模板中使用时需加(),没有缓存,使用一次就触发一次。
- filters,一般用于格式化(时间,取整,取余等),使用时用 “|” 隔开。
- watch,一般用于数据变化时执行异步或开销较大的操作,当观察的对象改变时触发,没有返回值。
略。
略。
对于一些反复请求获取相同数据的情况,可以借助缓存机制:首先查找缓存里是否存在;如果存在则直接读取缓存里的数据,如果不存在则请求数据并把数据写入缓存。
略。
如果发现本项目有错误,欢迎提交 issues 指正。