员工添加编辑员工列表
This commit is contained in:
@@ -23,7 +23,7 @@ const employeesConfig = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 添加员工
|
// 添加员工
|
||||||
const addEmployees = (data) => {
|
const addEmployees = data => {
|
||||||
return request({
|
return request({
|
||||||
url: 'companies/employees',
|
url: 'companies/employees',
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
@@ -31,8 +31,35 @@ const addEmployees = (data) => {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 员工详情
|
||||||
|
const employeesInfo = id => {
|
||||||
|
return request({
|
||||||
|
url: 'companies/employees/' + id
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除员工
|
||||||
|
const employeesDelete = id => {
|
||||||
|
return request({
|
||||||
|
url: 'companies/employees/' + id,
|
||||||
|
method: 'DELETE'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编辑员工
|
||||||
|
const employeesPut = (id, data) => {
|
||||||
|
return request({
|
||||||
|
url: 'companies/employees/' + id,
|
||||||
|
method: 'PUT',
|
||||||
|
data
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
export {
|
export {
|
||||||
employees,
|
employees,
|
||||||
employeesConfig,
|
employeesConfig,
|
||||||
addEmployees
|
addEmployees,
|
||||||
|
employeesInfo,
|
||||||
|
employeesDelete,
|
||||||
|
employeesPut
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -190,6 +190,7 @@
|
|||||||
"style" : {
|
"style" : {
|
||||||
"navigationBarTitleText": "员工",
|
"navigationBarTitleText": "员工",
|
||||||
"app-plus":{
|
"app-plus":{
|
||||||
|
"bounce":"none",
|
||||||
"titleNView": {
|
"titleNView": {
|
||||||
"backgroundColor": "#FFFFFF",
|
"backgroundColor": "#FFFFFF",
|
||||||
"buttons": [
|
"buttons": [
|
||||||
|
|||||||
@@ -24,7 +24,7 @@
|
|||||||
<input type="text" v-model="job" placeholder="职业"/>
|
<input type="text" v-model="job" placeholder="职业"/>
|
||||||
</view>
|
</view>
|
||||||
<view class="info-inputs">
|
<view class="info-inputs">
|
||||||
<picker :range="section" range-key="name" @change="pickerChange">
|
<picker :range="section" range-key="name" :value="sectionIndex" @change="pickerChange">
|
||||||
<view class="picker-text">
|
<view class="picker-text">
|
||||||
{{section[sectionIndex].name}}
|
{{section[sectionIndex].name}}
|
||||||
<uni-icons class="icon" type="arrowdown" color="#555"></uni-icons>
|
<uni-icons class="icon" type="arrowdown" color="#555"></uni-icons>
|
||||||
@@ -33,6 +33,15 @@
|
|||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
<view class="jurisdiction switch">
|
||||||
|
<view class="item">
|
||||||
|
<label>
|
||||||
|
<view class="item-title">设为推荐</view>
|
||||||
|
<view class="item-info">设为推荐员工将在企业自媒体中优先展示</view>
|
||||||
|
<switch class="item-switch" :checked="isPosition" color="#e93340" @change="switchChange" />
|
||||||
|
</label>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
<!-- 权限设置 -->
|
<!-- 权限设置 -->
|
||||||
<view class="title">权限设置</view>
|
<view class="title">权限设置</view>
|
||||||
<view class="jurisdiction">
|
<view class="jurisdiction">
|
||||||
@@ -41,24 +50,25 @@
|
|||||||
<label>
|
<label>
|
||||||
<view class="item-title">{{item.title}}</view>
|
<view class="item-title">{{item.title}}</view>
|
||||||
<view class="item-info">{{item.description}}</view>
|
<view class="item-info">{{item.description}}</view>
|
||||||
<checkbox class="item-checkbox" color="#c82626" :value="item.permission_id" />
|
<checkbox class="item-checkbox" :checked="item.check" color="#e93340" :value="item.permission_id" />
|
||||||
</label>
|
</label>
|
||||||
</view>
|
</view>
|
||||||
</checkbox-group>
|
</checkbox-group>
|
||||||
</view>
|
</view>
|
||||||
<!-- 按钮 -->
|
<!-- 按钮 -->
|
||||||
<view class="add-btns">
|
<view class="add-btns">
|
||||||
<button size="default" @click="onAddEmployees">确认添加</button>
|
<button size="default" @click="onAddEmployees">{{type === 'PUT' ? '修改': '添加'}}</button>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { employeesConfig, addEmployees } from '@/apis/interfaces/employees'
|
import { employeesConfig, addEmployees, employeesInfo, employeesPut } from '@/apis/interfaces/employees'
|
||||||
import { uploads } from '@/apis/interfaces/uploading'
|
import { uploads } from '@/apis/interfaces/uploading'
|
||||||
export default {
|
export default {
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
type : '',
|
||||||
loging : true,
|
loging : true,
|
||||||
section : [],
|
section : [],
|
||||||
permissions : [],
|
permissions : [],
|
||||||
@@ -70,17 +80,53 @@
|
|||||||
},
|
},
|
||||||
name : '',
|
name : '',
|
||||||
phone : '',
|
phone : '',
|
||||||
job : ''
|
job : '',
|
||||||
|
isPosition : false
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
|
if(this.$Route.query.type === 'PUT'){
|
||||||
|
this.type = 'PUT'
|
||||||
|
uni.setNavigationBarTitle({
|
||||||
|
title: '编辑员工'
|
||||||
|
})
|
||||||
|
}
|
||||||
employeesConfig().then(res => {
|
employeesConfig().then(res => {
|
||||||
this.section = res.store
|
this.section = res.store
|
||||||
this.permissions = res.permissions
|
if(this.$Route.query.type === 'PUT'){
|
||||||
this.loging = false
|
employeesInfo(this.$Route.query.id || 4).then(res => {
|
||||||
|
let permissionIds = []
|
||||||
|
for(let val of res.permission){
|
||||||
|
if(val.check){
|
||||||
|
permissionIds.push(val.permission_id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.permissions = res.permission
|
||||||
|
this.sectionIndex = this.section.findIndex(val => val.store_id == res.store.store_id)
|
||||||
|
this.permissionIds= permissionIds
|
||||||
|
this.name = res.name
|
||||||
|
this.phone = res.mobile
|
||||||
|
this.job = res.job
|
||||||
|
this.cover = res.cover
|
||||||
|
this.isPosition = res.is_position
|
||||||
|
this.loging = false
|
||||||
|
}).catch(err => {
|
||||||
|
uni.showToast({
|
||||||
|
title: err.message,
|
||||||
|
icon : 'none'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}else{
|
||||||
|
this.permissions = res.permissions
|
||||||
|
this.loging = false
|
||||||
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
methods:{
|
methods:{
|
||||||
|
// 设为推荐
|
||||||
|
switchChange(e){
|
||||||
|
this.isPosition = e.detail.value
|
||||||
|
},
|
||||||
// 选择部门
|
// 选择部门
|
||||||
pickerChange(e){
|
pickerChange(e){
|
||||||
this.sectionIndex = e.detail.value
|
this.sectionIndex = e.detail.value
|
||||||
@@ -112,16 +158,19 @@
|
|||||||
},
|
},
|
||||||
// 添加员工
|
// 添加员工
|
||||||
onAddEmployees(){
|
onAddEmployees(){
|
||||||
addEmployees({
|
let data = {
|
||||||
name : this.name,
|
name : this.name,
|
||||||
mobileNo : this.phone,
|
mobileNo : this.phone,
|
||||||
job : this.job,
|
job : this.job,
|
||||||
cover : this.cover.path,
|
cover : this.cover.path,
|
||||||
position : 1,
|
position : this.isPosition ? 1: 0,
|
||||||
order : 0,
|
order : 0,
|
||||||
store_id : this.section[this.sectionIndex].store_id,
|
store_id : this.section[this.sectionIndex].store_id,
|
||||||
permission : this.permissionIds
|
permission : this.permissionIds
|
||||||
}).then(res => {
|
}
|
||||||
|
let submitAdd = this.type == 'PUT' ? employeesPut(this.$Route.query.id, data) : addEmployees(data)
|
||||||
|
|
||||||
|
submitAdd.then(res => {
|
||||||
uni.showModal({
|
uni.showModal({
|
||||||
title : '提示',
|
title : '提示',
|
||||||
content : res,
|
content : res,
|
||||||
@@ -151,6 +200,9 @@
|
|||||||
}
|
}
|
||||||
.jurisdiction{
|
.jurisdiction{
|
||||||
background: white;
|
background: white;
|
||||||
|
&.switch{
|
||||||
|
margin-top: $margin;
|
||||||
|
}
|
||||||
.item{
|
.item{
|
||||||
position: relative;
|
position: relative;
|
||||||
padding: $padding 150rpx $padding $padding;
|
padding: $padding 150rpx $padding $padding;
|
||||||
@@ -171,7 +223,13 @@
|
|||||||
right: $padding;
|
right: $padding;
|
||||||
top: 50%;
|
top: 50%;
|
||||||
height: 40rpx;
|
height: 40rpx;
|
||||||
width: 40rpx;
|
margin-top: -27rpx;
|
||||||
|
}
|
||||||
|
.item-switch{
|
||||||
|
position: absolute;
|
||||||
|
right: $padding;
|
||||||
|
top: 50%;
|
||||||
|
height: 40rpx;
|
||||||
margin-top: -27rpx;
|
margin-top: -27rpx;
|
||||||
}
|
}
|
||||||
.item-title{
|
.item-title{
|
||||||
|
|||||||
@@ -1,37 +1,65 @@
|
|||||||
<template>
|
<template>
|
||||||
<view class="ios-bottom">
|
<view class="ios-bottom">
|
||||||
<uni-collapse>
|
<uni-collapse v-if="lists.length > 0">
|
||||||
<block v-for="(listItem, listIndex) in lists" :key="listIndex">
|
<block v-for="(listItem, listIndex) in lists" :key="listIndex">
|
||||||
<uni-collapse-item :show-animation="true" :open="listIndex === 0">
|
<uni-collapse-item :show-animation="true" :open="listIndex === 0">
|
||||||
<template v-slot:title>
|
<template v-slot:title>
|
||||||
<view class="collapse-title">{{listItem.name}}</view>
|
<view class="collapse-title">{{listItem.name}}</view>
|
||||||
</template>
|
</template>
|
||||||
<view class="employees-item" v-for="(item, index) in listItem.data" :key="index">
|
<block v-if="listItem.data.length > 0">
|
||||||
<view class="cover">
|
<view class="employees-border" v-for="(item, index) in listItem.data" :key="index">
|
||||||
<block v-if="item.user.avatar === ''">{{item.name.slice(0,1)}}</block>
|
<uni-swipe-action>
|
||||||
<block v-else>
|
<uni-swipe-action-item :rightOptions="options" @click="onEmployees($event, listIndex, index)">
|
||||||
<image class="cover-img" :src="item.user.avatar" mode="aspectFill"></image>
|
<view class="employees-item">
|
||||||
</block>
|
<view class="cover">
|
||||||
|
<block v-if="item.user.avatar === ''">{{item.name.slice(0,1)}}</block>
|
||||||
|
<block v-else>
|
||||||
|
<image class="cover-img" :src="item.user.avatar" mode="aspectFill"></image>
|
||||||
|
</block>
|
||||||
|
</view>
|
||||||
|
<view class="content">
|
||||||
|
<view class="nickname nowrap">{{item.name}}<text>{{item.job}}</text></view>
|
||||||
|
<view class="job nowrap">
|
||||||
|
<text v-for="(permissionItem, permissionIndex) in item.permission" :key="permissionIndex">{{permissionItem}}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</uni-swipe-action-item>
|
||||||
|
</uni-swipe-action>
|
||||||
</view>
|
</view>
|
||||||
<view class="content">
|
</block>
|
||||||
<view class="nickname nowrap">{{item.name}}<text>{{item.job}}</text></view>
|
<block v-else>
|
||||||
<view class="job nowrap">
|
<view class="employees-null">店铺暂无员工</view>
|
||||||
<text v-for="(permissionItem, permissionIndex) in item.permission" :key="permissionIndex">{{permissionItem}}</text>
|
</block>
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
</view>
|
|
||||||
</uni-collapse-item>
|
</uni-collapse-item>
|
||||||
</block>
|
</block>
|
||||||
</uni-collapse>
|
</uni-collapse>
|
||||||
|
<view v-else class="list-null">
|
||||||
|
<image class="icon" src="@/static/icons/listnull-icon.png" mode="widthFix" />
|
||||||
|
<view class="sub-title">暂未添加员工</view>
|
||||||
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { employees } from '@/apis/interfaces/employees'
|
import { employees, employeesDelete } from '@/apis/interfaces/employees'
|
||||||
export default {
|
export default {
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
lists: []
|
lists : [],
|
||||||
|
options : [{
|
||||||
|
text : '编辑',
|
||||||
|
type : 'PUT',
|
||||||
|
style : {
|
||||||
|
backgroundColor: '#3688ee'
|
||||||
|
}
|
||||||
|
},{
|
||||||
|
text : '删除',
|
||||||
|
type : 'DELETE',
|
||||||
|
style : {
|
||||||
|
backgroundColor: '#e93340'
|
||||||
|
}
|
||||||
|
}]
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
onShow(){
|
onShow(){
|
||||||
@@ -39,7 +67,58 @@
|
|||||||
this.lists = res
|
this.lists = res
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
methods:{
|
||||||
|
// 编辑,删除
|
||||||
|
onEmployees(e, upIndex, index){
|
||||||
|
let type = e.content.type,
|
||||||
|
val = this.lists[upIndex].data[index]
|
||||||
|
if(type == 'PUT'){
|
||||||
|
this.$Router.push({name: 'addEmployees', params: {type: 'PUT', id: val.employee_id}})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
uni.showModal({
|
||||||
|
title : '提示',
|
||||||
|
content : '删除后无法恢复,确定删除员工[' + val.name + ']吗?',
|
||||||
|
cancelText : '取消',
|
||||||
|
cancelColor : '#555',
|
||||||
|
confirmText : '确认',
|
||||||
|
confirmColor: '#e93340',
|
||||||
|
success : res => {
|
||||||
|
if(res.confirm) {
|
||||||
|
employeesDelete(val.employee_id).then(res => {
|
||||||
|
uni.showToast({
|
||||||
|
title: res,
|
||||||
|
icon : 'none'
|
||||||
|
})
|
||||||
|
this.lists[upIndex].data.splice(index, 1)
|
||||||
|
}).catch(err => {
|
||||||
|
uni.showToast({
|
||||||
|
title: err.message,
|
||||||
|
icon : 'none'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
onNavigationBarButtonTap() {
|
onNavigationBarButtonTap() {
|
||||||
|
if(this.lists.length <= 0){
|
||||||
|
uni.showModal({
|
||||||
|
title : '提示',
|
||||||
|
content : '暂未创建门店,无法添加员工',
|
||||||
|
cancelText : '稍后创建',
|
||||||
|
cancelColor : '#555',
|
||||||
|
confirmText : '立即创建',
|
||||||
|
confirmColor: '#e93340',
|
||||||
|
success : res => {
|
||||||
|
if(res.confirm) {
|
||||||
|
this.$Router.push({name: 'shopCreate'})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
this.$Router.push({name: 'addEmployees'})
|
this.$Router.push({name: 'addEmployees'})
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -65,21 +144,19 @@
|
|||||||
height: 1rpx;
|
height: 1rpx;
|
||||||
background: $border-color;
|
background: $border-color;
|
||||||
}
|
}
|
||||||
&:last-child::before {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
.cover {
|
.cover {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: $padding - 10;
|
top: $padding - 10;
|
||||||
left: $padding;
|
left: $padding;
|
||||||
background: $text-price;
|
background: $text-price;
|
||||||
color: white;
|
color: white;
|
||||||
height: 78rpx;
|
height: 88rpx;
|
||||||
width: 78rpx;
|
width: 88rpx;
|
||||||
line-height: 78rpx;
|
line-height: 88rpx;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
font-size: $title-size-lg;
|
||||||
.cover-img {
|
.cover-img {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@@ -89,11 +166,12 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
.content {
|
.content {
|
||||||
height: 78rpx;
|
height: 88rpx;
|
||||||
padding-left: 98rpx;
|
padding-left: 108rpx;
|
||||||
.nickname{
|
.nickname{
|
||||||
line-height: 40rpx;
|
line-height: 40rpx;
|
||||||
font-size: $title-size;
|
font-size: $title-size;
|
||||||
|
padding-bottom: 10rpx;
|
||||||
text{
|
text{
|
||||||
font-size: $title-size-sm;
|
font-size: $title-size-sm;
|
||||||
color: $text-gray-m;
|
color: $text-gray-m;
|
||||||
@@ -116,4 +194,28 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// 空提示
|
||||||
|
.list-null{
|
||||||
|
width: 100vw;
|
||||||
|
height: 100vh;
|
||||||
|
box-sizing: border-box;
|
||||||
|
text-align: center;
|
||||||
|
background: white;
|
||||||
|
padding-bottom: 20vh;
|
||||||
|
@extend .vertical;
|
||||||
|
.sub-title{
|
||||||
|
color: $text-gray;
|
||||||
|
font-size: $title-size-m;
|
||||||
|
}
|
||||||
|
.icon{
|
||||||
|
width: 288rpx;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.employees-null{
|
||||||
|
text-align: center;
|
||||||
|
line-height: 10vh;
|
||||||
|
padding-bottom: $padding;
|
||||||
|
font-size: $title-size-m;
|
||||||
|
color: $text-gray;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
24
uni_modules/uni-swipe-action/changelog.md
Normal file
24
uni_modules/uni-swipe-action/changelog.md
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
## 1.2.4(2021-08-20)
|
||||||
|
- 优化 close-all 方法
|
||||||
|
## 1.2.3(2021-08-20)
|
||||||
|
- 新增 close-all 方法,关闭所有已打开的组件
|
||||||
|
## 1.2.2(2021-08-17)
|
||||||
|
- 新增 resize() 方法,在非微信小程序、h5、app-vue端出现不能滑动的问题的时候,重置组件
|
||||||
|
- 修复 app 端偶尔出现类似 Page[x][-x,xx;-x,xx,x,x-x] 的问题
|
||||||
|
- 优化 微信小程序、h5、app-vue 滑动逻辑,避免出现动态新增组件后不能滑动的问题
|
||||||
|
## 1.2.1(2021-07-30)
|
||||||
|
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834)
|
||||||
|
- 修复 跨页面修改组件数据 ,导致不能滑动的问题
|
||||||
|
## 1.1.10(2021-06-17)
|
||||||
|
- 修复 按钮点击执行两次的bug
|
||||||
|
## 1.1.9(2021-05-12)
|
||||||
|
- 新增 项目示例地址
|
||||||
|
## 1.1.8(2021-03-26)
|
||||||
|
- 修复 微信小程序 nv_navigator is not defined 报错的bug
|
||||||
|
## 1.1.7(2021-02-05)
|
||||||
|
- 调整为uni_modules目录规范
|
||||||
|
- 新增 左侧滑动
|
||||||
|
- 新增 插槽使用方式
|
||||||
|
- 新增 threshold 属性,可以控制滑动缺省值
|
||||||
|
- 优化 长列表滚动性能
|
||||||
|
- 修复 滚动页面时触发组件滑动的Bug
|
||||||
@@ -0,0 +1,300 @@
|
|||||||
|
// #ifdef APP-NVUE
|
||||||
|
const BindingX = uni.requireNativePlugin('bindingx');
|
||||||
|
const dom = uni.requireNativePlugin('dom');
|
||||||
|
const animation = uni.requireNativePlugin('animation');
|
||||||
|
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {}
|
||||||
|
},
|
||||||
|
|
||||||
|
watch: {
|
||||||
|
show(newVal) {
|
||||||
|
if (this.autoClose) return
|
||||||
|
if (this.stop) return
|
||||||
|
this.stop = true
|
||||||
|
if (newVal) {
|
||||||
|
this.open(newVal)
|
||||||
|
} else {
|
||||||
|
this.close()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
leftOptions() {
|
||||||
|
this.getSelectorQuery()
|
||||||
|
this.init()
|
||||||
|
},
|
||||||
|
rightOptions(newVal) {
|
||||||
|
this.init()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
this.swipeaction = this.getSwipeAction()
|
||||||
|
if (this.swipeaction.children !== undefined) {
|
||||||
|
this.swipeaction.children.push(this)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.box = this.getEl(this.$refs['selector-box--hock'])
|
||||||
|
this.selector = this.getEl(this.$refs['selector-content--hock']);
|
||||||
|
this.leftButton = this.getEl(this.$refs['selector-left-button--hock']);
|
||||||
|
this.rightButton = this.getEl(this.$refs['selector-right-button--hock']);
|
||||||
|
this.init()
|
||||||
|
},
|
||||||
|
// beforeDestroy() {
|
||||||
|
// this.swipeaction.children.forEach((item, index) => {
|
||||||
|
// if (item === this) {
|
||||||
|
// this.swipeaction.children.splice(index, 1)
|
||||||
|
// }
|
||||||
|
// })
|
||||||
|
// },
|
||||||
|
methods: {
|
||||||
|
init() {
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.x = 0
|
||||||
|
this.button = {
|
||||||
|
show: false
|
||||||
|
}
|
||||||
|
setTimeout(() => {
|
||||||
|
this.getSelectorQuery()
|
||||||
|
}, 200)
|
||||||
|
})
|
||||||
|
},
|
||||||
|
onClick(index, item, position) {
|
||||||
|
this.$emit('click', {
|
||||||
|
content: item,
|
||||||
|
index,
|
||||||
|
position
|
||||||
|
})
|
||||||
|
},
|
||||||
|
touchstart(e) {
|
||||||
|
// 每次只触发一次,避免多次监听造成闪烁
|
||||||
|
if (this.stop) return
|
||||||
|
this.stop = true
|
||||||
|
if (this.autoClose) {
|
||||||
|
this.swipeaction.closeOther(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
const leftWidth = this.button.left.width
|
||||||
|
const rightWidth = this.button.right.width
|
||||||
|
let expression = this.range(this.x, -rightWidth, leftWidth)
|
||||||
|
let leftExpression = this.range(this.x - leftWidth, -leftWidth, 0)
|
||||||
|
let rightExpression = this.range(this.x + rightWidth, 0, rightWidth)
|
||||||
|
|
||||||
|
this.eventpan = BindingX.bind({
|
||||||
|
anchor: this.box,
|
||||||
|
eventType: 'pan',
|
||||||
|
props: [{
|
||||||
|
element: this.selector,
|
||||||
|
property: 'transform.translateX',
|
||||||
|
expression
|
||||||
|
}, {
|
||||||
|
element: this.leftButton,
|
||||||
|
property: 'transform.translateX',
|
||||||
|
expression: leftExpression
|
||||||
|
}, {
|
||||||
|
element: this.rightButton,
|
||||||
|
property: 'transform.translateX',
|
||||||
|
expression: rightExpression
|
||||||
|
}, ]
|
||||||
|
}, (e) => {
|
||||||
|
// nope
|
||||||
|
if (e.state === 'end') {
|
||||||
|
this.x = e.deltaX + this.x;
|
||||||
|
this.isclick = true
|
||||||
|
this.bindTiming(e.deltaX)
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
touchend(e) {
|
||||||
|
if (this.isopen !== 'none' && !this.isclick) {
|
||||||
|
this.open('none')
|
||||||
|
}
|
||||||
|
},
|
||||||
|
bindTiming(x) {
|
||||||
|
const left = this.x
|
||||||
|
const leftWidth = this.button.left.width
|
||||||
|
const rightWidth = this.button.right.width
|
||||||
|
const threshold = this.threshold
|
||||||
|
if (!this.isopen || this.isopen === 'none') {
|
||||||
|
if (left > threshold) {
|
||||||
|
this.open('left')
|
||||||
|
} else if (left < -threshold) {
|
||||||
|
this.open('right')
|
||||||
|
} else {
|
||||||
|
this.open('none')
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if ((x > -leftWidth && x < 0) || x > rightWidth) {
|
||||||
|
if ((x > -threshold && x < 0) || (x - rightWidth > threshold)) {
|
||||||
|
this.open('left')
|
||||||
|
} else {
|
||||||
|
this.open('none')
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if ((x < threshold && x > 0) || (x + leftWidth < -threshold)) {
|
||||||
|
this.open('right')
|
||||||
|
} else {
|
||||||
|
this.open('none')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 移动范围
|
||||||
|
* @param {Object} num
|
||||||
|
* @param {Object} mix
|
||||||
|
* @param {Object} max
|
||||||
|
*/
|
||||||
|
range(num, mix, max) {
|
||||||
|
return `min(max(x+${num}, ${mix}), ${max})`
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开启swipe
|
||||||
|
*/
|
||||||
|
open(type) {
|
||||||
|
this.animation(type)
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 关闭swipe
|
||||||
|
*/
|
||||||
|
close() {
|
||||||
|
this.animation('none')
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开启关闭动画
|
||||||
|
* @param {Object} type
|
||||||
|
*/
|
||||||
|
animation(type) {
|
||||||
|
const time = 300
|
||||||
|
const leftWidth = this.button.left.width
|
||||||
|
const rightWidth = this.button.right.width
|
||||||
|
if (this.eventpan && this.eventpan.token) {
|
||||||
|
BindingX.unbind({
|
||||||
|
token: this.eventpan.token,
|
||||||
|
eventType: 'pan'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case 'left':
|
||||||
|
Promise.all([
|
||||||
|
this.move(this.selector, leftWidth),
|
||||||
|
this.move(this.leftButton, 0),
|
||||||
|
this.move(this.rightButton, rightWidth * 2)
|
||||||
|
]).then(() => {
|
||||||
|
this.setEmit(leftWidth, type)
|
||||||
|
})
|
||||||
|
break
|
||||||
|
case 'right':
|
||||||
|
Promise.all([
|
||||||
|
this.move(this.selector, -rightWidth),
|
||||||
|
this.move(this.leftButton, -leftWidth * 2),
|
||||||
|
this.move(this.rightButton, 0)
|
||||||
|
]).then(() => {
|
||||||
|
this.setEmit(-rightWidth, type)
|
||||||
|
})
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
Promise.all([
|
||||||
|
this.move(this.selector, 0),
|
||||||
|
this.move(this.leftButton, -leftWidth),
|
||||||
|
this.move(this.rightButton, rightWidth)
|
||||||
|
]).then(() => {
|
||||||
|
this.setEmit(0, type)
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
},
|
||||||
|
setEmit(x, type) {
|
||||||
|
const leftWidth = this.button.left.width
|
||||||
|
const rightWidth = this.button.right.width
|
||||||
|
this.isopen = this.isopen || 'none'
|
||||||
|
this.stop = false
|
||||||
|
this.isclick = false
|
||||||
|
// 只有状态不一致才会返回结果
|
||||||
|
if (this.isopen !== type && this.x !== x) {
|
||||||
|
if (type === 'left' && leftWidth > 0) {
|
||||||
|
this.$emit('change', 'left')
|
||||||
|
}
|
||||||
|
if (type === 'right' && rightWidth > 0) {
|
||||||
|
this.$emit('change', 'right')
|
||||||
|
}
|
||||||
|
if (type === 'none') {
|
||||||
|
this.$emit('change', 'none')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.x = x
|
||||||
|
this.isopen = type
|
||||||
|
},
|
||||||
|
move(ref, value) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
animation.transition(ref, {
|
||||||
|
styles: {
|
||||||
|
transform: `translateX(${value})`,
|
||||||
|
},
|
||||||
|
duration: 150, //ms
|
||||||
|
timingFunction: 'linear',
|
||||||
|
needLayout: false,
|
||||||
|
delay: 0 //ms
|
||||||
|
}, function(res) {
|
||||||
|
resolve(res)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取ref
|
||||||
|
* @param {Object} el
|
||||||
|
*/
|
||||||
|
getEl(el) {
|
||||||
|
return el.ref
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* 获取节点信息
|
||||||
|
*/
|
||||||
|
getSelectorQuery() {
|
||||||
|
Promise.all([
|
||||||
|
this.getDom('left'),
|
||||||
|
this.getDom('right'),
|
||||||
|
]).then((data) => {
|
||||||
|
let show = 'none'
|
||||||
|
if (this.autoClose) {
|
||||||
|
show = 'none'
|
||||||
|
} else {
|
||||||
|
show = this.show
|
||||||
|
}
|
||||||
|
|
||||||
|
if (show === 'none') {
|
||||||
|
// this.close()
|
||||||
|
} else {
|
||||||
|
this.open(show)
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
},
|
||||||
|
getDom(str) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
dom.getComponentRect(this.$refs[`selector-${str}-button--hock`], (data) => {
|
||||||
|
if (data) {
|
||||||
|
this.button[str] = data.size
|
||||||
|
resolve(data)
|
||||||
|
} else {
|
||||||
|
reject()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifndef APP-NVUE
|
||||||
|
export default {}
|
||||||
|
// #endif
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
export function isPC() {
|
||||||
|
var userAgentInfo = navigator.userAgent;
|
||||||
|
var Agents = ["Android", "iPhone", "SymbianOS", "Windows Phone", "iPad", "iPod"];
|
||||||
|
var flag = true;
|
||||||
|
for (let v = 0; v < Agents.length - 1; v++) {
|
||||||
|
if (userAgentInfo.indexOf(Agents[v]) > 0) {
|
||||||
|
flag = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return flag;
|
||||||
|
}
|
||||||
@@ -0,0 +1,193 @@
|
|||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
x: 0,
|
||||||
|
transition: false,
|
||||||
|
width: 0,
|
||||||
|
viewWidth: 0,
|
||||||
|
swipeShow: 0
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
show(newVal) {
|
||||||
|
if (this.autoClose) return
|
||||||
|
if (newVal && newVal !== 'none' ) {
|
||||||
|
this.transition = true
|
||||||
|
this.open(newVal)
|
||||||
|
} else {
|
||||||
|
this.close()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
this.swipeaction = this.getSwipeAction()
|
||||||
|
if (this.swipeaction.children !== undefined) {
|
||||||
|
this.swipeaction.children.push(this)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.isopen = false
|
||||||
|
setTimeout(() => {
|
||||||
|
this.getQuerySelect()
|
||||||
|
}, 50)
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
appTouchStart(e) {
|
||||||
|
const {
|
||||||
|
clientX
|
||||||
|
} = e.changedTouches[0]
|
||||||
|
this.clientX = clientX
|
||||||
|
this.timestamp = new Date().getTime()
|
||||||
|
},
|
||||||
|
appTouchEnd(e, index, item, position) {
|
||||||
|
const {
|
||||||
|
clientX
|
||||||
|
} = e.changedTouches[0]
|
||||||
|
// fixed by xxxx 模拟点击事件,解决 ios 13 点击区域错位的问题
|
||||||
|
let diff = Math.abs(this.clientX - clientX)
|
||||||
|
let time = (new Date().getTime()) - this.timestamp
|
||||||
|
if (diff < 40 && time < 300) {
|
||||||
|
this.$emit('click', {
|
||||||
|
content: item,
|
||||||
|
index,
|
||||||
|
position
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* 移动触发
|
||||||
|
* @param {Object} e
|
||||||
|
*/
|
||||||
|
onChange(e) {
|
||||||
|
this.moveX = e.detail.x
|
||||||
|
this.isclose = false
|
||||||
|
},
|
||||||
|
touchstart(e) {
|
||||||
|
this.transition = false
|
||||||
|
this.isclose = true
|
||||||
|
this.autoClose && this.swipeaction.closeOther(this)
|
||||||
|
},
|
||||||
|
touchmove(e) {},
|
||||||
|
touchend(e) {
|
||||||
|
// 0的位置什么都不执行
|
||||||
|
if (this.isclose && this.isopen === 'none') return
|
||||||
|
if (this.isclose && this.isopen !== 'none') {
|
||||||
|
this.transition = true
|
||||||
|
this.close()
|
||||||
|
} else {
|
||||||
|
this.move(this.moveX + this.leftWidth)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 移动
|
||||||
|
* @param {Object} moveX
|
||||||
|
*/
|
||||||
|
move(moveX) {
|
||||||
|
// 打开关闭的处理逻辑不太一样
|
||||||
|
this.transition = true
|
||||||
|
// 未打开状态
|
||||||
|
if (!this.isopen || this.isopen === 'none') {
|
||||||
|
if (moveX > this.threshold) {
|
||||||
|
this.open('left')
|
||||||
|
} else if (moveX < -this.threshold) {
|
||||||
|
this.open('right')
|
||||||
|
} else {
|
||||||
|
this.close()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (moveX < 0 && moveX < this.rightWidth) {
|
||||||
|
const rightX = this.rightWidth + moveX
|
||||||
|
if (rightX < this.threshold) {
|
||||||
|
this.open('right')
|
||||||
|
} else {
|
||||||
|
this.close()
|
||||||
|
}
|
||||||
|
} else if (moveX > 0 && moveX < this.leftWidth) {
|
||||||
|
const leftX = this.leftWidth - moveX
|
||||||
|
if (leftX < this.threshold) {
|
||||||
|
this.open('left')
|
||||||
|
} else {
|
||||||
|
this.close()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 打开
|
||||||
|
*/
|
||||||
|
open(type) {
|
||||||
|
this.x = this.moveX
|
||||||
|
this.animation(type)
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 关闭
|
||||||
|
*/
|
||||||
|
close() {
|
||||||
|
this.x = this.moveX
|
||||||
|
// TODO 解决 x 值不更新的问题,所以会多触发一次 nextTick ,待优化
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.x = -this.leftWidth
|
||||||
|
if(this.isopen!=='none'){
|
||||||
|
this.$emit('change', 'none')
|
||||||
|
}
|
||||||
|
this.isopen = 'none'
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 执行结束动画
|
||||||
|
* @param {Object} type
|
||||||
|
*/
|
||||||
|
animation(type) {
|
||||||
|
this.$nextTick(() => {
|
||||||
|
if (type === 'left') {
|
||||||
|
this.x = 0
|
||||||
|
} else {
|
||||||
|
this.x = -this.rightWidth - this.leftWidth
|
||||||
|
}
|
||||||
|
|
||||||
|
if(this.isopen!==type){
|
||||||
|
this.$emit('change', type)
|
||||||
|
}
|
||||||
|
this.isopen = type
|
||||||
|
})
|
||||||
|
|
||||||
|
},
|
||||||
|
getSlide(x) {},
|
||||||
|
getQuerySelect() {
|
||||||
|
const query = uni.createSelectorQuery().in(this);
|
||||||
|
query.selectAll('.movable-view--hock').boundingClientRect(data => {
|
||||||
|
this.leftWidth = data[1].width
|
||||||
|
this.rightWidth = data[2].width
|
||||||
|
this.width = data[0].width
|
||||||
|
this.viewWidth = this.width + this.rightWidth + this.leftWidth
|
||||||
|
if (this.leftWidth === 0) {
|
||||||
|
// TODO 疑似bug ,初始化的时候如果x 是0,会导致移动位置错误,所以让元素超出一点
|
||||||
|
this.x = -0.1
|
||||||
|
} else {
|
||||||
|
this.x = -this.leftWidth
|
||||||
|
}
|
||||||
|
this.moveX = this.x
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.swipeShow = 1
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!this.buttonWidth) {
|
||||||
|
this.disabledView = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.autoClose) return
|
||||||
|
if (this.show !== 'none') {
|
||||||
|
this.transition = true
|
||||||
|
this.open(this.shows)
|
||||||
|
}
|
||||||
|
}).exec();
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,258 @@
|
|||||||
|
// #ifndef APP-PLUS|| MP-WEIXIN || H5
|
||||||
|
|
||||||
|
const MIN_DISTANCE = 10;
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
// TODO 随机生生元素ID,解决百度小程序获取同一个元素位置信息的bug
|
||||||
|
const elClass = `Uni_${Math.ceil(Math.random() * 10e5).toString(36)}`
|
||||||
|
return {
|
||||||
|
uniShow: false,
|
||||||
|
left: 0,
|
||||||
|
buttonShow: 'none',
|
||||||
|
ani: false,
|
||||||
|
moveLeft:'',
|
||||||
|
elClass
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
show(newVal) {
|
||||||
|
if (this.autoClose) return
|
||||||
|
this.openState(newVal)
|
||||||
|
},
|
||||||
|
left(){
|
||||||
|
this.moveLeft = `translateX(${this.left}px)`
|
||||||
|
},
|
||||||
|
buttonShow(newVal){
|
||||||
|
if (this.autoClose) return
|
||||||
|
this.openState(newVal)
|
||||||
|
},
|
||||||
|
leftOptions() {
|
||||||
|
this.init()
|
||||||
|
},
|
||||||
|
rightOptions() {
|
||||||
|
this.init()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.swipeaction = this.getSwipeAction()
|
||||||
|
if (this.swipeaction.children !== undefined) {
|
||||||
|
this.swipeaction.children.push(this)
|
||||||
|
}
|
||||||
|
this.init()
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
init(){
|
||||||
|
clearTimeout(this.timer)
|
||||||
|
this.timer = setTimeout(() => {
|
||||||
|
this.getSelectorQuery()
|
||||||
|
}, 100)
|
||||||
|
// 移动距离
|
||||||
|
this.left = 0
|
||||||
|
this.x = 0
|
||||||
|
},
|
||||||
|
|
||||||
|
closeSwipe(e) {
|
||||||
|
if (!this.autoClose) return
|
||||||
|
this.swipeaction.closeOther(this)
|
||||||
|
},
|
||||||
|
appTouchStart(e) {
|
||||||
|
const {
|
||||||
|
clientX
|
||||||
|
} = e.changedTouches[0]
|
||||||
|
this.clientX = clientX
|
||||||
|
this.timestamp = new Date().getTime()
|
||||||
|
},
|
||||||
|
appTouchEnd(e, index, item, position) {
|
||||||
|
const {
|
||||||
|
clientX
|
||||||
|
} = e.changedTouches[0]
|
||||||
|
// fixed by xxxx 模拟点击事件,解决 ios 13 点击区域错位的问题
|
||||||
|
let diff = Math.abs(this.clientX - clientX)
|
||||||
|
let time = (new Date().getTime()) - this.timestamp
|
||||||
|
if (diff < 40 && time < 300) {
|
||||||
|
this.$emit('click', {
|
||||||
|
content: item,
|
||||||
|
index,
|
||||||
|
position
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
touchstart(e) {
|
||||||
|
if (this.disabled) return
|
||||||
|
this.ani = false
|
||||||
|
this.x = this.left || 0
|
||||||
|
this.stopTouchStart(e)
|
||||||
|
this.autoClose && this.closeSwipe()
|
||||||
|
},
|
||||||
|
touchmove(e) {
|
||||||
|
if (this.disabled) return
|
||||||
|
// 是否可以滑动页面
|
||||||
|
this.stopTouchMove(e);
|
||||||
|
if (this.direction !== 'horizontal') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.move(this.x + this.deltaX)
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
touchend() {
|
||||||
|
if (this.disabled) return
|
||||||
|
this.moveDirection(this.left)
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* 设置移动距离
|
||||||
|
* @param {Object} value
|
||||||
|
*/
|
||||||
|
move(value) {
|
||||||
|
value = value || 0
|
||||||
|
const leftWidth = this.leftWidth
|
||||||
|
const rightWidth = this.rightWidth
|
||||||
|
// 获取可滑动范围
|
||||||
|
this.left = this.range(value, -rightWidth, leftWidth);
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取范围
|
||||||
|
* @param {Object} num
|
||||||
|
* @param {Object} min
|
||||||
|
* @param {Object} max
|
||||||
|
*/
|
||||||
|
range(num, min, max) {
|
||||||
|
return Math.min(Math.max(num, min), max);
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* 移动方向判断
|
||||||
|
* @param {Object} left
|
||||||
|
* @param {Object} value
|
||||||
|
*/
|
||||||
|
moveDirection(left) {
|
||||||
|
const threshold = this.threshold
|
||||||
|
const isopen = this.isopen || 'none'
|
||||||
|
const leftWidth = this.leftWidth
|
||||||
|
const rightWidth = this.rightWidth
|
||||||
|
if (this.deltaX === 0) {
|
||||||
|
this.openState('none')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if ((isopen === 'none' && rightWidth > 0 && -left > threshold) || (isopen !== 'none' && rightWidth > 0 && rightWidth +
|
||||||
|
left < threshold)) {
|
||||||
|
// right
|
||||||
|
this.openState('right')
|
||||||
|
} else if ((isopen === 'none' && leftWidth > 0 && left > threshold) || (isopen !== 'none' && leftWidth > 0 &&
|
||||||
|
leftWidth - left < threshold)) {
|
||||||
|
// left
|
||||||
|
this.openState('left')
|
||||||
|
} else {
|
||||||
|
// default
|
||||||
|
this.openState('none')
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开启状态
|
||||||
|
* @param {Boolean} type
|
||||||
|
*/
|
||||||
|
openState(type) {
|
||||||
|
const leftWidth = this.leftWidth
|
||||||
|
const rightWidth = this.rightWidth
|
||||||
|
let left = ''
|
||||||
|
this.isopen = this.isopen ? this.isopen : 'none'
|
||||||
|
switch (type) {
|
||||||
|
case "left":
|
||||||
|
left = leftWidth
|
||||||
|
break
|
||||||
|
case "right":
|
||||||
|
left = -rightWidth
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
left = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (this.isopen !== type) {
|
||||||
|
this.throttle = true
|
||||||
|
this.$emit('change', type)
|
||||||
|
}
|
||||||
|
|
||||||
|
this.isopen = type
|
||||||
|
// 添加动画类
|
||||||
|
this.ani = true
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.move(left)
|
||||||
|
})
|
||||||
|
// 设置最终移动位置,理论上只要进入到这个函数,肯定是要打开的
|
||||||
|
},
|
||||||
|
close() {
|
||||||
|
this.openState('none')
|
||||||
|
},
|
||||||
|
getDirection(x, y) {
|
||||||
|
if (x > y && x > MIN_DISTANCE) {
|
||||||
|
return 'horizontal';
|
||||||
|
}
|
||||||
|
if (y > x && y > MIN_DISTANCE) {
|
||||||
|
return 'vertical';
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 重置滑动状态
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
resetTouchStatus() {
|
||||||
|
this.direction = '';
|
||||||
|
this.deltaX = 0;
|
||||||
|
this.deltaY = 0;
|
||||||
|
this.offsetX = 0;
|
||||||
|
this.offsetY = 0;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置滑动开始位置
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
stopTouchStart(event) {
|
||||||
|
this.resetTouchStatus();
|
||||||
|
const touch = event.touches[0];
|
||||||
|
this.startX = touch.clientX;
|
||||||
|
this.startY = touch.clientY;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 滑动中,是否禁止打开
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
stopTouchMove(event) {
|
||||||
|
const touch = event.touches[0];
|
||||||
|
this.deltaX = touch.clientX - this.startX;
|
||||||
|
this.deltaY = touch.clientY - this.startY;
|
||||||
|
this.offsetX = Math.abs(this.deltaX);
|
||||||
|
this.offsetY = Math.abs(this.deltaY);
|
||||||
|
this.direction = this.direction || this.getDirection(this.offsetX, this.offsetY);
|
||||||
|
},
|
||||||
|
|
||||||
|
getSelectorQuery() {
|
||||||
|
const views = uni.createSelectorQuery().in(this)
|
||||||
|
views
|
||||||
|
.selectAll('.'+this.elClass)
|
||||||
|
.boundingClientRect(data => {
|
||||||
|
if(data.length === 0) return
|
||||||
|
let show = 'none'
|
||||||
|
if (this.autoClose) {
|
||||||
|
show = 'none'
|
||||||
|
} else {
|
||||||
|
show = this.show
|
||||||
|
}
|
||||||
|
this.leftWidth = data[0].width || 0
|
||||||
|
this.rightWidth = data[1].width || 0
|
||||||
|
this.buttonShow = show
|
||||||
|
})
|
||||||
|
.exec()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifdef APP-PLUS|| MP-WEIXIN || H5
|
||||||
|
export default { }
|
||||||
|
// #endif
|
||||||
@@ -0,0 +1,81 @@
|
|||||||
|
// #ifdef APP-VUE|| MP-WEIXIN || H5
|
||||||
|
import { isPC } from "./isPC"
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
is_show:'none'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
show(newVal){
|
||||||
|
this.is_show = this.show
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
this.swipeaction = this.getSwipeAction()
|
||||||
|
if (this.swipeaction.children !== undefined) {
|
||||||
|
this.swipeaction.children.push(this)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted(){
|
||||||
|
this.is_show = this.show
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
// wxs 中调用
|
||||||
|
closeSwipe(e) {
|
||||||
|
if (!this.autoClose) return
|
||||||
|
this.swipeaction.closeOther(this)
|
||||||
|
},
|
||||||
|
|
||||||
|
change(e) {
|
||||||
|
this.$emit('change', e.open)
|
||||||
|
if (this.is_show !== e.open) {
|
||||||
|
this.is_show = e.open
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
appTouchStart(e) {
|
||||||
|
// #ifdef H5
|
||||||
|
if(isPC()) return
|
||||||
|
// #endif
|
||||||
|
const {
|
||||||
|
clientX
|
||||||
|
} = e.changedTouches[0]
|
||||||
|
this.clientX = clientX
|
||||||
|
this.timestamp = new Date().getTime()
|
||||||
|
},
|
||||||
|
appTouchEnd(e, index, item, position) {
|
||||||
|
// #ifdef H5
|
||||||
|
if(isPC()) return
|
||||||
|
// #endif
|
||||||
|
const {
|
||||||
|
clientX
|
||||||
|
} = e.changedTouches[0]
|
||||||
|
// fixed by xxxx 模拟点击事件,解决 ios 13 点击区域错位的问题
|
||||||
|
let diff = Math.abs(this.clientX - clientX)
|
||||||
|
let time = (new Date().getTime()) - this.timestamp
|
||||||
|
if (diff < 40 && time < 300) {
|
||||||
|
this.$emit('click', {
|
||||||
|
content: item,
|
||||||
|
index,
|
||||||
|
position
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onClickForPC(index, item, position) {
|
||||||
|
// #ifdef H5
|
||||||
|
if(!isPC()) return
|
||||||
|
this.$emit('click', {
|
||||||
|
content: item,
|
||||||
|
index,
|
||||||
|
position
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// #endif
|
||||||
|
// #ifndef APP-VUE|| MP-WEIXIN || H5
|
||||||
|
export default {}
|
||||||
|
// #endif
|
||||||
@@ -0,0 +1,265 @@
|
|||||||
|
const MIN_DISTANCE = 10;
|
||||||
|
export default {
|
||||||
|
showWatch(newVal, oldVal, ownerInstance, instance,self) {
|
||||||
|
let state = self.state
|
||||||
|
this.getDom(instance, ownerInstance,self)
|
||||||
|
if (newVal && newVal !== 'none') {
|
||||||
|
this.openState(newVal, instance, ownerInstance,self)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state.left) {
|
||||||
|
this.openState('none', instance, ownerInstance,self)
|
||||||
|
}
|
||||||
|
this.resetTouchStatus(instance,self)
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开始触摸操作
|
||||||
|
* @param {Object} e
|
||||||
|
* @param {Object} ins
|
||||||
|
*/
|
||||||
|
touchstart(e, ownerInstance, self) {
|
||||||
|
let instance = e.instance;
|
||||||
|
let disabled = instance.getDataset().disabled
|
||||||
|
let state = self.state;
|
||||||
|
this.getDom(instance, ownerInstance, self)
|
||||||
|
// fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
|
||||||
|
disabled = this.getDisabledType(disabled)
|
||||||
|
if (disabled) return
|
||||||
|
// 开始触摸时移除动画类
|
||||||
|
instance.requestAnimationFrame(function() {
|
||||||
|
instance.removeClass('ani');
|
||||||
|
ownerInstance.callMethod('closeSwipe');
|
||||||
|
})
|
||||||
|
|
||||||
|
// 记录上次的位置
|
||||||
|
state.x = state.left || 0
|
||||||
|
// 计算滑动开始位置
|
||||||
|
this.stopTouchStart(e, ownerInstance, self)
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开始滑动操作
|
||||||
|
* @param {Object} e
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
touchmove(e, ownerInstance, self) {
|
||||||
|
let instance = e.instance;
|
||||||
|
let disabled = instance.getDataset().disabled
|
||||||
|
let state = self.state
|
||||||
|
// fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
|
||||||
|
disabled = this.getDisabledType(disabled)
|
||||||
|
if (disabled) return
|
||||||
|
// 是否可以滑动页面
|
||||||
|
this.stopTouchMove(e, self);
|
||||||
|
if (state.direction !== 'horizontal') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (e.preventDefault) {
|
||||||
|
// 阻止页面滚动
|
||||||
|
e.preventDefault()
|
||||||
|
}
|
||||||
|
let x = state.x + state.deltaX
|
||||||
|
this.move(x, instance, ownerInstance, self)
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 结束触摸操作
|
||||||
|
* @param {Object} e
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
touchend(e, ownerInstance, self) {
|
||||||
|
let instance = e.instance;
|
||||||
|
let disabled = instance.getDataset().disabled
|
||||||
|
let state = self.state
|
||||||
|
// fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
|
||||||
|
disabled = this.getDisabledType(disabled)
|
||||||
|
|
||||||
|
if (disabled) return
|
||||||
|
// 滑动过程中触摸结束,通过阙值判断是开启还是关闭
|
||||||
|
// fixed by mehaotian 定时器解决点击按钮,touchend 触发比 click 事件时机早的问题 ,主要是 ios13
|
||||||
|
this.moveDirection(state.left, instance, ownerInstance, self)
|
||||||
|
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置移动距离
|
||||||
|
* @param {Object} value
|
||||||
|
* @param {Object} instance
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
move(value, instance, ownerInstance, self) {
|
||||||
|
value = value || 0
|
||||||
|
let state = self.state
|
||||||
|
let leftWidth = state.leftWidth
|
||||||
|
let rightWidth = state.rightWidth
|
||||||
|
// 获取可滑动范围
|
||||||
|
state.left = this.range(value, -rightWidth, leftWidth);
|
||||||
|
instance.requestAnimationFrame(function() {
|
||||||
|
instance.setStyle({
|
||||||
|
transform: 'translateX(' + state.left + 'px)',
|
||||||
|
'-webkit-transform': 'translateX(' + state.left + 'px)'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取元素信息
|
||||||
|
* @param {Object} instance
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
getDom(instance, ownerInstance, self) {
|
||||||
|
let state = self.state
|
||||||
|
var leftDom = ownerInstance.$el.querySelector('.button-group--left')
|
||||||
|
var rightDom = ownerInstance.$el.querySelector('.button-group--right')
|
||||||
|
|
||||||
|
state.leftWidth = leftDom.offsetWidth || 0
|
||||||
|
state.rightWidth = rightDom.offsetWidth || 0
|
||||||
|
state.threshold = instance.getDataset().threshold
|
||||||
|
},
|
||||||
|
|
||||||
|
getDisabledType(value) {
|
||||||
|
return (typeof(value) === 'string' ? JSON.parse(value) : value) || false;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取范围
|
||||||
|
* @param {Object} num
|
||||||
|
* @param {Object} min
|
||||||
|
* @param {Object} max
|
||||||
|
*/
|
||||||
|
range(num, min, max) {
|
||||||
|
return Math.min(Math.max(num, min), max);
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 移动方向判断
|
||||||
|
* @param {Object} left
|
||||||
|
* @param {Object} value
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
* @param {Object} ins
|
||||||
|
*/
|
||||||
|
moveDirection(left, ins, ownerInstance, self) {
|
||||||
|
var state = self.state
|
||||||
|
var threshold = state.threshold
|
||||||
|
var position = state.position
|
||||||
|
var isopen = state.isopen || 'none'
|
||||||
|
var leftWidth = state.leftWidth
|
||||||
|
var rightWidth = state.rightWidth
|
||||||
|
if (state.deltaX === 0) {
|
||||||
|
this.openState('none', ins, ownerInstance, self)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if ((isopen === 'none' && rightWidth > 0 && -left > threshold) || (isopen !== 'none' && rightWidth > 0 &&
|
||||||
|
rightWidth +
|
||||||
|
left < threshold)) {
|
||||||
|
// right
|
||||||
|
this.openState('right', ins, ownerInstance, self)
|
||||||
|
} else if ((isopen === 'none' && leftWidth > 0 && left > threshold) || (isopen !== 'none' && leftWidth > 0 &&
|
||||||
|
leftWidth - left < threshold)) {
|
||||||
|
// left
|
||||||
|
this.openState('left', ins, ownerInstance, self)
|
||||||
|
} else {
|
||||||
|
// default
|
||||||
|
this.openState('none', ins, ownerInstance, self)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开启状态
|
||||||
|
* @param {Boolean} type
|
||||||
|
* @param {Object} ins
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
openState(type, ins, ownerInstance, self) {
|
||||||
|
let state = self.state
|
||||||
|
let leftWidth = state.leftWidth
|
||||||
|
let rightWidth = state.rightWidth
|
||||||
|
let left = ''
|
||||||
|
state.isopen = state.isopen ? state.isopen : 'none'
|
||||||
|
switch (type) {
|
||||||
|
case "left":
|
||||||
|
left = leftWidth
|
||||||
|
break
|
||||||
|
case "right":
|
||||||
|
left = -rightWidth
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
left = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// && !state.throttle
|
||||||
|
|
||||||
|
if (state.isopen !== type) {
|
||||||
|
state.throttle = true
|
||||||
|
ownerInstance.callMethod('change', {
|
||||||
|
open: type
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
state.isopen = type
|
||||||
|
// 添加动画类
|
||||||
|
ins.requestAnimationFrame(()=> {
|
||||||
|
ins.addClass('ani');
|
||||||
|
this.move(left, ins, ownerInstance, self)
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
getDirection(x, y) {
|
||||||
|
if (x > y && x > MIN_DISTANCE) {
|
||||||
|
return 'horizontal';
|
||||||
|
}
|
||||||
|
if (y > x && y > MIN_DISTANCE) {
|
||||||
|
return 'vertical';
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 重置滑动状态
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
resetTouchStatus(instance, self) {
|
||||||
|
let state = self.state;
|
||||||
|
state.direction = '';
|
||||||
|
state.deltaX = 0;
|
||||||
|
state.deltaY = 0;
|
||||||
|
state.offsetX = 0;
|
||||||
|
state.offsetY = 0;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置滑动开始位置
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
stopTouchStart(event, ownerInstance, self) {
|
||||||
|
let instance = event.instance;
|
||||||
|
let state = self.state
|
||||||
|
this.resetTouchStatus(instance, self);
|
||||||
|
var touch = event.touches[0];
|
||||||
|
state.startX = touch.clientX;
|
||||||
|
state.startY = touch.clientY;
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 滑动中,是否禁止打开
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
stopTouchMove(event, self) {
|
||||||
|
let instance = event.instance;
|
||||||
|
let state = self.state;
|
||||||
|
let touch = event.touches[0];
|
||||||
|
|
||||||
|
state.deltaX = touch.clientX - state.startX;
|
||||||
|
state.deltaY = touch.clientY - state.startY;
|
||||||
|
state.offsetY = Math.abs(state.deltaY);
|
||||||
|
state.offsetX = Math.abs(state.deltaX);
|
||||||
|
state.direction = state.direction || this.getDirection(state.offsetX, state.offsetY);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,348 @@
|
|||||||
|
<template>
|
||||||
|
<!-- 在微信小程序 app vue端 h5 使用wxs 实现-->
|
||||||
|
<!-- #ifdef APP-VUE || MP-WEIXIN || H5 -->
|
||||||
|
<view class="uni-swipe">
|
||||||
|
<!-- #ifdef MP-WEIXIN || VUE3 -->
|
||||||
|
<view class="uni-swipe_box" :change:prop="wxsswipe.showWatch"
|
||||||
|
:prop="is_show" :data-threshold="threshold" :data-disabled="disabled" @touchstart="wxsswipe.touchstart" @touchmove="wxsswipe.touchmove" @touchend="wxsswipe.touchend">
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifndef MP-WEIXIN || VUE3 -->
|
||||||
|
<view class="uni-swipe_box" :change:prop="renderswipe.showWatch"
|
||||||
|
:prop="is_show" :data-threshold="threshold" :data-disabled="disabled+''" @touchstart="renderswipe.touchstart" @touchmove="renderswipe.touchmove" @touchend="renderswipe.touchend">
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- 在微信小程序 app vue端 h5 使用wxs 实现-->
|
||||||
|
<view class="uni-swipe_button-group button-group--left">
|
||||||
|
<slot name="left">
|
||||||
|
<view v-for="(item,index) in leftOptions" :key="index" :style="{
|
||||||
|
backgroundColor: item.style && item.style.backgroundColor ? item.style.backgroundColor : '#C7C6CD',
|
||||||
|
fontSize: item.style && item.style.fontSize ? item.style.fontSize : '16px'
|
||||||
|
}" class="uni-swipe_button button-hock" @touchstart="appTouchStart"
|
||||||
|
@touchend="appTouchEnd($event,index,item,'left')" @click.stop="onClickForPC(index,item,'left')">
|
||||||
|
<text class="uni-swipe_button-text"
|
||||||
|
:style="{color: item.style && item.style.color ? item.style.color : '#FFFFFF',}">{{ item.text }}</text>
|
||||||
|
</view>
|
||||||
|
</slot>
|
||||||
|
</view>
|
||||||
|
<view class="uni-swipe_text--center">
|
||||||
|
<slot></slot>
|
||||||
|
</view>
|
||||||
|
<view class="uni-swipe_button-group button-group--right">
|
||||||
|
<slot name="right">
|
||||||
|
<view v-for="(item,index) in rightOptions" :key="index" :style="{
|
||||||
|
backgroundColor: item.style && item.style.backgroundColor ? item.style.backgroundColor : '#C7C6CD',
|
||||||
|
fontSize: item.style && item.style.fontSize ? item.style.fontSize : '16px'
|
||||||
|
}" class="uni-swipe_button button-hock" @touchstart="appTouchStart"
|
||||||
|
@touchend="appTouchEnd($event,index,item,'right')"
|
||||||
|
@click.stop="onClickForPC(index,item,'right')"><text class="uni-swipe_button-text"
|
||||||
|
:style="{color: item.style && item.style.color ? item.style.color : '#FFFFFF',}">{{ item.text }}</text>
|
||||||
|
</view>
|
||||||
|
</slot>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- app nvue端 使用 bindingx -->
|
||||||
|
<!-- #ifdef APP-NVUE -->
|
||||||
|
<view ref="selector-box--hock" class="uni-swipe" @horizontalpan="touchstart" @touchend="touchend">
|
||||||
|
<view ref='selector-left-button--hock' class="uni-swipe_button-group button-group--left">
|
||||||
|
<slot name="left">
|
||||||
|
<view v-for="(item,index) in leftOptions" :data-button="btn" :key="index" :style="{
|
||||||
|
backgroundColor: item.style && item.style.backgroundColor ? item.style.backgroundColor : '#C7C6CD',
|
||||||
|
fontSize: item.style && item.style.fontSize ? item.style.fontSize : '16px'
|
||||||
|
}" class="uni-swipe_button button-hock" @click.stop="onClick(index,item,'left')"><text
|
||||||
|
class="uni-swipe_button-text"
|
||||||
|
:style="{color: item.style && item.style.color ? item.style.color : '#FFFFFF',}">{{ item.text }}</text>
|
||||||
|
</view>
|
||||||
|
</slot>
|
||||||
|
</view>
|
||||||
|
<view ref='selector-right-button--hock' class="uni-swipe_button-group button-group--right">
|
||||||
|
<slot name="right">
|
||||||
|
<view v-for="(item,index) in rightOptions" :data-button="btn" :key="index" :style="{
|
||||||
|
backgroundColor: item.style && item.style.backgroundColor ? item.style.backgroundColor : '#C7C6CD',
|
||||||
|
fontSize: item.style && item.style.fontSize ? item.style.fontSize : '16px'
|
||||||
|
}" class="uni-swipe_button button-hock" @click.stop="onClick(index,item,'right')"><text
|
||||||
|
class="uni-swipe_button-text"
|
||||||
|
:style="{color: item.style && item.style.color ? item.style.color : '#FFFFFF',}">{{ item.text }}</text>
|
||||||
|
</view>
|
||||||
|
</slot>
|
||||||
|
</view>
|
||||||
|
<view ref='selector-content--hock' class="uni-swipe_box">
|
||||||
|
<slot></slot>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- 其他平台使用 js ,长列表性能可能会有影响-->
|
||||||
|
<!-- #ifdef MP-ALIPAY || MP-BAIDU || MP-TOUTIAO || MP-QQ -->
|
||||||
|
<view class="uni-swipe">
|
||||||
|
<view class="uni-swipe_box" @touchstart="touchstart" @touchmove="touchmove" @touchend="touchend"
|
||||||
|
:style="{transform:moveLeft}" :class="{ani:ani}">
|
||||||
|
<view class="uni-swipe_button-group button-group--left" :class="[elClass]">
|
||||||
|
<slot name="left">
|
||||||
|
<view v-for="(item,index) in leftOptions" :data-button="btn" :key="index" :style="{
|
||||||
|
backgroundColor: item.style && item.style.backgroundColor ? item.style.backgroundColor : '#C7C6CD',
|
||||||
|
fontSize: item.style && item.style.fontSize ? item.style.fontSize : '16px'
|
||||||
|
}" class="uni-swipe_button button-hock" @touchstart="appTouchStart"
|
||||||
|
@touchend="appTouchEnd($event,index,item,'left')"><text class="uni-swipe_button-text"
|
||||||
|
:style="{color: item.style && item.style.color ? item.style.color : '#FFFFFF',}">{{ item.text }}</text>
|
||||||
|
</view>
|
||||||
|
</slot>
|
||||||
|
</view>
|
||||||
|
<slot></slot>
|
||||||
|
<view class="uni-swipe_button-group button-group--right" :class="[elClass]">
|
||||||
|
<slot name="right">
|
||||||
|
<view v-for="(item,index) in rightOptions" :data-button="btn" :key="index" :style="{
|
||||||
|
backgroundColor: item.style && item.style.backgroundColor ? item.style.backgroundColor : '#C7C6CD',
|
||||||
|
fontSize: item.style && item.style.fontSize ? item.style.fontSize : '16px'
|
||||||
|
}" @touchstart="appTouchStart" @touchend="appTouchEnd($event,index,item,'right')"
|
||||||
|
class="uni-swipe_button button-hock"><text class="uni-swipe_button-text"
|
||||||
|
:style="{color: item.style && item.style.color ? item.style.color : '#FFFFFF',}">{{ item.text }}</text>
|
||||||
|
</view>
|
||||||
|
</slot>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- #endif -->
|
||||||
|
|
||||||
|
</template>
|
||||||
|
<script src="./wx.wxs" module="wxsswipe" lang="wxs"></script>
|
||||||
|
|
||||||
|
<script module="renderswipe" lang="renderjs">
|
||||||
|
import render from './render.js'
|
||||||
|
export default {
|
||||||
|
mounted(e,ins,owner) {
|
||||||
|
this.state = {}
|
||||||
|
},
|
||||||
|
methods:{
|
||||||
|
showWatch(newVal, oldVal, ownerInstance, instance){
|
||||||
|
render.showWatch(newVal, oldVal, ownerInstance, instance,this)
|
||||||
|
},
|
||||||
|
touchstart(e,ownerInstance){
|
||||||
|
render.touchstart(e,ownerInstance,this)
|
||||||
|
},
|
||||||
|
touchmove(e, ownerInstance){
|
||||||
|
render.touchmove(e,ownerInstance,this)
|
||||||
|
},
|
||||||
|
touchend(e,ownerInstance){
|
||||||
|
render.touchend(e,ownerInstance,this)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<script>
|
||||||
|
import mpwxs from './mpwxs'
|
||||||
|
import bindingx from './bindingx.js'
|
||||||
|
import mpother from './mpother'
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SwipeActionItem 滑动操作子组件
|
||||||
|
* @description 通过滑动触发选项的容器
|
||||||
|
* @tutorial https://ext.dcloud.net.cn/plugin?id=181
|
||||||
|
* @property {Boolean} show = [left|right|none] 开启关闭组件,auto-close = false 时生效
|
||||||
|
* @property {Boolean} disabled = [true|false] 是否禁止滑动
|
||||||
|
* @property {Boolean} autoClose = [true|false] 滑动打开当前组件,是否关闭其他组件
|
||||||
|
* @property {Number} threshold 滑动缺省值
|
||||||
|
* @property {Array} leftOptions 左侧选项内容及样式
|
||||||
|
* @property {Array} rgihtOptions 右侧选项内容及样式
|
||||||
|
* @event {Function} click 点击选项按钮时触发事件,e = {content,index} ,content(点击内容)、index(下标)
|
||||||
|
* @event {Function} change 组件打开或关闭时触发,left\right\none
|
||||||
|
*/
|
||||||
|
|
||||||
|
export default {
|
||||||
|
mixins: [mpwxs,bindingx,mpother],
|
||||||
|
emits:['click','change'],
|
||||||
|
props: {
|
||||||
|
// 控制开关
|
||||||
|
show: {
|
||||||
|
type: String,
|
||||||
|
default: 'none'
|
||||||
|
},
|
||||||
|
|
||||||
|
// 禁用
|
||||||
|
disabled: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false
|
||||||
|
},
|
||||||
|
|
||||||
|
// 是否自动关闭
|
||||||
|
autoClose: {
|
||||||
|
type: Boolean,
|
||||||
|
default: true
|
||||||
|
},
|
||||||
|
|
||||||
|
// 滑动缺省距离
|
||||||
|
threshold: {
|
||||||
|
type: Number,
|
||||||
|
default: 20
|
||||||
|
},
|
||||||
|
|
||||||
|
// 左侧按钮内容
|
||||||
|
leftOptions: {
|
||||||
|
type: Array,
|
||||||
|
default () {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 右侧按钮内容
|
||||||
|
rightOptions: {
|
||||||
|
type: Array,
|
||||||
|
default () {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
},
|
||||||
|
// #ifndef VUE3
|
||||||
|
// TODO vue2
|
||||||
|
destroyed() {
|
||||||
|
if (this.__isUnmounted) return
|
||||||
|
this.uninstall()
|
||||||
|
},
|
||||||
|
// #endif
|
||||||
|
// #ifdef VUE3
|
||||||
|
// TODO vue3
|
||||||
|
unmounted() {
|
||||||
|
this.__isUnmounted = true
|
||||||
|
this.uninstall()
|
||||||
|
},
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
uninstall() {
|
||||||
|
if (this.swipeaction) {
|
||||||
|
this.swipeaction.children.forEach((item, index) => {
|
||||||
|
if (item === this) {
|
||||||
|
this.swipeaction.children.splice(index, 1)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* 获取父元素实例
|
||||||
|
*/
|
||||||
|
getSwipeAction(name = 'uniSwipeAction') {
|
||||||
|
let parent = this.$parent;
|
||||||
|
let parentName = parent.$options.name;
|
||||||
|
while (parentName !== name) {
|
||||||
|
parent = parent.$parent;
|
||||||
|
if (!parent) return false;
|
||||||
|
parentName = parent.$options.name;
|
||||||
|
}
|
||||||
|
return parent;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.uni-swipe {
|
||||||
|
position: relative;
|
||||||
|
/* #ifndef APP-NVUE */
|
||||||
|
overflow: hidden;
|
||||||
|
/* #endif */
|
||||||
|
}
|
||||||
|
|
||||||
|
.uni-swipe_box {
|
||||||
|
/* #ifndef APP-NVUE */
|
||||||
|
display: flex;
|
||||||
|
flex-shrink: 0;
|
||||||
|
// touch-action: none;
|
||||||
|
/* #endif */
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.uni-swipe_content {
|
||||||
|
// border: 1px red solid;
|
||||||
|
}
|
||||||
|
|
||||||
|
.uni-swipe_text--center {
|
||||||
|
width: 100%;
|
||||||
|
/* #ifndef APP-NVUE */
|
||||||
|
cursor: grab;
|
||||||
|
/* #endif */
|
||||||
|
}
|
||||||
|
|
||||||
|
.uni-swipe_button-group {
|
||||||
|
/* #ifndef APP-NVUE */
|
||||||
|
box-sizing: border-box;
|
||||||
|
display: flex;
|
||||||
|
/* #endif */
|
||||||
|
flex-direction: row;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
/* #ifdef H5 */
|
||||||
|
cursor: pointer;
|
||||||
|
/* #endif */
|
||||||
|
}
|
||||||
|
|
||||||
|
.button-group--left {
|
||||||
|
left: 0;
|
||||||
|
transform: translateX(-100%)
|
||||||
|
}
|
||||||
|
|
||||||
|
.button-group--right {
|
||||||
|
right: 0;
|
||||||
|
transform: translateX(100%)
|
||||||
|
}
|
||||||
|
|
||||||
|
.uni-swipe_button {
|
||||||
|
/* #ifdef APP-NVUE */
|
||||||
|
flex: 1;
|
||||||
|
/* #endif */
|
||||||
|
/* #ifndef APP-NVUE */
|
||||||
|
display: flex;
|
||||||
|
/* #endif */
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.uni-swipe_button-text {
|
||||||
|
/* #ifndef APP-NVUE */
|
||||||
|
flex-shrink: 0;
|
||||||
|
/* #endif */
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ani {
|
||||||
|
transition-property: transform;
|
||||||
|
transition-duration: 0.3s;
|
||||||
|
transition-timing-function: cubic-bezier(0.165, 0.84, 0.44, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* #ifdef MP-ALIPAY */
|
||||||
|
.movable-area {
|
||||||
|
/* width: 100%; */
|
||||||
|
height: 45px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.movable-view {
|
||||||
|
display: flex;
|
||||||
|
/* justify-content: center; */
|
||||||
|
position: relative;
|
||||||
|
flex: 1;
|
||||||
|
height: 45px;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.movable-view-button {
|
||||||
|
display: flex;
|
||||||
|
flex-shrink: 0;
|
||||||
|
flex-direction: row;
|
||||||
|
height: 100%;
|
||||||
|
background: #C0C0C0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* .transition {
|
||||||
|
transition: all 0.3s;
|
||||||
|
} */
|
||||||
|
|
||||||
|
.movable-view-box {
|
||||||
|
flex-shrink: 0;
|
||||||
|
height: 100%;
|
||||||
|
background-color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* #endif */
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,341 @@
|
|||||||
|
var MIN_DISTANCE = 10;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 判断当前是否为H5、app-vue
|
||||||
|
*/
|
||||||
|
var IS_HTML5 = false
|
||||||
|
if (typeof window === 'object') IS_HTML5 = true
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 监听页面内值的变化,主要用于动态开关swipe-action
|
||||||
|
* @param {Object} newValue
|
||||||
|
* @param {Object} oldValue
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
* @param {Object} instance
|
||||||
|
*/
|
||||||
|
function showWatch(newVal, oldVal, ownerInstance, instance) {
|
||||||
|
var state = instance.getState()
|
||||||
|
getDom(instance, ownerInstance)
|
||||||
|
if (newVal && newVal !== 'none') {
|
||||||
|
openState(newVal, instance, ownerInstance)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state.left) {
|
||||||
|
openState('none', instance, ownerInstance)
|
||||||
|
}
|
||||||
|
resetTouchStatus(instance)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开始触摸操作
|
||||||
|
* @param {Object} e
|
||||||
|
* @param {Object} ins
|
||||||
|
*/
|
||||||
|
function touchstart(e, ownerInstance) {
|
||||||
|
var instance = e.instance;
|
||||||
|
var disabled = instance.getDataset().disabled
|
||||||
|
var state = instance.getState();
|
||||||
|
getDom(instance, ownerInstance)
|
||||||
|
// fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
|
||||||
|
disabled = (typeof(disabled) === 'string' ? JSON.parse(disabled) : disabled) || false;
|
||||||
|
if (disabled) return
|
||||||
|
// 开始触摸时移除动画类
|
||||||
|
instance.requestAnimationFrame(function() {
|
||||||
|
instance.removeClass('ani');
|
||||||
|
ownerInstance.callMethod('closeSwipe');
|
||||||
|
})
|
||||||
|
|
||||||
|
// 记录上次的位置
|
||||||
|
state.x = state.left || 0
|
||||||
|
// 计算滑动开始位置
|
||||||
|
stopTouchStart(e, ownerInstance)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开始滑动操作
|
||||||
|
* @param {Object} e
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
function touchmove(e, ownerInstance) {
|
||||||
|
var instance = e.instance;
|
||||||
|
var disabled = instance.getDataset().disabled
|
||||||
|
var state = instance.getState()
|
||||||
|
// fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
|
||||||
|
disabled = (typeof(disabled) === 'string' ? JSON.parse(disabled) : disabled) || false;
|
||||||
|
if (disabled) return
|
||||||
|
// 是否可以滑动页面
|
||||||
|
stopTouchMove(e);
|
||||||
|
if (state.direction !== 'horizontal') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.preventDefault) {
|
||||||
|
// 阻止页面滚动
|
||||||
|
e.preventDefault()
|
||||||
|
}
|
||||||
|
|
||||||
|
move(state.x + state.deltaX, instance, ownerInstance)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 结束触摸操作
|
||||||
|
* @param {Object} e
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
function touchend(e, ownerInstance) {
|
||||||
|
var instance = e.instance;
|
||||||
|
var disabled = instance.getDataset().disabled
|
||||||
|
var state = instance.getState()
|
||||||
|
// fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
|
||||||
|
disabled = (typeof(disabled) === 'string' ? JSON.parse(disabled) : disabled) || false;
|
||||||
|
|
||||||
|
if (disabled) return
|
||||||
|
// 滑动过程中触摸结束,通过阙值判断是开启还是关闭
|
||||||
|
// fixed by mehaotian 定时器解决点击按钮,touchend 触发比 click 事件时机早的问题 ,主要是 ios13
|
||||||
|
moveDirection(state.left, instance, ownerInstance)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置移动距离
|
||||||
|
* @param {Object} value
|
||||||
|
* @param {Object} instance
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
function move(value, instance, ownerInstance) {
|
||||||
|
value = value || 0
|
||||||
|
var state = instance.getState()
|
||||||
|
var leftWidth = state.leftWidth
|
||||||
|
var rightWidth = state.rightWidth
|
||||||
|
// 获取可滑动范围
|
||||||
|
state.left = range(value, -rightWidth, leftWidth);
|
||||||
|
instance.requestAnimationFrame(function() {
|
||||||
|
instance.setStyle({
|
||||||
|
transform: 'translateX(' + state.left + 'px)',
|
||||||
|
'-webkit-transform': 'translateX(' + state.left + 'px)'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取元素信息
|
||||||
|
* @param {Object} instance
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
function getDom(instance, ownerInstance) {
|
||||||
|
var state = instance.getState()
|
||||||
|
var leftDom = ownerInstance.selectComponent('.button-group--left')
|
||||||
|
var rightDom = ownerInstance.selectComponent('.button-group--right')
|
||||||
|
var leftStyles = {
|
||||||
|
width: 0
|
||||||
|
}
|
||||||
|
var rightStyles = {
|
||||||
|
width: 0
|
||||||
|
}
|
||||||
|
leftStyles = leftDom.getBoundingClientRect()
|
||||||
|
rightStyles = rightDom.getBoundingClientRect()
|
||||||
|
|
||||||
|
state.leftWidth = leftStyles.width || 0
|
||||||
|
state.rightWidth = rightStyles.width || 0
|
||||||
|
state.threshold = instance.getDataset().threshold
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取范围
|
||||||
|
* @param {Object} num
|
||||||
|
* @param {Object} min
|
||||||
|
* @param {Object} max
|
||||||
|
*/
|
||||||
|
function range(num, min, max) {
|
||||||
|
return Math.min(Math.max(num, min), max);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 移动方向判断
|
||||||
|
* @param {Object} left
|
||||||
|
* @param {Object} value
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
* @param {Object} ins
|
||||||
|
*/
|
||||||
|
function moveDirection(left, ins, ownerInstance) {
|
||||||
|
var state = ins.getState()
|
||||||
|
var threshold = state.threshold
|
||||||
|
var position = state.position
|
||||||
|
var isopen = state.isopen || 'none'
|
||||||
|
var leftWidth = state.leftWidth
|
||||||
|
var rightWidth = state.rightWidth
|
||||||
|
if (state.deltaX === 0) {
|
||||||
|
openState('none', ins, ownerInstance)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if ((isopen === 'none' && rightWidth > 0 && -left > threshold) || (isopen !== 'none' && rightWidth > 0 &&
|
||||||
|
rightWidth +
|
||||||
|
left < threshold)) {
|
||||||
|
// right
|
||||||
|
openState('right', ins, ownerInstance)
|
||||||
|
} else if ((isopen === 'none' && leftWidth > 0 && left > threshold) || (isopen !== 'none' && leftWidth > 0 &&
|
||||||
|
leftWidth - left < threshold)) {
|
||||||
|
// left
|
||||||
|
openState('left', ins, ownerInstance)
|
||||||
|
} else {
|
||||||
|
// default
|
||||||
|
openState('none', ins, ownerInstance)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 开启状态
|
||||||
|
* @param {Boolean} type
|
||||||
|
* @param {Object} ins
|
||||||
|
* @param {Object} ownerInstance
|
||||||
|
*/
|
||||||
|
function openState(type, ins, ownerInstance) {
|
||||||
|
var state = ins.getState()
|
||||||
|
var leftWidth = state.leftWidth
|
||||||
|
var rightWidth = state.rightWidth
|
||||||
|
var left = ''
|
||||||
|
state.isopen = state.isopen ? state.isopen : 'none'
|
||||||
|
switch (type) {
|
||||||
|
case "left":
|
||||||
|
left = leftWidth
|
||||||
|
break
|
||||||
|
case "right":
|
||||||
|
left = -rightWidth
|
||||||
|
break
|
||||||
|
default:
|
||||||
|
left = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// && !state.throttle
|
||||||
|
|
||||||
|
if (state.isopen !== type) {
|
||||||
|
state.throttle = true
|
||||||
|
ownerInstance.callMethod('change', {
|
||||||
|
open: type
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
state.isopen = type
|
||||||
|
// 添加动画类
|
||||||
|
ins.requestAnimationFrame(function() {
|
||||||
|
ins.addClass('ani');
|
||||||
|
move(left, ins, ownerInstance)
|
||||||
|
})
|
||||||
|
// 设置最终移动位置,理论上只要进入到这个函数,肯定是要打开的
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function getDirection(x, y) {
|
||||||
|
if (x > y && x > MIN_DISTANCE) {
|
||||||
|
return 'horizontal';
|
||||||
|
}
|
||||||
|
if (y > x && y > MIN_DISTANCE) {
|
||||||
|
return 'vertical';
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 重置滑动状态
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
function resetTouchStatus(instance) {
|
||||||
|
var state = instance.getState();
|
||||||
|
state.direction = '';
|
||||||
|
state.deltaX = 0;
|
||||||
|
state.deltaY = 0;
|
||||||
|
state.offsetX = 0;
|
||||||
|
state.offsetY = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置滑动开始位置
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
function stopTouchStart(event) {
|
||||||
|
var instance = event.instance;
|
||||||
|
var state = instance.getState();
|
||||||
|
resetTouchStatus(instance);
|
||||||
|
var touch = event.touches[0];
|
||||||
|
if (IS_HTML5 && isPC()) {
|
||||||
|
touch = event;
|
||||||
|
}
|
||||||
|
state.startX = touch.clientX;
|
||||||
|
state.startY = touch.clientY;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 滑动中,是否禁止打开
|
||||||
|
* @param {Object} event
|
||||||
|
*/
|
||||||
|
function stopTouchMove(event) {
|
||||||
|
var instance = event.instance;
|
||||||
|
var state = instance.getState();
|
||||||
|
var touch = event.touches[0];
|
||||||
|
if (IS_HTML5 && isPC()) {
|
||||||
|
touch = event;
|
||||||
|
}
|
||||||
|
state.deltaX = touch.clientX - state.startX;
|
||||||
|
state.deltaY = touch.clientY - state.startY;
|
||||||
|
state.offsetY = Math.abs(state.deltaY);
|
||||||
|
state.offsetX = Math.abs(state.deltaX);
|
||||||
|
state.direction = state.direction || getDirection(state.offsetX, state.offsetY);
|
||||||
|
}
|
||||||
|
|
||||||
|
function isPC() {
|
||||||
|
var userAgentInfo = navigator.userAgent;
|
||||||
|
var Agents = ["Android", "iPhone", "SymbianOS", "Windows Phone", "iPad", "iPod"];
|
||||||
|
var flag = true;
|
||||||
|
for (var v = 0; v < Agents.length - 1; v++) {
|
||||||
|
if (userAgentInfo.indexOf(Agents[v]) > 0) {
|
||||||
|
flag = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return flag;
|
||||||
|
}
|
||||||
|
|
||||||
|
var movable = false
|
||||||
|
|
||||||
|
function mousedown(e, ins) {
|
||||||
|
if (!IS_HTML5) return
|
||||||
|
if (!isPC()) return
|
||||||
|
touchstart(e, ins)
|
||||||
|
movable = true
|
||||||
|
}
|
||||||
|
|
||||||
|
function mousemove(e, ins) {
|
||||||
|
if (!IS_HTML5) return
|
||||||
|
if (!isPC()) return
|
||||||
|
if (!movable) return
|
||||||
|
touchmove(e, ins)
|
||||||
|
}
|
||||||
|
|
||||||
|
function mouseup(e, ins) {
|
||||||
|
if (!IS_HTML5) return
|
||||||
|
if (!isPC()) return
|
||||||
|
touchend(e, ins)
|
||||||
|
movable = false
|
||||||
|
}
|
||||||
|
|
||||||
|
function mouseleave(e, ins) {
|
||||||
|
if (!IS_HTML5) return
|
||||||
|
if (!isPC()) return
|
||||||
|
movable = false
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
showWatch: showWatch,
|
||||||
|
touchstart: touchstart,
|
||||||
|
touchmove: touchmove,
|
||||||
|
touchend: touchend,
|
||||||
|
mousedown: mousedown,
|
||||||
|
mousemove: mousemove,
|
||||||
|
mouseup: mouseup,
|
||||||
|
mouseleave: mouseleave
|
||||||
|
}
|
||||||
@@ -0,0 +1,60 @@
|
|||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<slot></slot>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
/**
|
||||||
|
* SwipeAction 滑动操作
|
||||||
|
* @description 通过滑动触发选项的容器
|
||||||
|
* @tutorial https://ext.dcloud.net.cn/plugin?id=181
|
||||||
|
*/
|
||||||
|
export default {
|
||||||
|
name:"uniSwipeAction",
|
||||||
|
data() {
|
||||||
|
return {};
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
this.children = [];
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
// 公开给用户使用,重制组件样式
|
||||||
|
resize(){
|
||||||
|
// wxs 会自己计算组件大小,所以无需执行下面代码
|
||||||
|
// #ifndef APP-VUE || H5 || MP-WEIXIN
|
||||||
|
this.children.forEach(vm=>{
|
||||||
|
vm.init()
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
},
|
||||||
|
// 公开给用户使用,关闭全部 已经打开的组件
|
||||||
|
closeAll(){
|
||||||
|
this.children.forEach(vm=>{
|
||||||
|
// #ifdef APP-VUE || H5 || MP-WEIXIN
|
||||||
|
vm.is_show = 'none'
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifndef APP-VUE || H5 || MP-WEIXIN
|
||||||
|
vm.close()
|
||||||
|
// #endif
|
||||||
|
})
|
||||||
|
},
|
||||||
|
closeOther(vm) {
|
||||||
|
if (this.openItem && this.openItem !== vm) {
|
||||||
|
// #ifdef APP-VUE || H5 || MP-WEIXIN
|
||||||
|
this.openItem.is_show = 'none'
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifndef APP-VUE || H5 || MP-WEIXIN
|
||||||
|
this.openItem.close()
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
// 记录上一个打开的 swipe-action-item ,用于 auto-close
|
||||||
|
this.openItem = vm
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style></style>
|
||||||
87
uni_modules/uni-swipe-action/package.json
Normal file
87
uni_modules/uni-swipe-action/package.json
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
{
|
||||||
|
"id": "uni-swipe-action",
|
||||||
|
"displayName": "uni-swipe-action 滑动操作",
|
||||||
|
"version": "1.2.4",
|
||||||
|
"description": "SwipeAction 滑动操作操作组件",
|
||||||
|
"keywords": [
|
||||||
|
"",
|
||||||
|
"uni-ui",
|
||||||
|
"uniui",
|
||||||
|
"滑动删除",
|
||||||
|
"侧滑删除"
|
||||||
|
],
|
||||||
|
"repository": "https://github.com/dcloudio/uni-ui",
|
||||||
|
"engines": {
|
||||||
|
"HBuilderX": ""
|
||||||
|
},
|
||||||
|
"directories": {
|
||||||
|
"example": "../../temps/example_temps"
|
||||||
|
},
|
||||||
|
"dcloudext": {
|
||||||
|
"category": [
|
||||||
|
"前端组件",
|
||||||
|
"通用组件"
|
||||||
|
],
|
||||||
|
"sale": {
|
||||||
|
"regular": {
|
||||||
|
"price": "0.00"
|
||||||
|
},
|
||||||
|
"sourcecode": {
|
||||||
|
"price": "0.00"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"contact": {
|
||||||
|
"qq": ""
|
||||||
|
},
|
||||||
|
"declaration": {
|
||||||
|
"ads": "无",
|
||||||
|
"data": "无",
|
||||||
|
"permissions": "无"
|
||||||
|
},
|
||||||
|
"npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui"
|
||||||
|
},
|
||||||
|
"uni_modules": {
|
||||||
|
"dependencies": [],
|
||||||
|
"encrypt": [],
|
||||||
|
"platforms": {
|
||||||
|
"cloud": {
|
||||||
|
"tcb": "y",
|
||||||
|
"aliyun": "y"
|
||||||
|
},
|
||||||
|
"client": {
|
||||||
|
"App": {
|
||||||
|
"app-vue": "y",
|
||||||
|
"app-nvue": "y"
|
||||||
|
},
|
||||||
|
"H5-mobile": {
|
||||||
|
"Safari": "y",
|
||||||
|
"Android Browser": "y",
|
||||||
|
"微信浏览器(Android)": "y",
|
||||||
|
"QQ浏览器(Android)": "y"
|
||||||
|
},
|
||||||
|
"H5-pc": {
|
||||||
|
"Chrome": "y",
|
||||||
|
"IE": "y",
|
||||||
|
"Edge": "y",
|
||||||
|
"Firefox": "y",
|
||||||
|
"Safari": "y"
|
||||||
|
},
|
||||||
|
"小程序": {
|
||||||
|
"微信": "y",
|
||||||
|
"阿里": "y",
|
||||||
|
"百度": "y",
|
||||||
|
"字节跳动": "y",
|
||||||
|
"QQ": "y"
|
||||||
|
},
|
||||||
|
"快应用": {
|
||||||
|
"华为": "y",
|
||||||
|
"联盟": "u"
|
||||||
|
},
|
||||||
|
"Vue": {
|
||||||
|
"vue2": "y",
|
||||||
|
"vue3": "u"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
193
uni_modules/uni-swipe-action/readme.md
Normal file
193
uni_modules/uni-swipe-action/readme.md
Normal file
@@ -0,0 +1,193 @@
|
|||||||
|
|
||||||
|
|
||||||
|
## SwipeAction 滑动操作
|
||||||
|
> **组件名:uni-swipe-action**
|
||||||
|
> 代码块: `uSwipeAction`、`uSwipeActionItem`
|
||||||
|
|
||||||
|
|
||||||
|
通过滑动触发选项的容器
|
||||||
|
|
||||||
|
> **注意事项**
|
||||||
|
> 为了避免错误使用,给大家带来不好的开发体验,请在使用组件前仔细阅读下面的注意事项,可以帮你避免一些错误。
|
||||||
|
> - swipeAction的跟手联动是非常考验性能的。为了提高交互体验,本组件在 app 端 vue 页面、h5、微信小程序使用了wxs 技术,nvue 页面使用 bindingx 技术,可以达到流畅的体验。在其他小程序平台由于底层不支持优化技术,只能使用使用普通 js ,此时性能一般。
|
||||||
|
> - `uni-swipe-action` 和 `uni-swipe-action-item` 需要同时使用
|
||||||
|
> - `uni-swipe-action` 不能嵌套在 `swiper` 中使用
|
||||||
|
> - 长列表不建议使用 autoClose属性,会影响组件性能,造成卡顿,原因是打开之后要通知其他已经打开的组件关闭,会导致多个组件重新渲染
|
||||||
|
> - 事件中传入 `$event` 获取额外参数
|
||||||
|
> - 向下兼容,需要将 `options` 属性替换成 `right-options`
|
||||||
|
> - 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839
|
||||||
|
|
||||||
|
|
||||||
|
### 安装方式
|
||||||
|
|
||||||
|
本组件符合[easycom](https://uniapp.dcloud.io/collocation/pages?id=easycom)规范,`HBuilderX 2.5.5`起,只需将本组件导入项目,在页面`template`中即可直接使用,无需在页面中`import`和注册`components`。
|
||||||
|
|
||||||
|
如需通过`npm`方式使用`uni-ui`组件,另见文档:[https://ext.dcloud.net.cn/plugin?id=55](https://ext.dcloud.net.cn/plugin?id=55)
|
||||||
|
|
||||||
|
|
||||||
|
## 基本用法
|
||||||
|
|
||||||
|
在 ``template`` 中的使用
|
||||||
|
|
||||||
|
```html
|
||||||
|
<uni-swipe-action>
|
||||||
|
<!-- 基础用法 -->
|
||||||
|
<uni-swipe-action-item :right-options="options" :left-options="options" @click="onClick" @change="change">
|
||||||
|
<view>SwipeAction 基础使用场景</view>
|
||||||
|
</uni-swipe-action-item>
|
||||||
|
<!-- 使用插槽 (请自行给定插槽内容宽度)-->
|
||||||
|
<uni-swipe-action-item>
|
||||||
|
<template v-slot:left>
|
||||||
|
<view><text>置顶</text></view>
|
||||||
|
</template>
|
||||||
|
<view>
|
||||||
|
<text >使用插槽</text>
|
||||||
|
</view>
|
||||||
|
<template v-slot:right>
|
||||||
|
<view><text>删除</text></view>
|
||||||
|
</template>
|
||||||
|
</uni-swipe-action-item>
|
||||||
|
<!-- 混合用法 -->
|
||||||
|
<uni-swipe-action-item :right-options="options">
|
||||||
|
<template v-slot:left>
|
||||||
|
<view><text>置顶</text></view>
|
||||||
|
</template>
|
||||||
|
<view><text>混合使用</text></view>
|
||||||
|
</uni-swipe-action-item>
|
||||||
|
</uni-swipe-action>
|
||||||
|
|
||||||
|
<!-- 禁止滑动 -->
|
||||||
|
<uni-swipe-action>
|
||||||
|
<uni-swipe-action-item :disabled="true" :right-options="options">
|
||||||
|
<view>SwipeAction 基础使用场景</view>
|
||||||
|
</uni-swipe-action-item>
|
||||||
|
</uni-swipe-action>
|
||||||
|
|
||||||
|
<!-- 按组使用 -->
|
||||||
|
<uni-swipe-action>
|
||||||
|
<uni-swipe-action-item :right-options="options" @click="bindClick" @change="swipeChange($event, index)">
|
||||||
|
<view >item1</view>
|
||||||
|
</uni-swipe-action-item>
|
||||||
|
<uni-swipe-action-item :right-options="options" @click="bindClick" @change="swipeChange($event, index)">
|
||||||
|
<view>item2</view>
|
||||||
|
</uni-swipe-action-item>
|
||||||
|
<uni-swipe-action-item :right-options="options" @click="bindClick" @change="swipeChange($event, index)">
|
||||||
|
<view>item3</view>
|
||||||
|
</uni-swipe-action-item>
|
||||||
|
</uni-swipe-action>
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
export default {
|
||||||
|
data(){
|
||||||
|
return {
|
||||||
|
options:[
|
||||||
|
{
|
||||||
|
text: '取消',
|
||||||
|
style: {
|
||||||
|
backgroundColor: '#007aff'
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
text: '确认',
|
||||||
|
style: {
|
||||||
|
backgroundColor: '#dd524d'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods:{
|
||||||
|
onClick(e){
|
||||||
|
console.log('点击了'+(e.position === 'left' ? '左侧' : '右侧') + e.content.text + '按钮')
|
||||||
|
},
|
||||||
|
swipeChange(e,index){
|
||||||
|
console.log('当前状态:'+ e +',下标:' + index)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
### SwipeAciton Props
|
||||||
|
|
||||||
|
|属性名|类型|可选值|默认值|是否必填|说明|
|
||||||
|
|:-:|:-:|:-:|:-:|:-:|:-:|
|
||||||
|
|show|String|left/right/none|none |否|开启关闭组件,auto-close = false 时生效|
|
||||||
|
|threshold|Number|-|20|否|滑动阙值|
|
||||||
|
|disabled|Boolean|-|false|否|是否禁止滑动|
|
||||||
|
|autoClose|Boolean|-|true|否|其他组件开启的时候,当前组件是否自动关闭,**注意:长列表使用会有性能问题**|
|
||||||
|
|left-options|Array/Object |-|-|否|左侧选项内容及样式|
|
||||||
|
|right-options|Array/Object |-|-|否|右侧选项内容及样式|
|
||||||
|
|
||||||
|
#### LeftOptions & RightOptions Options
|
||||||
|
|
||||||
|
|参数|类型|是否必填 |说明|
|
||||||
|
|:-:|:-:|:-:|:-:|
|
||||||
|
|text|String|是|按钮的文字 |
|
||||||
|
|style|Object|否|按钮样式{backgroundColor,color,fontSize},backgroundColor默认为:#C7C6CD,color默认为:#FFFFFF,fontSize默认为:14px |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### SwipeAction Events
|
||||||
|
|
||||||
|
|事件称名 |说明|返回值|
|
||||||
|
|:-:|:-:|:-:|
|
||||||
|
|@click|点击选项按钮时触发事件|e = {content,index} ,content(点击内容)、index(下标)、position (位置信息) |
|
||||||
|
|@change|组件打开或关闭时触发|left:左侧 ,right:右侧,none:关闭|
|
||||||
|
|
||||||
|
### SwipeAction Methods
|
||||||
|
|
||||||
|
方法通过 ref 调用
|
||||||
|
|
||||||
|
|方法称名 |说明|
|
||||||
|
|:-:|:-:|
|
||||||
|
|resize()|动态添加数据后,如不能正常滑动,需要主动调用此方法,微信小程序、h5、app-vue 不生效|
|
||||||
|
|close-all()|关闭所有已经打开的组件|
|
||||||
|
### SwipeAction Slots
|
||||||
|
|
||||||
|
|名称|说明|
|
||||||
|
|:-:|:-:|
|
||||||
|
|-|默认插槽自定义显示内容|
|
||||||
|
|default|默认内容插槽|
|
||||||
|
|left|左侧滑动内容 ,会覆盖 leftOptions 内容|
|
||||||
|
|right|右侧滑动内容 ,会覆盖 rightOptions 内容|
|
||||||
|
|
||||||
|
> **提示**
|
||||||
|
> - iOS 端由于存在bounce效果,滑动体验略差,建议禁止bounce效果,禁止方式如下:
|
||||||
|
> ```javascript
|
||||||
|
> {
|
||||||
|
> "path": "swipe-action/swipe-action",
|
||||||
|
> "style": {
|
||||||
|
> "navigationBarTitleText": "SwipeAction 滑动操作",
|
||||||
|
> "disableScroll":true,
|
||||||
|
> "app-plus":{
|
||||||
|
> "bounce":"none"
|
||||||
|
> }
|
||||||
|
> }
|
||||||
|
> }
|
||||||
|
> ```
|
||||||
|
|
||||||
|
|
||||||
|
### Q&A
|
||||||
|
1. Q:动态加载数据,组件滑动失效是怎么回事
|
||||||
|
- A:是因为组件会在加载的时候获取相应的节点信息数据 ,获取需要滑动的距离,所以有时候动态加载数据之后,可能是时机的问题,导致节点信息获取失败 ,那么组件就不能正常滑动。
|
||||||
|
- A:如果是在其他页面通过 vuex 或者uni.$emit 等手段来更新其他页面 uni-swipe-action 数据 ,同样会发生不能滑动的现象,原因是页面隐藏后是不能获取到页面信息的,所以回到 uni-swipe-action 页面后,新增的组件节点信息获取肯定是错误的,所以不能滑动。
|
||||||
|
- A:值的高兴的是在 1.2.2 版本中重构了组件滑动逻辑 ,在微信小程序、h5、app-vue 中使用了 wxs 优化滑动性能,并且不需要担心动态新增组件导致组件无法滑动的问题,节点信息在滑动时实时获取。
|
||||||
|
- A:因为其他平台无法使用 wxs ,所以还是会出现无法滑动的问题怎么处理?1.2.2 版本提供了 resize() 方法,无法滑动时调用 resize() 方法重新渲染组件即可,调用方法时要保证节点已经渲染完毕。
|
||||||
|
|
||||||
|
2. Q:运行到 nvue 下没有样式
|
||||||
|
- A:因为 nvue 下样式默认不能使用复杂的css选择器,所以需要在 manifest.json 中配置 "nvueStyleCompiler" 属性
|
||||||
|
```json
|
||||||
|
// manifest.json
|
||||||
|
{
|
||||||
|
"nvueStyleCompiler" : "uni-app",
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 组件示例
|
||||||
|
|
||||||
|
点击查看:[https://hellouniapp.dcloud.net.cn/pages/extUI/swipe-action/swipe-action](https://hellouniapp.dcloud.net.cn/pages/extUI/swipe-action/swipe-action)
|
||||||
@@ -2,7 +2,7 @@
|
|||||||
var isReady=false;var onReadyCallbacks=[];
|
var isReady=false;var onReadyCallbacks=[];
|
||||||
var isServiceReady=false;var onServiceReadyCallbacks=[];
|
var isServiceReady=false;var onServiceReadyCallbacks=[];
|
||||||
var __uniConfig = {"pages":["pages/equity/index","pages/market/index","pages/store/index","pages/property/index","pages/goods/details","pages/login/login","pages/company/registered","pages/company/prompt","pages/vip/index","pages/equity/search","pages/market/logs","pages/order/buy","pages/order/index","pages/order/details","pages/order/sales","pages/goods/lists","pages/company/approve","pages/store/visitors","pages/store/customer","pages/store/basics","pages/employees/list","pages/employees/add","pages/goods/management","pages/goods/add","pages/coupons/index","pages/coupons/management","pages/coupons/add","pages/coupons/selectGoods","pages/coupons/magDetails","pages/verification/index","pages/verification/details","pages/shop/lists","pages/shop/create"],"window":{"navigationBarTextStyle":"black","navigationBarTitleText":"易货","navigationBarBackgroundColor":"#f5f5f5","backgroundColor":"#f5f5f5"},"tabBar":{"color":"#bababa","selectedColor":"#e93340","backgroundColor":"#FFFFFF","borderStyle":"white","list":[{"pagePath":"pages/equity/index","text":"通证权易","iconPath":"static/tabBar/tabBar_icon_00.png","selectedIconPath":"static/tabBar/tabBar_show_00.png"},{"pagePath":"pages/market/index","text":"转让市场","iconPath":"static/tabBar/tabBar_icon_01.png","selectedIconPath":"static/tabBar/tabBar_show_01.png"},{"pagePath":"pages/store/index","text":"企业工具","iconPath":"static/tabBar/tabBar_icon_02.png","selectedIconPath":"static/tabBar/tabBar_show_02.png"},{"pagePath":"pages/property/index","text":"我的资产","iconPath":"static/tabBar/tabBar_icon_03.png","selectedIconPath":"static/tabBar/tabBar_show_03.png"}]},"nvueCompiler":"uni-app","nvueStyleCompiler":"uni-app","renderer":"auto","splashscreen":{"alwaysShowBeforeRender":false,"autoclose":true},"appname":"易品新境","compilerVersion":"3.2.3","entryPagePath":"pages/equity/index","networkTimeout":{"request":60000,"connectSocket":60000,"uploadFile":60000,"downloadFile":60000}};
|
var __uniConfig = {"pages":["pages/equity/index","pages/market/index","pages/store/index","pages/property/index","pages/goods/details","pages/login/login","pages/company/registered","pages/company/prompt","pages/vip/index","pages/equity/search","pages/market/logs","pages/order/buy","pages/order/index","pages/order/details","pages/order/sales","pages/goods/lists","pages/company/approve","pages/store/visitors","pages/store/customer","pages/store/basics","pages/employees/list","pages/employees/add","pages/goods/management","pages/goods/add","pages/coupons/index","pages/coupons/management","pages/coupons/add","pages/coupons/selectGoods","pages/coupons/magDetails","pages/verification/index","pages/verification/details","pages/shop/lists","pages/shop/create"],"window":{"navigationBarTextStyle":"black","navigationBarTitleText":"易货","navigationBarBackgroundColor":"#f5f5f5","backgroundColor":"#f5f5f5"},"tabBar":{"color":"#bababa","selectedColor":"#e93340","backgroundColor":"#FFFFFF","borderStyle":"white","list":[{"pagePath":"pages/equity/index","text":"通证权易","iconPath":"static/tabBar/tabBar_icon_00.png","selectedIconPath":"static/tabBar/tabBar_show_00.png"},{"pagePath":"pages/market/index","text":"转让市场","iconPath":"static/tabBar/tabBar_icon_01.png","selectedIconPath":"static/tabBar/tabBar_show_01.png"},{"pagePath":"pages/store/index","text":"企业工具","iconPath":"static/tabBar/tabBar_icon_02.png","selectedIconPath":"static/tabBar/tabBar_show_02.png"},{"pagePath":"pages/property/index","text":"我的资产","iconPath":"static/tabBar/tabBar_icon_03.png","selectedIconPath":"static/tabBar/tabBar_show_03.png"}]},"nvueCompiler":"uni-app","nvueStyleCompiler":"uni-app","renderer":"auto","splashscreen":{"alwaysShowBeforeRender":false,"autoclose":true},"appname":"易品新境","compilerVersion":"3.2.3","entryPagePath":"pages/equity/index","networkTimeout":{"request":60000,"connectSocket":60000,"uploadFile":60000,"downloadFile":60000}};
|
||||||
var __uniRoutes = [{"path":"/pages/equity/index","meta":{"isQuit":true,"isTabBar":true},"window":{"navigationStyle":"custom","navigationBarTextStyle":"white"}},{"path":"/pages/market/index","meta":{"isQuit":true,"isTabBar":true},"window":{"navigationBarTitleText":"转让市场","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"成交历史","fontSize":"14","width":"80","color":"#555555"}]}}},{"path":"/pages/store/index","meta":{"isQuit":true,"isTabBar":true},"window":{"navigationStyle":"custom","navigationBarTitleText":"企业工具","navigationBarTextStyle":"white","navigationBarBackgroundColor":"#e93340"}},{"path":"/pages/property/index","meta":{"isQuit":true,"isTabBar":true},"window":{"navigationBarTitleText":"","navigationStyle":"custom","navigationBarTextStyle":"white"}},{"path":"/pages/goods/details","meta":{},"window":{"navigationBarTitleText":"","titleNView":{"backgroundColor":"#FFFFFF","type":"transparent","buttons":[{"text":"分享","fontSize":"12","color":"#555555"}]}}},{"path":"/pages/login/login","meta":{},"window":{"navigationBarTitleText":"","navigationBarBackgroundColor":"#FFFFFF","disableScroll":true}},{"path":"/pages/company/registered","meta":{},"window":{"navigationBarTitleText":""}},{"path":"/pages/company/prompt","meta":{},"window":{"navigationBarTitleText":"","navigationBarBackgroundColor":"#FFFFFF","disableScroll":true,"titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"先逛一逛","fontSize":"14","width":"80","color":"#555555"}]}}},{"path":"/pages/vip/index","meta":{},"window":{"navigationBarTitleText":"会员","navigationBarBackgroundColor":"#1f1b1c","navigationBarTextStyle":"white","backgroundColor":"#fefaef"}},{"path":"/pages/equity/search","meta":{},"window":{"navigationBarTitleText":"搜索"}},{"path":"/pages/market/logs","meta":{},"window":{"navigationBarTitleText":"成交历史"}},{"path":"/pages/order/buy","meta":{},"window":{"navigationBarTitleText":"确认订单","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/order/index","meta":{},"window":{"navigationBarTitleText":"订单管理"}},{"path":"/pages/order/details","meta":{},"window":{"navigationBarTitleText":"订单详情"}},{"path":"/pages/order/sales","meta":{},"window":{"navigationBarTitleText":"售后"}},{"path":"/pages/goods/lists","meta":{},"window":{"navigationBarTitleText":"商品列表","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/company/approve","meta":{},"window":{"navigationBarTitleText":""}},{"path":"/pages/store/visitors","meta":{},"window":{"navigationBarTitleText":"访客统计","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/store/customer","meta":{},"window":{"navigationBarTitleText":"成交客户","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/store/basics","meta":{},"window":{"navigationBarTitleText":"基础信息","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"保存","fontSize":"16","width":"80","color":"#e93340"}]}}},{"path":"/pages/employees/list","meta":{},"window":{"navigationBarTitleText":"员工","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"添加","fontSize":"14","width":"80","color":"#e93340"}]}}},{"path":"/pages/employees/add","meta":{},"window":{"navigationBarTitleText":"添加员工","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/goods/management","meta":{},"window":{"navigationBarTitleText":"产品权证","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"发布","fontSize":"16","width":"80","color":"#e93340"}]}}},{"path":"/pages/goods/add","meta":{},"window":{"navigationBarTitleText":"发布权证","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/coupons/index","meta":{},"window":{"navigationBarTitleText":"优惠券"}},{"path":"/pages/coupons/management","meta":{},"window":{"navigationBarTitleText":"优惠券管理","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"发布","fontSize":"14","width":"80","color":"#e93340"}]}}},{"path":"/pages/coupons/add","meta":{},"window":{"navigationBarTitleText":"发布优惠券","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/coupons/selectGoods","meta":{},"window":{"navigationBarTitleText":"选择产品","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/coupons/magDetails","meta":{},"window":{"navigationBarTitleText":"优惠券详情"}},{"path":"/pages/verification/index","meta":{},"window":{"navigationBarTitleText":"扫码验证"}},{"path":"/pages/verification/details","meta":{},"window":{"navigationBarTitleText":"核销券详情"}},{"path":"/pages/shop/lists","meta":{},"window":{"navigationBarTitleText":"部门/门店","navigationBarBackgroundColor":"#FFFFFF","titleNView":{"buttons":[{"text":"创建","fontSize":"14","width":"80","color":"#e93340"}]}}},{"path":"/pages/shop/create","meta":{},"window":{"navigationBarTitleText":"创建门店/部门","navigationBarBackgroundColor":"#FFFFFF"}}];
|
var __uniRoutes = [{"path":"/pages/equity/index","meta":{"isQuit":true,"isTabBar":true},"window":{"navigationStyle":"custom","navigationBarTextStyle":"white"}},{"path":"/pages/market/index","meta":{"isQuit":true,"isTabBar":true},"window":{"navigationBarTitleText":"转让市场","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"成交历史","fontSize":"14","width":"80","color":"#555555"}]}}},{"path":"/pages/store/index","meta":{"isQuit":true,"isTabBar":true},"window":{"navigationStyle":"custom","navigationBarTitleText":"企业工具","navigationBarTextStyle":"white","navigationBarBackgroundColor":"#e93340"}},{"path":"/pages/property/index","meta":{"isQuit":true,"isTabBar":true},"window":{"navigationBarTitleText":"","navigationStyle":"custom","navigationBarTextStyle":"white"}},{"path":"/pages/goods/details","meta":{},"window":{"navigationBarTitleText":"","titleNView":{"backgroundColor":"#FFFFFF","type":"transparent","buttons":[{"text":"分享","fontSize":"12","color":"#555555"}]}}},{"path":"/pages/login/login","meta":{},"window":{"navigationBarTitleText":"","navigationBarBackgroundColor":"#FFFFFF","disableScroll":true}},{"path":"/pages/company/registered","meta":{},"window":{"navigationBarTitleText":""}},{"path":"/pages/company/prompt","meta":{},"window":{"navigationBarTitleText":"","navigationBarBackgroundColor":"#FFFFFF","disableScroll":true,"titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"先逛一逛","fontSize":"14","width":"80","color":"#555555"}]}}},{"path":"/pages/vip/index","meta":{},"window":{"navigationBarTitleText":"会员","navigationBarBackgroundColor":"#1f1b1c","navigationBarTextStyle":"white","backgroundColor":"#fefaef"}},{"path":"/pages/equity/search","meta":{},"window":{"navigationBarTitleText":"搜索"}},{"path":"/pages/market/logs","meta":{},"window":{"navigationBarTitleText":"成交历史"}},{"path":"/pages/order/buy","meta":{},"window":{"navigationBarTitleText":"确认订单","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/order/index","meta":{},"window":{"navigationBarTitleText":"订单管理"}},{"path":"/pages/order/details","meta":{},"window":{"navigationBarTitleText":"订单详情"}},{"path":"/pages/order/sales","meta":{},"window":{"navigationBarTitleText":"售后"}},{"path":"/pages/goods/lists","meta":{},"window":{"navigationBarTitleText":"商品列表","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/company/approve","meta":{},"window":{"navigationBarTitleText":""}},{"path":"/pages/store/visitors","meta":{},"window":{"navigationBarTitleText":"访客统计","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/store/customer","meta":{},"window":{"navigationBarTitleText":"成交客户","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/store/basics","meta":{},"window":{"navigationBarTitleText":"基础信息","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"保存","fontSize":"16","width":"80","color":"#e93340"}]}}},{"path":"/pages/employees/list","meta":{},"window":{"navigationBarTitleText":"员工","bounce":"none","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"添加","fontSize":"14","width":"80","color":"#e93340"}]}}},{"path":"/pages/employees/add","meta":{},"window":{"navigationBarTitleText":"添加员工","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/goods/management","meta":{},"window":{"navigationBarTitleText":"产品权证","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"发布","fontSize":"16","width":"80","color":"#e93340"}]}}},{"path":"/pages/goods/add","meta":{},"window":{"navigationBarTitleText":"发布权证","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/coupons/index","meta":{},"window":{"navigationBarTitleText":"优惠券"}},{"path":"/pages/coupons/management","meta":{},"window":{"navigationBarTitleText":"优惠券管理","titleNView":{"backgroundColor":"#FFFFFF","buttons":[{"text":"发布","fontSize":"14","width":"80","color":"#e93340"}]}}},{"path":"/pages/coupons/add","meta":{},"window":{"navigationBarTitleText":"发布优惠券","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/coupons/selectGoods","meta":{},"window":{"navigationBarTitleText":"选择产品","navigationBarBackgroundColor":"#FFFFFF"}},{"path":"/pages/coupons/magDetails","meta":{},"window":{"navigationBarTitleText":"优惠券详情"}},{"path":"/pages/verification/index","meta":{},"window":{"navigationBarTitleText":"扫码验证"}},{"path":"/pages/verification/details","meta":{},"window":{"navigationBarTitleText":"核销券详情"}},{"path":"/pages/shop/lists","meta":{},"window":{"navigationBarTitleText":"部门/门店","navigationBarBackgroundColor":"#FFFFFF","titleNView":{"buttons":[{"text":"创建","fontSize":"14","width":"80","color":"#e93340"}]}}},{"path":"/pages/shop/create","meta":{},"window":{"navigationBarTitleText":"创建门店/部门","navigationBarBackgroundColor":"#FFFFFF"}}];
|
||||||
__uniConfig.onReady=function(callback){if(__uniConfig.ready){callback()}else{onReadyCallbacks.push(callback)}};Object.defineProperty(__uniConfig,"ready",{get:function(){return isReady},set:function(val){isReady=val;if(!isReady){return}const callbacks=onReadyCallbacks.slice(0);onReadyCallbacks.length=0;callbacks.forEach(function(callback){callback()})}});
|
__uniConfig.onReady=function(callback){if(__uniConfig.ready){callback()}else{onReadyCallbacks.push(callback)}};Object.defineProperty(__uniConfig,"ready",{get:function(){return isReady},set:function(val){isReady=val;if(!isReady){return}const callbacks=onReadyCallbacks.slice(0);onReadyCallbacks.length=0;callbacks.forEach(function(callback){callback()})}});
|
||||||
__uniConfig.onServiceReady=function(callback){if(__uniConfig.serviceReady){callback()}else{onServiceReadyCallbacks.push(callback)}};Object.defineProperty(__uniConfig,"serviceReady",{get:function(){return isServiceReady},set:function(val){isServiceReady=val;if(!isServiceReady){return}const callbacks=onServiceReadyCallbacks.slice(0);onServiceReadyCallbacks.length=0;callbacks.forEach(function(callback){callback()})}});
|
__uniConfig.onServiceReady=function(callback){if(__uniConfig.serviceReady){callback()}else{onServiceReadyCallbacks.push(callback)}};Object.defineProperty(__uniConfig,"serviceReady",{get:function(){return isServiceReady},set:function(val){isServiceReady=val;if(!isServiceReady){return}const callbacks=onServiceReadyCallbacks.slice(0);onServiceReadyCallbacks.length=0;callbacks.forEach(function(callback){callback()})}});
|
||||||
service.register("uni-app-config",{create(a,b,c){if(!__uniConfig.viewport){var d=b.weex.config.env.scale,e=b.weex.config.env.deviceWidth,f=Math.ceil(e/d);Object.assign(__uniConfig,{viewport:f,defaultFontSize:Math.round(f/20)})}return{instance:{__uniConfig:__uniConfig,__uniRoutes:__uniRoutes,global:void 0,window:void 0,document:void 0,frames:void 0,self:void 0,location:void 0,navigator:void 0,localStorage:void 0,history:void 0,Caches:void 0,screen:void 0,alert:void 0,confirm:void 0,prompt:void 0,fetch:void 0,XMLHttpRequest:void 0,WebSocket:void 0,webkit:void 0,print:void 0}}}});
|
service.register("uni-app-config",{create(a,b,c){if(!__uniConfig.viewport){var d=b.weex.config.env.scale,e=b.weex.config.env.deviceWidth,f=Math.ceil(e/d);Object.assign(__uniConfig,{viewport:f,defaultFontSize:Math.round(f/20)})}return{instance:{__uniConfig:__uniConfig,__uniRoutes:__uniRoutes,global:void 0,window:void 0,document:void 0,frames:void 0,self:void 0,location:void 0,navigator:void 0,localStorage:void 0,history:void 0,Caches:void 0,screen:void 0,alert:void 0,confirm:void 0,prompt:void 0,fetch:void 0,XMLHttpRequest:void 0,WebSocket:void 0,webkit:void 0,print:void 0}}}});
|
||||||
|
|||||||
1358
unpackage/dist/dev/app-plus/app-service.js
vendored
1358
unpackage/dist/dev/app-plus/app-service.js
vendored
File diff suppressed because one or more lines are too long
2077
unpackage/dist/dev/app-plus/app-view.js
vendored
2077
unpackage/dist/dev/app-plus/app-view.js
vendored
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user