1. vue.js 组件添加事件
<Buttom @click.native='handleClick'>
.native 是用来区分组件的事件和组件内部的事件,当然也可以是用的$emit('myClick') 从组件内部派发事件来实现
2. vue 微信支付遇到的坑
使用的vue-router 的hash模式, 所以页面的路径是www.ssss.com/shop/#/home 的样子,但是微信支付目录验证不支持这种hash模式,所以在微信看来目录是错误。
// Recharge.vue
created() {
let config = {};
config.url = window.location.href;
// 判断当前url是否存在?参数匹配符
if(!config.url.match(/\?/)) {
location.replace(window.location.href.split('#')[0] + '?' + window.location.hash);
return ;
}
}
3. 微信中拉起微信支付控件
1. 使用wx的[jssdk](https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1421141115);是比较新的方式,需要引入威信度的jssdk
2. 使用老的WeixinJSBridgeReady方式拉起支付控件,
这次使用的是后面这种方法: 步需要引入七牛的jssdk直接就可以拉起
```
handleWxpay() {
if(this.isweixin) {
//options 是接口返回的wx的一些配置信息
this.wxReadyToPay(options)
}else {
console.log('open in wx')
}
},
onBridgeReady(options){
let that = this
console.log(options)
WeixinJSBridge.invoke(
'getBrandWCPayRequest',
options,
function(res){
console.log(res);
//使用以上方式判断前端返回,微信团队郑重提示:res.err_msg将在用户支付成功后返回 ok,但并不保证它绝对可靠。
switch(res.err_msg){
case "get_brand_wcpay_request:ok": //支付成功
console.log('支付成功')
// that.$router.push({path:'/SettlemenSuccess'})
break;
case "get_brand_wcpay_request:cancel": //支付取消
console.log('取消支付')
break;
case "get_brand_wcpay_request:fail": //支付失败
console.log('支付失败')
break;
default:
// console.log(res.err_msg);
break;
}
}
)
},
wxReadyToPay(options){
let that = this
if (typeof WeixinJSBridge == "undefined"){
if( document.addEventListener ){
document.addEventListener('WeixinJSBridgeReady', that.onBridgeReady(options), false);
}else if (document.attachEvent){
document.attachEvent('WeixinJSBridgeReady', that.onBridgeReady(options));
document.attachEvent('onWeixinJSBridgeReady', that.onBridgeReady(options));
}
}else{
that.onBridgeReady(options);
}
},
isweixin() {
const ua = window.navigator.userAgent.toLowerCase();
if(ua.match(/MicroMessenger/i) == 'micromessenger'){
return true;
} else {
return false;
}
},
```
4. 微信中预览图片(类似微信支付)
1.微信的jssdk
2.WeixinJSBridgeReady
```
handleToSwiper(index) {
this.current = index
this.wxReady(index)
},
wxSwiper(index){
let newImgs = this.newImgs.map((item) => {
return this.host + item.filename + this.previewParameter //添加的七牛图片的参数,根据自己项目的需求添加
})
WeixinJSBridge.invoke('imagePreview', {
current: newImgs[index],
urls:newImgs
}
)
},
wxReady(index){
let that = this
if (typeof WeixinJSBridge == "undefined"){
if( document.addEventListener ){
document.addEventListener('WeixinJSBridgeReady', that.wxSwiper(index), false);
}else if (document.attachEvent){
document.attachEvent('WeixinJSBridgeReady', that.wxSwiper(index));
document.attachEvent('onWeixinJSBridgeReady', that.wxSwiper(index));
}
}else{
that.wxSwiper(index);
}
},
```
[七牛图片处理](https://developer.qiniu.com/dora/manual/1270/the-advanced-treatment-of-images-imagemogr2#imagemogr2-thumbnail-spec) 例如:`?imageView2/1/w/200/h/200/interlace/1/q/100'`参数,让图片,无论是横图还是竖图在n * n大小的范围你不变形的显示出来
5. vue开发环境 跨域问题 使用代理的方式解决
在开发的过程中后端给的接口,存在跨域的问题,导致在本地调试无法正常的进行,通常解决这样的问题有两种方式:
1. 后端允许跨域,(为了安全可以加一些限制,有很多种的方法,可以和后端的小伙伴进行协商解决)
2. 利用node.js 在vue-cli中使用 代理的方式,服务器去请求服务器不会存在跨域的问题。
proxyTable的配置:实际是webpack中devServer
的proxy
的配置
//在vue-cli config下dev环境配置中
const getIP = utils.getIP // 获取ip的方法
const host = getIP() ? getIP() : 'localhost'
const port = 9092
const proxyTableUrl = `http://${host}:${port}/yiqi`
const baseUrl = 'http://xxxxxxxxx.cn/'
proxyTable: {
[proxyTableUrl]: {
target: baseUrl,
changeOrigin: true, //允许跨域
pathRewrite: {
'^/yiqi': ''
}
}
}
访问 http://192.168.13.233:8081/yiqi/apixxxxx
代理了http://xxxxxxxxx.cn/apixxxxx
6. 获取的本地内网的ip 目的是为了达到在手机预览的目的
//环境是mac机子 window系统存在问题
const glob = require('glob');
module.exports = {
getIP:function () {
var os = require('os')
var IPv4 = '127.0.0.1'
var interfaces = os.networkInterfaces()
for (var key in interfaces) {
interfaces[key].some(function (details) {
if (details.family == 'IPv4' && key == 'en0') {
IPv4 = details.address
return true
}
})
}
return IPv4
}
}
7. 使用webpack 配置外部引入,避免vender.js打包过大的问题
在使用vue-cli进行打包的时候,会把一些第三方的插件打包在vender.js中,随着项目的迭代,会因为插件使用的越来越多导致vender.js过大。这种情况下可以把一些插件使用cdn,或着使用自己的地址来作为外部的引用来使用
//wwbpack.base.config.js
externals: {//CDN 引入文件 减少vendor.js体积
// vue: 'Vue',
swiper:"Swiper"
},
在index.html模板中引入你想使用的插件的cdn
在项目中就可以正常的使用Swiper插件了
// sipwer.vue
import '@/plugins/css/swiper-3.4.2.min.css';
import Swiper from 'swiper'
在项目打包之后就在控制台查看就会看见已经引入了你想要的js
当然你也可以吧vue从外部引入使用 。本人的项目选择打包在vender.js中。
8. 图片 ,字体 , 媒体 是否打包在js中的配置(webpack)
{
test: /\.(png|jpe?g|gif|svg)(\?.*)?$/,
loader: 'url-loader',
options: {
// 超过10字节 会打包到img目录下,否则打包在js中; 想要打包在 img limit:1
limit: 1,
name: utils.assetsPath('img/[name].[hash:7].[ext]')
}
},
{
test: /\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/,
loader: 'url-loader',
options: {
limit: 10000,
name: utils.assetsPath('media/[name].[hash:7].[ext]')
}
},
{
test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/,
loader: 'url-loader',
options: {
limit: 10000,
name: utils.assetsPath('fonts/[name].[hash:7].[ext]')
}
}
9. webpack路径的配置 结合vue-cli
//vue-cli 中配置
build:{
// Paths
assetsRoot: path.resolve(__dirname, '../dist'),
assetsSubDirectory: '',
assetsPublicPath: '/shop/',
}
// webpack
output: {
path: config.build.assetsRoot,
filename: utils.assetsPath('js/[name].[chunkhash].js'),
chunkFilename: utils.assetsPath('js/[id].[chunkhash].js'),
publicPath: process.env.NODE_ENV === 'production'
? config.build.assetsPublicPath
: config.dev.assetsPublicPath
},
plugins:[
// copy custom static assets
new CopyWebpackPlugin([
{
from: path.resolve(__dirname, '../static'),
to: config.build.assetsSubDirectory,
ignore: ['.*']
}
])
]
assetsRoot
决定了webpack
中输出的路径;在目录下的 dist目录
assetsSubDirectory
决定了打包之后dist下的目录,比如
assetsSubDirectory:''
如果
assetsSubDirectory:'static'
assetsPublicPath
决定了放在服务器时的项目路径
如果assetsPublicPath:'/shop/'
如果assetsPublicPath:''
则上面的路径中的shop不存在,根据自己姓名不同的需求来配置该路径
10. vconsole在vue中的使用
const debug = process.env.NODE_ENV !== 'production'
if (debug) { //vconsole 移动端调试
const Vconsole = () => import('vconsole')
let vConsole = new Vconsole()
vConsole.then(res => {
vConsole = new res()
}, err => {
console.log(err)
})
}
11. 分包加载
// 没有进行分包的加载 会在首次进入首页的时候加载所有的组件
import ProfilePicture from '@/components/ProfilePicture'
// 就版本的分包加载的方法
const ProfilePicture = r => require.ensure([], () => r(require('../components/ProfilePicture')), 'ProfilePicture')
// 新版webpack 3.6的分包加载
const ProfilePicture = () => import('../components/ProfilePicture')
12. tabbar的封装 当然方法也很多,懒得写了直接贴代码吧
<template>
<div id="table-bar" class="border-top-1px">
<div class="items" @click='switchTab("Home")'
:class="$route.path.indexOf('Home') !== -1? 'active' : ''">
<div class="icon">
<img v-if="$route.path.indexOf('Home') !== -1"
src="../images/icon_shouye.png" alt='active'/>
<img v-else src="../images/icon_shouye2.png" alt="normal">
</div>
<div class="text">
首页
</div>
</div>
<div class="items" @click='switchTab("Order")'
:class="$route.path.indexOf('Order') !== -1? 'active' : ''">
<div class="icon">
<img v-if="$route.path.indexOf('Order') !== -1"
src="../images/icon_dingdan.png" alt='active'>
<img v-else src="../images/icon_dingdan2.png" alt="normal">
</div>
<div class="text">
订单
</div>
</div>
<div class="items" @click='switchTab("Mine")'
:class="$route.path.indexOf('Mine') !== -1? 'active' : ''">
<div class="icon">
<img v-if="$route.path.indexOf('Mine') !== -1"
src="../images/icon_me2.png" alt="active">
<img v-else src="../images/icon_me.png" alt="normal">
</div>
<div class="text">
我的
</div>
</div>
</div>
</template>
<script>
export default {
name: "TableBar",
data(){
return {
}
},
methods: {
switchTab(path) {
this.$router.replace(path)
}
},
mounted() {
},
}
</script>
<style lang='less' scoped>
@import '../styles/mixin.less';
@import '../styles/base.less';
#table-bar{
display: flex;
justify-content: space-between;
align-items: center;
width: 100%;
height: 0.5rem;
}
.items{
flex: 1;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
height: 100%;
color: @subtitle-color;
font-size: 0.11rem;
}
.text{
font-size: 0.11rem;
}
.icon{
width: 0.22rem;
height: 0.22rem;
overflow: hidden;
margin-bottom: 0.03rem;
img{
width: 100%;
height: 100%;
}
}
.active{
color: @title-color;
}
</style>
13. swiper插件 自定义分页器 直接上代码吧
<template>
<div class="swiper">
<div class="swiper-container">
<div class="swiper-wrapper">
<div class="swiper-slide"
v-for='(item, index) in banners'
:key='index'>
<img :src="item" alt="" style='width:100%;height:100%'>
</div>
</div>
<!-- 如果需要分页器 -->
<!-- <div class="swiper-pagination"></div> -->
</div>
<!--自定义的分页器 非官方-->
<div v-cloak class="pagination swiper-pagination-custom">
<span class="swiper-pagination-customs"
:class="{'swiper-pagination-customs-active':index+1 == swiperActIndex}"
v-for="(item, index) in banners"
:key='index'></span>
</div>
</div>
</template>
<script>
import '@/plugins/css/swiper-3.4.2.min.css';
import Swiper from 'swiper';
export default {
name: "",
data: () => ({
swiperActIndex:1
}),
props:['banners'],
methods:{
_initScroll() {
let that = this
if(this.mySwiper) {
this.mySwiper.destroy(true, true)
}else {
this.mySwiper = new Swiper('.swiper-container', {
autoplay: 3000,// 可选选项,自动滑动
speed:1000,
loop:true,
autoplayDisableOnInteraction:false, // hack 滑动后自动轮播停止
pagination: '.swiper-pagination',
paginationType : 'custom',
paginationCustomRender(mySwiper, current, total) {
var customPaginationHtml = "";
for(var i = 0; i < total; i++) {
if(i == (current - 1)) {
customPaginationHtml += '<span class="swiper-pagination-customs swiper-pagination-customs-active"></span>';
} else {
customPaginationHtml += '<span class="swiper-pagination-customs"></span>';
}
}
return customPaginationHtml;
},
onSlideChangeStart: function(swiper){
// loop属性会影响len的值
let len = swiper.slides.length - 2
if(swiper.activeIndex === len+1) {
that.swiperActIndex = 1
}else if(swiper.activeIndex === 0) {
that.swiperActIndex = len
}else {
that.swiperActIndex = swiper.activeIndex
}
}
})
}
}
},
mounted() {
setTimeout(function(){
this._initScroll()
}.bind(this),300)
}
}
</script>
<style lang="less" scoped>
@import '../styles/base.less';
.swiper-container{
width: 3.3rem;
height: 3.3rem;
}
.swiper-slide{
width: 100%;
height: 100%;
}
.pagination{
display: flex;
justify-content: center;
align-items: flex-end;
height: 0.2rem;
}
/*包裹自定义分页器的div的位置等CSS样式*/
.swiper-pagination-custom {
bottom: -0.2rem;
left: 0;
width: 100%;
}
/*自定义分页器的样式,这个你自己想要什么样子自己写*/
.swiper-pagination-customs {
width: 0.08rem;
height: .01rem;
display: inline-block;
background: #000;
opacity: .3;
margin: 0 .07rem;
background-color: @subtitle-color
}
/*自定义分页器激活时的样式表现*/
.swiper-pagination-customs-active {
opacity: 1;
background-color: @title-color;
}
</style>
-
webpack打包后可以看见源码
原理是你使用了 webpack 的 source-map 功能,与 vue 无关。
source-map 是“打包压缩后的代码”和“源代码”的对应关系,高级浏览器会自动加载这个文件以便调试。
解决办法:
将 config/index.js 中 build 下的 productionSourceMap: true,
改为 productionSourceMap: false, 即可.