元素设置z-index之后上层阻挡下层的点击事件
对上层元素使用:
pointer-events: none; /元素永远不会成为鼠标事件的target(目标)/
这样点击事件就能穿透上层元素,但若是你想让上层元素也能得到点击事件,只需要在你需要获得点击的上层元素子元素中加入:
pointer-events: auto;
element-ui
Col Attributes
xm=x-small(extra small特小号)
sm(small小号)
md(medium中号)
lg(large大号)
x-large(特大号)
自定义返回按钮
<!-- 返回键 -->
<view class="nav-box">
<status-bar></status-bar>
<view class="navi" style="height: 44px;">
<view class="back-box">
<view class="btn-menu">
<view class='iconfont icon-fenxiang1' @click="clickBack"></view>
<view class="iconfont icon-home" @click="clickHome"></view>
</view>
</view>
<view class="title-box">
<view class="title rk-line">房屋详情</view>
</view>
</view>
</view>
<view class="nav-space">
<status-bar></status-bar>
<view class="navi" style="height: 44px;"></view>
</view>
// js
clickBack(){
uni.navigateBack();
},
clickHome(){
uni.reLaunch({
url:"/pages/index/index",
})
},
//css
.nav-box{
background-color: #fff;
position: fixed;
z-index: 999;
left: 0;
top: 0;
right: 0;
.navi{
display: flex;
align-items: center;
justify-content: center;
position: relative;
.back-box {
position: absolute;
left: 16rpx;
.btn-menu{
display: flex;
align-items: center;
width: 140rpx;
height: 54rpx;
background: rgba(0, 0, 0, .25);
border-radius: 100rpx;
.iconfont {
flex: 1;
text-align: center;
color: #fff;
box-sizing: border-box;
&.icon-fenxiang1 {
border-right: 1px solid #fff;
}
}
}
}
.title-box{
max-width: 50%;
.title{
width: 100%;
font-size: 16px;
font-weight: bold;
}
}
}
}
.nav-space {
background-color: transparent;
}
阻止遮罩层滑动事件穿透
// template 父视图添加
@touchmove.stop.prevent="moveStop"
// js method 实现空方法 moveStop
数据源更新
当数据源嵌套有多层的时候,改变最内层的值,页面未刷新,可使用
$set
;如:
this.dataList.forEach(item=>{
if(this.value.includes(item[this.map.value])){
// item.selected = true;
this.$set(item,'selected',true)
}else{
// item.selected = false;
this.$set(item,'selected',false)
}
})
replace()
let previewDes = this.desContent;
this.quickType.forEach(item => {
console.log("rk===>[foreach]" + item.title);
let filterStr = item.title+':';
previewDes = previewDes.replace(RegExp(filterStr, "g"),'');
});
previewDes = previewDes.replace(/[\r\n]/g,'');
CSS部分
背景图写法
background-image: url('~@/static/images/code-bg.png');
background:linear-gradient(to bottom, #EA7E00 0%, #E82F00 100%);
底部固定布局、刘海屏适配
position: fixed;
bottom: 98rpx;
bottom: calc(98rpx + constant(safe-area-inset-bottom));
bottom: calc(98rpx + env(safe-area-inset-bottom));
弹窗
<view class="com-alert-pop">
<view class="mask" v-if="alertShow"></view>
<view class="popup-alert" :class="alertShow?'popup-active':''">
<view class="title"></view>
<view class="des-box">
<text class="des"></text>
</view>
<view class="bot-box">
<view v-if="showCancel" class="bot-btn cancel" @click="clickCancel" ></view>
<view class="bot-btn confirm" @click="clickConfirm"></view>
</view>
</view>
</view>
.mask {
position: fixed;
left: 0;
top: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.3);
z-index: 1000;
}
.popup-alert {
position: fixed;
top: 45%;
left: 50%;
z-index: 1001;
width: 544rpx;
padding-top: 48rpx;
// padding-bottom: 30rpx;
border-radius: 24rpx;
background-color: #ffffff;
transform: translate(-50%, -50%) scale(0);
opacity: 0;
transition: 0.3s;
line-height: 1;
text-align: center;
color: #282828;
.title {
max-width: 90%;
margin-right: auto;
margin-left: auto;
font-weight: bold;
font-size: 32rpx;
}
.des-box {
margin-top: 24rpx;
padding: 0 30rpx;
.des {
font-size: 28rpx;
color:#323232;
line-height: 1.1em;
}
}
.bot-box {
display: flex;
justify-content: space-evenly;
margin: 30rpx 0 0;
border-top: 2rpx solid #eee;
.bot-btn {
width: 50%;
padding: 20rpx 50rpx;
color:#323232;
// border-radius: 50rpx;
box-sizing: border-box;
}
.cancel {
// background:#969696;
border-right: 3rpx solid #eee;
}
.confirm {
color: $main-color;
// background: linear-gradient(to right, #FE6E21 0%, #FF1F13 100%);
}
}
}
.popup-active {
transform: translate(-50%, -50%) scale(1);
opacity: 1;
}
name:'alertPop',
props:{
alertShow:{
type:Boolean,
default:false,
},
showCancel:{
type:Boolean,
default:true,
},
alertTitle:{
type:String,
default:'提示',
},
alertContent:{
type:String,
default:'提示内容',
},
cancelTitle:{
type:String,
default:'取消',
},
confirmTitle:{
type:String,
default:'确认',
}
},
弹窗2
<template>
<view class="com-alert-pop">
<view class="mask" v-if="alertShow"></view>
<view class="popup-alert" :class="alertShow?'popup-active':''">
<view class="title"></view>
<view class="right-top-close" @click="clickClose">
<uni-icons type="closeempty" color="#969696"></uni-icons>
</view>
<view class="des-box">
<text class="des"></text>
</view>
<view class="bot-box2">
<view v-if="showCancel" class="bot-btn cancel" @click="clickCancel"></view>
<view class="bot-btn confirm" @click="clickConfirm"></view>
</view>
</view>
</view>
</template>
<script>
export default {
name: 'rk-alert',
props: {
alertShow: {
type: Boolean,
default: false,
},
showCancel: {
type: Boolean,
default: true,
},
alertTitle: {
type: String,
default: '提示',
},
alertContent: {
type: String,
default: '提示内容',
},
cancelTitle: {
type: String,
default: '取消',
},
confirmTitle: {
type: String,
default: '确认',
}
},
methods:{
clickClose(){
this.$emit('close')
},
clickCancel(){
this.$emit('cancel')
},
clickConfirm(){
this.$emit('confirm')
},
}
}
</script>
<style lang="scss">
.mask {
position: fixed;
left: 0;
top: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.3);
z-index: 1000;
}
.popup-alert {
position: fixed;
top: 45%;
left: 50%;
z-index: 1001;
width: 544rpx;
padding-top: 48rpx;
// padding-bottom: 30rpx;
border-radius: 24rpx;
background-color: #ffffff;
transform: translate(-50%, -50%) scale(0);
opacity: 0;
transition: 0.3s;
line-height: 1;
text-align: center;
color: #282828;
.right-top-close{
position: absolute;
right: 0rpx;
top: 0rpx;
padding: 20rpx;
}
.title {
max-width: 90%;
margin-right: auto;
margin-left: auto;
font-weight: bold;
font-size: 32rpx;
}
.des-box {
margin-top: 30rpx;
padding: 0 30rpx;
.des {
font-size: 28rpx;
color: #323232;
line-height: 1.1em;
}
}
// 样式1
.bot-box {
display: flex;
justify-content: space-evenly;
margin: 30rpx 0 0;
border-top: 2rpx solid #eee;
.bot-btn {
width: 50%;
padding: 20rpx 50rpx;
color: #323232;
box-sizing: border-box;
}
.cancel {
border-right: 3rpx solid #eee;
}
.confirm {
color: $app-color-main;
}
}
// 样式2
.bot-box2{
display: flex;
justify-content: space-evenly;
margin: 30rpx 0;
.bot-btn {
width: 40%;
padding: 20rpx 50rpx;
color: #323232;
box-sizing: border-box;
background-color: $app-color-gold;
border-radius: 10rpx;
color: #fff;
font-size: 13px;
}
.cancel {
}
.confirm {
background-color: #000;
color: $app-color-gold;
}
}
}
.popup-active {
transform: translate(-50%, -50%) scale(1);
opacity: 1;
}
</style>
弹窗3
<template>
<view class="com-wifi-pop">
<view class="wifi-mask" v-if="alertShow"></view>
<view class="wifi-popup-alert" :class="{'wifi-popup-active':alertShow}">
<view class="title"></view>
</view>
</view>
</template>
<script>
export default {
name: 'rk-alert',
props: {
alertShow: {
type: Boolean | Number,
default: false,
},
},
methods:{
}
}
</script>
<style lang="scss">
.mask {
position: fixed;
left: 0;
top: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.3);
z-index: 100;
}
.wifi-popup-alert {
position: fixed;
top: 45%;
left: 50%;
z-index: 101;
width: 544rpx;
padding-top: 48rpx;
border-radius: 24rpx;
background-color: #ffffff;
transform: translate(-50%, -50%) scale(0);
opacity: 0;
transition: 0.3s;
line-height: 1;
text-align: center;
color: #282828;
}
.wifi-popup-active {
transform: translate(-50%, -50%) scale(1);
opacity: 1;
}
</style>
elment-dialog
<template>
<div class="rp-detail-box" v-if="showRppop">
<el-dialog title="" :visible.sync="showRppop" width="650px" :before-close="handleClose" :close-on-press-escape="false" :close-on-click-modal="false">
<div class="rp-content">
</div>
</el-dialog>
</div>
</template>
<script>
export default {
name: 'rp-detail',
props: {
showRppop: {
type: Boolean,
default: false
}
},
data() {
return {};
},
methods: {
handleClose() {
this.$emit('close');
}
}
};
</script>
<style lang="scss" scoped></style>
动画
.animation-class {
// 定义动画名称 mymove
animation: mymove 1s;
-moz-animation: mymove 1s;
-webkit-animation: mymove 1s;
-o-animation: mymove 1s;
// 执行1次
animation-iteration-count: 1;
-moz-animation-iteration-count: 1;
-webkit-animation-iteration-count: 1;
-o-animation-iteration-count: 1;
// 执行完毕停留在动画结束的最后一帧
animation-fill-mode: forwards;
-moz-animation-fill-mode: forwards;
-webkit-animation-fill-mode: forwards;
-o-animation-fill-mode: forwards;
}
@keyframes mymove {
0% {
transform: translate(-100%);
}
100% {
transform: translate(0);
}
}
单行、多行文本
display: -webkit-box;
overflow: hidden;
text-overflow: ellipsis;
word-wrap: break-word;
white-space: normal !important;
-webkit-box-orient: vertical;
-webkit-line-clamp: 1;
scroll-view的横向滚动
<scroll-view :scroll-x="true" class="scrollCon">
<view class="monthItem">1月</view>
<view class="monthItem">2月</view>
<view class="monthItem">3月</view>
<view class="monthItem">4月</view>
<view class="monthItem">5月</view>
<view class="monthItem">6月</view>
<view class="monthItem">7月</view>
<view class="monthItem">8月</view>
<view class="monthItem">9月</view>
<view class="monthItem">10月</view>
<view class="monthItem">11月</view>
<view class="monthItem">12月</view>
</scroll-view>
CSS部分
.scrollCon{
white-space: nowrap;
display: flex;
align-items: center;
}
.monthItem{
display: inline-block;
width: calc(100% / 6);
font-size: 26rpx;
color: #3D3D3D;
text-align: center;
}
核心
// scroll-view的外层元素需要
white-space: nowrap;
// scroll-view的内层view元素需要
display: inline-block;
返回键
from 触发返回行为的来源:
‘backbutton’: 左上角导航栏按钮及安卓返回键;
‘navigateBack’: uni.navigateBack() 方法;
支付宝小程序端不支持返回此字段.
export default {
data() {
return {};
},
onBackPress(options) {
console.log('from:' + options.from)
}
}
获取元素
监听
let active;
document.getElementById("pageIndexs").children.forEach((dom) => {
dom.addEventListener("click", (e) => {
/*
* stopPropagation() 方法防止调用相同事件的传播。
* 传播意味着向上冒泡到父元素或向下捕获到子元素。
* https://www.w3school.com.cn/jsref/event_preventdefault.asp
* https://www.w3school.com.cn/jsref/event_stoppropagation.asp
*/
e.stopPropagation();
e.preventDefault();
if (dom === active) return;
dom.classList.add("borderShow");
active && active.classList.remove("borderShow");
active = dom;
});
});
复用
# 会复用
<template v-if="loginType === 'username'">
<label>Username</label>
<input placeholder="Enter your username">
</template>
<template v-else>
<label>Email</label>
<input placeholder="Enter your email address">
</template>
# 不会复用
<template v-if="loginType === 'username'">
<label>Username</label>
<input placeholder="Enter your username" key="username-input">
</template>
<template v-else>
<label>Email</label>
<input placeholder="Enter your email address" key="email-input">
</template>
v-if vs v-show
一般来说,v-if
有更高的切换开销,而v-show
有更高的初始渲染开销.因此,如果需要非常频繁地切换,则使用v-show
较好;如果在运行时条件很少改变,则使用v-if
较好.
v-for
<ul id="v-for-object" class="demo">
值、键、下标
<li v-for="(value, name, index) in object">
</li>
</ul>
new Vue({
el: '#v-for-object',
data: {
object: {
title: 'How to do lists in Vue',
author: 'Jane Doe',
publishedAt: '2016-04-10'
}
}
})
小程序定位权限
chooseLocation:fail the api need to be declared in the requiredPrivateInfos field in app.json/ext.json错误的解决办法
解决方案源码视图
-"mp-weixin"
节点,添加requiredPrivateInfos
如下:
"mp-weixin" : {
"appid" : "wxec7ecf3d44019688",
"setting" : {
"urlCheck" : false,
"es6" : false
},
"usingComponents" : true,
"betterScopedSlots" : true,
"permission" : {
"scope.userLocation" : {
"desc" : "演示在onShow生命周期获取地理位置"
}
},
"requiredPrivateInfos" : [ "chooseLocation", "getLocation" ]
},
H5定位权限
h5节点"h5"
内添加:
"sdkConfigs": {
// 使用地图或位置相关功能必须填写其一
"maps": {
"qqmap": {
// 腾讯地图秘钥 https://lbs.qq.com/dev/console/key/manage
"key": ""
},
"google": {
// 谷歌地图秘钥(HBuilderX 3.2.10+)https://developers.google.com/maps/documentation/javascript/get-api-key
"key": ""
},
"amap": {
// 高德地图秘钥(HBuilderX 3.6.0+)https://console.amap.com/dev/key/app
"key": "a88713f7236ab59b9a3a537544b6c9bb",
// 高德地图安全密钥(HBuilderX 3.6.0+)https://console.amap.com/dev/key/app
"securityJsCode": "b5254b0eb737c762b70b83461e2194d8",
// 高德地图安全密钥代理服务器地址(HBuilderX 3.6.0+)https://lbs.amap.com/api/jsapi-v2/guide/abc/prepare
"serviceHost": "https://api.bspapp.com"
}
}
}
Uniapp-微信小程序设置打包忽略文件
设置方法源码视图
-"mp-weixin"
节点,添加packOptions
如下:
{
"packOptions": {
"ignore": [{
"type": "file",
"value": "test/test.js"
}, {
"type": "folder",
"value": "test"
}, {
"type": "suffix",
"value": ".webp"
}, {
"type": "prefix",
"value": "test-"
}, {
"type": "glob",
"value": "test/**/*.js"
}, {
"type": "regexp",
"value": "\\.jsx$"
}]
}
}
其中,type 可以取的值为 folder、file、suffix、prefix、regexp2、glob2,分别对应文件夹、文件、后缀、前缀、正则表达式、Glob 规则。所有规则值都会自动忽略大小写。
注 1: value 字段的值若表示文件或文件夹路径,以小程序目录 (miniprogramRoot) 为根目录。
注 2: regexp、glob 仅 1.02.1809260 及以上版本工具支持。
uniCloud项目配置
【App报错】Client platform is app, but app-plus was found in config.
【解决】
【配置路径】uni-id的云端配置文件在uniCloud/cloudfunctions/common/uni-config-center/uni-id/config.json中
旧项目建议将所有platform为app的场景统一为app-plus,即建议使用如下配置
// 以下仅列出相关配置
{
"preferedAppPlatform": "app-plus", // uni-id内部会将收到的app平台全部转化为app-plus平台
"app-plus": { // 配置内的平台名称和preferedAppPlatform保持一致
"oauth": {}
}
}
新项目建议将platform统一为app,即建议使用如下配置
// 以下仅列出相关配置
{
"preferedAppPlatform": "app", // uni-id内部会将收到的app-plus平台全部转化为app平台
"app": { // 配置内的平台名称和preferedAppPlatform保持一致
"oauth": {}
}
}
uniapp文件引用
script
import "./static/yzb-icon.css";
style
@import './static/yzb-icon.css';
uniapp应用template中点语法取值问题
1.当返回数据嵌套了三层甚至四层的时候template
中如果使用item.one.two.three.content
可能会出问题;
2.需要格式化(如:日期);
这时候可以采用以下形式
template中
<template>
<text class="text-color-grey">{ formatDates(item.start_date) }-{ formatDates(item.end_date) }</text>
</template>
script中
export default {
methods: {
formatDates(time) {
if (time == null || time === '') {
return 'N/A';
}
let date = new Date(time);
// return formatDate(date, 'yyyy-MM-dd hh:mm:ss')
return formatDate(date, 'yyyy.MM');
},
}
}