Vue 项目总结
1.搜索历史纪录
- 使用 localStorage 解决 vuex 在页面刷新后数据被清除的问题
使用逻辑
- 1.在 vuex 获取到用户信息时(例:state.user_id =user_info.user_id),同时把这个信息存入 localStorage
- 2.定义一个 mutation 方法,再次把数据从 localStorage 传给 state;
- 3.我们的问题发生在页面刷新时,可能很多同学就会去想在监听页面刷新事件时触发上面第 2 步的那个方法。其实不用这么麻烦,在 computed 属性时,只要我们判断一下用户信息是否为空,为空时调用第 2 步的那个方法就好了,然后我们把这个 state return 给那个响应属性就好了;
- 4.至于为什么这么麻烦在 state 和 localStorage 中传来传去,是因为 state 的值刷新后会没了,而 localStorage 的值不能响应式地变化(Vue 仅可以对其管理的数据做响应式处理,可以理解为 data 中的数据,localStorage 并不在 Vue 的管理下,自然不会有响应特性);
2.v-model 指令的三个参数
1.number
3.vee-validate 表单验证
安装
npm install vee-validate --save
配置和使用
配置使用主要是 3 个文件。
- validate.js :自己创建一个文件,单独将关于表单验证的代码抽离出来,从 node_modules 中引入 VeeValidate,配置相关项
- main.js :vue 主文件入口,引入 validate.js
- form.vue :表单组件
validate.js
import Vue from 'vue'
import VeeValidate, { Validator } from 'vee-validate'
import zh from 'vee-validate/dist/locale/zh_CN' //引入中文文件
// 配置中文
Validator.addLocale(zh)
const config = {
locale: 'zh_CN'
}
Vue.use(VeeValidate, config)
// 自定义validate
const dictionary = {
zh_CN: {
messages: {
email: () => '请输入正确的邮箱格式',
required: field => '请输入' + field
},
attributes: {
email: '邮箱',
password: '密码',
name: '账号',
phone: '手机'
}
}
}
Validator.updateDictionary(dictionary)
Validator.extend('phone', {
messages: {
zh_CN: field => field + '必须是11位手机号码'
},
validate: value => {
return value.length == 11 && /^((13|14|15|17|18)[0-9]{1}\d{8})$/.test(value)
}
})
3-1 引入表单验证依赖文件,并且引入的是支持中文错误提示的文件。
import Vue from 'vue'
import VeeValidate, { Validator } from 'vee-validate'
import zh from 'vee-validate/dist/locale/zh_CN' //引入中文文件
3-2 进行中文错误提示的配置。
// 配置中文
Validator.addLocale(zh)
const config = {
locale: 'zh_CN'
}
Vue.use(VeeValidate, config)
3-3 如果你想自定义表单验证的提示语,不配置有默认。
// 自定义validate
const dictionary = {
zh_CN: {
messages: {
email: () => '请输入正确的邮箱格式',
required: field => '请输入' + field
},
attributes: {
email: '邮箱',
password: '密码',
name: '账号',
phone: '手机'
}
}
}
Validator.updateDictionary(dictionary)
//message : 提示语。
//attributes: 就是 filed。
3-4 扩展自定义的验证,比如这边自定义了电话的表单验证。
Validator.extend('phone', {
messages: {
zh_CN: field => field + '必须是11位手机号码'
},
validate: value => {
return value.length == 11 && /^((13|14|15|17|18)[0-9]{1}\d{8})$/.test(value)
}
})
main.js 中代码
import './static/js/validate.js'
只需要将 validate.js 引入主文件入口就完成了铺路工作,现在就可以开始写表单界面了。
form.vue 组件中的代码:
<div class="layui-form-item">
<label class="layui-form-label">账户</label>
<div class="layui-block">
<input v-model="name" v-validate="'required|min:3|alpha'" :class="{'input': true, 'is-danger': errors.has('name') }" type="text" name="name" class="layui-input" placeholder="账户">
<span v-show="errors.has('name')" class="text-style" v-cloak> {{ errors.first('name') }} </span>
</div>
</div>
这是其中的一个 input 拿出来讲:
1、首先在 input 中你得有 name 属性。
2、v-validate 属性:管道形式进行过滤,验证条件。
3、span 就是错误提示 。
errors.first('field') // 获取关于当前field的第一个错误信息
collect('field') // 获取关于当前field的所有错误信息(list)
has('field') // 当前filed是否有错误(true/false)
all() // 当前表单所有错误(list)
any() // 当前表单是否有任何错误(true/false)12345
到这边你可以完成基础的表单验证了,不同的项目都会有不同的需求,表单验证也不一样,但是 VeeValidate 支持你去扩展,完成各种不同的需求。你可以参照着官网文档,造一个自己的表单验证,这样不同项目时只需要进行简单的修改就能用上了。
官网:https://baianat.github.io/vee-validate/configuration.html
4.vue-amap
- 基于 Vue 2.x 与高德的地图组件
4.1 到高德地图注册一个帐号,创建应用,创建一个 key
4.2 安装
npm install vue-amap --save
配置
// 引入vue-amap
import AMap from 'vue-amap'
Vue.use(AMap)
// 初始化vue-amap
AMap.initAMapApiLoader({
// 申请的高德key
key: 'YOUR_KEY',
// 插件集合
plugin: ['AMap.PlaceSearch', 'AMap.Geolocation']
})
4.3 开始使用
<input type="text" ref="searchText" id="searchText" @keyup="keyUpSearch" placeholder="请输入地址"/>
<div class="address_items" id="address_result" v-if="searchData.length > 0">
<div class="address_item" v-for="item in searchData">
<div class="title">{{ item.name }}</div>
<div class="description">{{ item.pname }}{{ item.cityname }}{{ item.address }}</div>
</div>
</div>
methods 里面添加对应的 keyUpSearch 方法
<script>
export default {
methods: {
keyUpSearch() {
var _this = this;
var txt = this.$refs.searchText.value;
AMap.service(["AMap.PlaceSearch"], function() {
var placeSearch = new AMap.PlaceSearch({
//构造地点查询类 pageSize: 12, pageIndex: 1, city: "成都", //城市 cityLimit: 'true', panel: 'temp'//搜索结果的展示面板对元素id,不知道为什么一定要有该参数,最终获取的结果才更完整,参数更多跟完整,//所以我在页面随便写了一个<div id="temp" style="display:none"></div>
});
//关键字查询
placeSearch.search(txt, function(status, result) {
if (status == "complete" && result.info == "OK") {
//这里可以console.log(result),查看所有返回的参数,遍历展示这些基本的,我就不赘述
//_this.searchData = result.poiList.pois
}
});
});
}
}
};
</script>
5.骨架屏
我们希望在构建时渲染 skeleton 组件,将渲染 DOM 插入 html 的挂载点中,同时将使用的样式通过 style 标签内联。这样在前端 JS 渲染完成之前,用户将看到页面的大致骨架,感知到页面是正在加载的。
我们当然可以选择在开发时直接将页面骨架内容写入 html 模版中,但是这会带来两个问题:
开发 skeleton 与其他组件体验不一致。 多页应用中多个页面可能共用同一个 html 模版,而又有独立的 skeleton。 下面我们将看看插件在具体实现中是如何解决这两个问题的:
vue-skeleton-webpack-plugin
github 地址:https://github.com/lavas-project/vue-skeleton-webpack-plugin
具体实现步骤:
1、我们用 vue-cli 直接构建一下项目跑起来(具体怎么构建就不说了)
2、进去当前项目,执行命令 : npm install vue-skeleton-webpack-plugin
3、我们在 src 目录下创建 Skeleton.vue
<template>
<div class="skeleton-wrapper">
<header class="skeleton-header"></header>
<section class="skeleton-block">
<img
src=""
/>
<img
src=""
/>
</section>
</div>
</template>
<script>
export default {
name: 'skeleton'
}
</script>
<style scoped>
.skeleton-header {
height: 40px;
background: #1976d2;
padding: 0;
margin: 0;
width: 100%;
}
.skeleton-block {
display: flex;
flex-direction: column;
padding-top: 8px;
}
</style>
4、创建入口文件:entry-skeleton.js
import Vue from 'vue'
import Skeleton from './Skeleton'
export default new Vue({
components: {
Skeleton
},
template: '<Skeleton />'
})
5、我们在 build 目录下创建 webpack.skeleton.conf.js
'use strict'
const path = require('path')
const merge = require('webpack-merge')
const baseWebpackConfig = require('./webpack.base.conf')
const nodeExternals = require('webpack - node-externals')
function resolve(dir) {
return path.join(__dirname, dir)
}
module.exports = merge(baseWebpackConfig, {
target: 'node',
devtool: false,
entry: {
app: resolve('../src/entry-skeleton.js')
},
output: Object.assign({}, baseWebpackConfig.output, {
libraryTarget: 'commonjs2'
}),
externals: nodeExternals({
whitelist: /\.css$/
}),
plugins: []
})
6.然后在 webpack.dev.conf.js 和 webpack.prod.conf.js 分别加入
const SkeletonWebpackPlugin = require('vue-skeleton-webpack-plugin')
// inject skeleton content(DOM & CSS) into HTML
new SkeletonWebpackPlugin({
webpackConfig: require('./webpack.skeleton.conf'),
quiet: true
})
6.为什么要有 vue,从历史角度谈谈
因为 vue 大大提高了前端开发的效率,vue 带来了前端开发模式的转变,传统的前端开发是使用 html+css 搭建前端页面,使用 js 或 jquery 等技术进行页面的交互逻辑,然后使用 ajax 技术进行与后台的通信,后台返回数据浏览器进行页面的渲染。
使用 vue 之后我们可以进行数据的双向绑定,使用 vue-router 处理页面路由,使用 vue 处理各组件直接的通信,使用 vue-resource 或者 axios 处理 http 请求,使用 element-ui 或者 mint-ui 来搭建前端页面
- 优点:目前路西的单页面流行框架,减轻服务器压力,前后端完全分离,比 augular 更省资源
- 缺点:不利于 seo,(对国内搜索引擎来说,比如百度),第一次加载速度较慢
- 解决:页面静态化,可以使爬虫抓取页面
7.vue 的核心内容:数据双向绑定 MVVM 思想 组件化开发 虚拟 DOM 树 自定义指令 生命周期
- 为什么要有指令?操作 DOM,放到页面上
- 1.为了把我们得到的数据放到页面上
- 2.大牛把操作 DOM 方法封装,让程序员专注于操作数据
8.vue 的基本用法
8.1 常见指令
- v-if 根据表达式的值在 dom 中生成或移除一个元素
- v-show 根据表达式的值显示或隐藏 html 元素
- v-if 和 v-show 都可以用来控制页面的隐藏和显示,他们的区别在于,v-if 是直接在页面中删除某个元素,v-show 是改变元素的 display 属性来控制隐藏或者显示
- v-else 是 js 中 else 的意思,必须跟着 v-if 或者 v-show 使用
- v-model 用来在表单控件元素上创建双向数据绑定
- v-bind 用来响应更新 html 新特性,将一个或多个属性绑定到表达式上
- v-on 用于绑定事件监视器
- v-text 和 v-html 以及都是用来实现数据绑定的,v-text 和 v-html 的区别在于会不会输出 thml 标签 使用的话,在网速较慢的情况下会显示出来,可以使用 v-cloak 来解决这个问题
8.2 组件怎么做的?
在 vue 中定义全局组件有三种方式 1.使用 vue.extend({})定义全局组件,然后再通过 vue.component 把 vue.extend 定义的组件注册到 vue 全局上 2.直接用 vue.component("引用组件的名称”,{组件实际的定义模板}) 3.vue.component(“组件名称”,
8.3 用户怎么请求数据?
使用 vue-resource 或者使用 axios
getData(){this.$http
.get("http://vue.studyit.io/api/ getnewslist").then(res =>{
console.log(res.body);
this.list = res.body.message
})
.catch(err =>{console.log(err);})
},
postData(){this.$http
.post("http://vue.studyit.io/api/postcomment/17",
{content:"完美!"},
{emulateJSON:true})
.then(resp =>{
console.log(resp.body)})}
8.4 用什么做路由
官方提供了 vue-router 插件,路由是浏览器中的哈希值(#hash)与展示视图内容(template)之间的对应规则 vue 中的路由是:hash 与 component 的对应关系
8.5 父组件给子组件传值
父--->子:用 props;(单项数据流)组件想要使用父组件的数据,我们需要通过子组件的 prop 选项来获得父组件传过来的数据,当父组件的 name 发生改变,子组件也会自动地更新视图,如果是改动了子组件中的 prop 的值得时候,父组件中的值是不会响应的变化的,如果要修改 prop,就把 prop 赋值给一个局部变量,然后需要修改的话就修改这个局部变量,而不影响 prop
8.6 子组件给父组件传值
子-->父:用$.emit(event ,[..args]) 父组件用$.on 接收,通过在父组件 $on(eventName) 监听自 定义事件,当子组件里 $emit(eventName) 触发该自定义事件的时候,父组件执行相应的操作方法。
8.7 兄弟组件间传值
兄弟-->兄弟 :用 vuex(状态管理模式)和 event.bus( 空实例对象)
<!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>
</head>
<body>
<div id="app">
<aaa></aaa>
<bbb></bbb>
</div>
<script src="./vue.js"></script>
<script>
// 非父子组件通讯:
var bus = new Vue()
var vm = new Vue({
el: '#app',
data: {},
// 通过 components 来创建属于该实例的局部组件
components: {
aaa: {
template: `
<div>
<h1>组件A</h1>
<button @click="fn">传递数据给组件B</button>
</div>
`,
created() {
bus.$emit('bfn', '组件A说:你好组件B')
},
methods: {
fn() {
// 触发事件,传递数据
bus.$emit('bfn', '组件A说:你好组件B')
}
}
},
bbb: {
template: `
<div>
<hr>
<h1>组件B: {{ msg }}</h1>
</div>
`,
data() {
return {
msg: ''
}
},
// 在钩子函数中,绑定事件,这样以进入页面,事件就绑定好了,等待组件A触发
created() {
// 绑定事件,接受数据
bus.$on('bfn', data => {
this.msg = data
})
}
}
}
})
</script>
</body>
</html>
9. vuex 里面有一个仓库是 state(数据仓库)改变数据仓库数据的一些方法 motation.异步的
和 motation(里面放同步的方法)
9.1 双向数据绑定的原理
答:vue 中数据的双向绑定采用的时候,数据劫持的模式。其实主要是用了 Es5 中的 Object.defineProperty;来劫持每个属性的 getter,和 setter
vue 执行的过程(双向数据绑定的原理解析):
- 1 根据传入的 el 配置项,找到页面中需要被 Vue 管理的内容区域
- 2 遍历所有的后代元素,收集出现的所有指令(v-model)和表达式()
- 3 遍历传入 data 中的数据,分别通过 Object.defineProperty() 实现每个数据的 get/set
- 4 每个数据的 get/和 set 中,分别与页面中使用该数据的指令和表达式对应起来
- 5 将来当数据改变的时候,通过 设定好的 set,将数据的变化同步到页面中
9.2 路由的原理。
答:main.js 入口配置好路由,
在 app.vue 根组件中引入 router-view 标签,在 router 组件中配置各种路由 location.hash hashchange 事件
在访问 google plus 和网易 m 站时,细心的用户也许会发现页面之间的点击是通过 ajax 异步请求的,同时页面的 URL 发生了了改变。并且能够很好的支持浏览器的前进和后退。不禁让人想问,是什么有这么强大的功能呢?
HTML5 里引用了新的 API,就是 history.pushState 和 history.replaceState,就是通过这个接口做到无刷新改变页面 URL 的。
对应 history.pushState 和 history.replaceState 的具体用法可以参考下面这几篇文章:
HTML5 history 新特性 pushState、replaceState 传统的 ajax 的问题
虽然 ajax 可以无刷新改变页面内容,但无法改变页面 URL 为了更好的可访问性,内容发生改变后,改变 URL 的 hash。
但是 hash 的方式不能很好的处理浏览器的前进、后退等问题 有的浏览器引入了 onhashchange 的接口,不支持的浏览器只能定时去判断 hash 是否改变
ajax 的使用对搜索引擎很不友好,往往蜘蛛爬到的区域是空的 为了解决传统 ajax 带来的问题,HTML5 里引入了新的 API,即:history.pushState, history.replaceState 可以通过 pushState 和 replaceState 接口操作浏览器历史,并且改变当前页面的 URL。
9.3 数据请求的原理。
答:axios 原理也是 ajax。 3.3 vue2.0 源码的实现过程
10.工作中遇到的 bug?
浏览器兼容,移动端兼容问题
11.模块化与组件化
模块化开发以 js 文件功能分类, 组件化开发把 js,html,css 放到一个文件里,功能独立 vue~resource 支持 jsonp,vue2.0 以上不支持
12.axio 跨域问题怎么解决?
需要封装 jsonp,或者用 cros 服务器设置, 跨域:jsonp 支持低版本浏览器, cros 高版本浏览器,反向代理 服务器允许跨域的代码:
//允许所有的域名访问这个接口
header('Access-Control-Allow-Origin:*')
//允许www.study.com这个域名访问这个接口
header('Access-Control-Allow-Origin:http://www.jepson.com')
//vue获取更新后的数据
ajax 返回出现 error,datatype 没设置 模板引擎的原理是利用正则表达式定义标签替换{}语句中的内容 ,模板引擎的缺点不利于 seo,无法准确定位错误
h5css3 browersync 调试工具 httpcache 离线缓存
fiddler 拦截后端请求,模拟数据请求返回
测试前后端数据 get 请求直接在 url 地址栏,也可以用 fiddler,postmessage
13.协作测试 api 接口 ip 统一设置
const os = require('os')
var IPv4, net
const platform = os.platform()
switch (platform) {
case 'darwin':
// mac
net = os.networkInterfaces().en0
break
case 'win32':
// windows
net = os.networkInterfaces()['以太网']
break
default:
net = []
break
}
for (var i = 0; i < net.length; i++) {
if (net[i].family == 'IPv4') {
IPv4 = net[i].address
}
}
if (!IPv4) {
IPv4 = 'localhost'
}
14.多行溢出样式打包问题
/*! autoprefixer: off */
-webkit-box-orient: vertical;
/* autoprefixer: on */
15.生命周期的具体使用
- created 初始化静态数据
- created 在实例创建完成后被立即调用。在这一步,实例已完成以下的配置:数据观测 (data observer),属性和方法的运算,watch/event 事件回调。然而,挂载阶段还没开始,$el 属性目前不可见。
- mounted 初始化列表数据和搜索下拉数据 - mounted el 被新创建的 vm.$el 替换,并挂载到实例上去之后调用该钩子。如果 root 实例挂载了一个文档内元素,当 mounted 被调用时 vm.$el 也在文档内。 注意 mounted 不会承诺所有的子组件也都一起被挂载。如果你希望等到整个视图都渲染完毕,可以用 vm.$nextTick 替换掉 mounted
16.页面刷新 vuex 数据丢失
// 在页面加载时读取sessionStorage里的状态信息
const sessionStore = window.sessionStorage.getItem(storeKey.vuexStore)
if (sessionStore) {
let store = {}
Object.assign(store, this.$store.state, JSON.parse(sessionStore))
this.$store.replaceState(store)
window.sessionStorage.removeItem(storeKey.vuexStore)
}
// 在页面刷新时将vuex里的信息保存到sessionStorage里
// ie、谷歌、360 页面刷新执行顺序 onbeforeunload -> onunload -> onload,关闭执行顺序 onbeforeunload -> onunload
// firefox 页面刷新只执行 onunload,页面关闭只执行 onbeforeunload
let eventName = 'beforeunload'
const fireFox = navigator.userAgent.indexOf('Firefox') !== -1
if (fireFox) {
eventName = 'unload'
}
window.addEventListener(eventName, () => {
// 根据用户名是否存在判断是退出还是刷新
const uname = this.$store.state.account.userInfo.uname
if (uname) {
window.sessionStorage.setItem(
storeKey.vuexStore,
JSON.stringify(this.$store.state)
)
}