Browse Source

修复更新内容

master
396316021 6 months ago
parent
commit
70f8c0f286
  1. 2
      App.vue
  2. 70
      config/config.js
  3. 7
      config/host.js
  4. 2
      main.js
  5. 170
      pages/ChatDetail/index.vue
  6. 13
      pages/OrderPage/index.vue
  7. 1
      uni.scss
  8. 21
      uni_modules/m-view/LICENSE
  9. 3
      uni_modules/m-view/changelog.md
  10. 125
      uni_modules/m-view/components/m-avatar/m-avatar.vue
  11. 17
      uni_modules/m-view/components/m-avatar/props.js
  12. 106
      uni_modules/m-view/components/m-button/m-button.vue
  13. 44
      uni_modules/m-view/components/m-checkbox/m-checkbox-group.vue
  14. 79
      uni_modules/m-view/components/m-checkbox/m-checkbox.vue
  15. 120
      uni_modules/m-view/components/m-form-item/m-form-item.vue
  16. 79
      uni_modules/m-view/components/m-form/m-form.vue
  17. 55
      uni_modules/m-view/components/m-header/m-header.vue
  18. 17
      uni_modules/m-view/components/m-header/props.js
  19. 209
      uni_modules/m-view/components/m-input/m-input.vue
  20. 100
      uni_modules/m-view/components/m-private/m-private.vue
  21. 17
      uni_modules/m-view/components/m-private/props.js
  22. 106
      uni_modules/m-view/components/m-radio/m-radio.vue
  23. 41
      uni_modules/m-view/components/m-safe-bottom/m-safe-bottom.vue
  24. 17
      uni_modules/m-view/components/m-safe-bottom/props.js
  25. 41
      uni_modules/m-view/components/m-safe-top/m-safe-top.vue
  26. 17
      uni_modules/m-view/components/m-safe-top/props.js
  27. 79
      uni_modules/m-view/components/m-sticky/m-sticky.vue
  28. 28
      uni_modules/m-view/components/m-sticky/props.js
  29. 71
      uni_modules/m-view/components/m-text/m-text.vue
  30. 26
      uni_modules/m-view/index.js
  31. 1
      uni_modules/m-view/index.scss
  32. 8
      uni_modules/m-view/lib/api.js
  33. 598
      uni_modules/m-view/lib/functions.js
  34. 3
      uni_modules/m-view/lib/qqmap-wx-jssdk.min.js
  35. 74
      uni_modules/m-view/lib/request.js
  36. 215
      uni_modules/m-view/lib/tools.js
  37. 50
      uni_modules/m-view/package.json
  38. 3
      uni_modules/m-view/readme.md
  39. 142
      uni_modules/m-view/theme/common.scss
  40. 6
      uni_modules/m-view/theme/theme.scss
  41. 2
      utils/functions.js

2
App.vue

@ -17,7 +17,7 @@
created() {
if(!uni.getStorageSync('sysInfo')){
uni.showLoading({title:'加载中'});
api.uniSysInfo().then(data => {
uni.$m.uniSysInfo().then(data => {
uni.hideLoading();
}).catch(err => {
uni.hideLoading();

70
config/config.js

@ -4,12 +4,76 @@ import UserCache from "@/config/UserCache";
* 当前环境
* @type {string}
*/
const ENV = 'dev';
export const ENV = 'dev';
/**
* 打开日志打印
* @type {boolean}
*/
const OPEN_LOG = true;
export const OPEN_LOG = true;
export {OPEN_LOG, ENV, UserCache}
export {UserCache}
/**
* 首页路径
* @type {string}
*/
export const INDEX_PATH = '/pages/Index/index';
/**
* 登录页路径
* @type {string}
*/
export const LOGIN_PATH = '/pages/Login/index';
/**
* token字段
* @type {string}
*/
export const TOKEN_FIELD = 'token';
/**
* 用户信息字段
* @type {string}
*/
export const USERINFO_FIELD = '';
/**
* 其他信息字段
* @type {string}
*/
export const OTHER_FIELD = '';
/**
* 页面信息字段
* @type {string}
*/
export const PAGEINFO_FIELD = 'page_info';
/**
* 列表分页数量
* @type {string}
*/
export const LIMIT = '10';
/**
* 默认列表请求数据
* @type {string}
*/
export const DEFAULT_LIST_PAGE = {
page: 1,
limit: LIMIT,
};
/**
* 默认列表请求数据
* @type {string}
*/
export const UPLOAD_PATH = '/common/upload';
/**
* 列表分页数量名称
* @type {string}
*/
export const LIMIT_FIELD = 'limit';

7
config/host.js

@ -1,11 +1,14 @@
// const host='http://192.168.2.92:9509/api'
const host='https://weiyang.test.scyp.ink/api'
// const host='https://weiyang.test.scyp.ink/api'
const host='https://wechat.wwxtx.com/api'
// const host='https://wechat.wwxtx.com/api'
// const host='https://bml.test.scyp.ink/api'
// const host='http://192.168.2.92:9503/api'
// const imghost='https://weiyang.test.scyp.ink/static/image'
const imghost='https://wechat.wwxtx.com/static/image'
const imgHost=imghost
// const imghost='https://wechat.wwxtx.com/static/image'
const title = '汪汪行天下'
const oss = 'https://pet-feed.'+'oss-cn-chengdu.aliyuncs.com'
// const codehost='http://code.baomingle.cn'
export {host,imghost,title,oss}
export {host,imghost,title,oss,imgHost}

2
main.js

@ -13,6 +13,7 @@ app.$mount()
// #ifdef VUE3
import uviewPlus from 'uview-plus'
import mView from '@/uni_modules/m-view'
import { createSSRApp } from 'vue'
import { title } from '@/config/host.js'
import store from "@/store";
@ -21,6 +22,7 @@ export function createApp() {
const app = createSSRApp(App)
app.use(uviewPlus)
app.use(store)
app.use(mView)
uni.$u.config.unit = 'rpx'
return {
app

170
pages/ChatDetail/index.vue

@ -4,22 +4,24 @@
backgroundImage:'url('+staticImage.wallpaperBgImage+')'
}"></view>
<view class="app-wallpaper">
<view class="detail-container">
<view class="detail-container">
<view class="detail-content-container">
<scroll-view class="detail-content" :scroll-y="true" :scroll-top="scrollTop"
style="padding-bottom: 90px;box-sizing: border-box;" :style="{height:windowHeihgt-40+'px'}">
<scroll-view class="detail-content" :scroll-y="true" :scroll-top="scrollTop"
style="padding-bottom: 90px;box-sizing: border-box;" :style="{height:windowHeihgt-40+'px'}"
@scroll="testScroll"
>
<view id="scroll-view-content">
<view :class="item.send_user_id === user_id ? 'chat-item my-chat row' : 'chat-item other-chat row'"
v-for="(item, index) in messageList">
<view class="chat-item-content row">
<view class="chat-avator " v-if="item.send_user_id !== user_id">
<view class="chat-avator " v-if="item.send_user_id !== user_id">
<image mode="aspectFill" class="img" :src="item.avatar ? item.avatar : user.avatar"/>
</view>
<view class="chat-content-container " style="max-width: calc(100% - 50px);">
<view class="chat-time">{{item.user_name}} {{ item.create_time }}</view>
<view class="chat-time">{{ item.user_name }} {{ item.create_time }}</view>
<view class="chat-content" style="word-break: break-all;">{{ item.info }}</view>
</view>
<view class="chat-avator " v-if="item.send_user_id === user_id">
<view class="chat-avator " v-if="item.send_user_id === user_id">
<image mode="aspectFill" class="img" :src="item.avatar ? item.avatar : user.avatar"/>
</view>
</view>
@ -54,9 +56,7 @@
</template>
<script>
import {
imghost
} from '@/config/host'
import {imghost} from '@/config/host'
import api from '../../utils/functions.js';
import HeaderNav from '../../components/HeaderNav/Index.vue';
import MzButton from '../../components/MzButton/Index.vue';
@ -71,7 +71,7 @@ export default {
MzButton
},
data: {
scrollTop:0,
scrollTop: 0,
windowHeihgt: 1000,
staticImage: {
wallpaperBgImage: imghost + '/background.png',
@ -92,35 +92,50 @@ export default {
type: 0,
user_id: 0,
messageList: [],
title:"在线咨询"
title: "在线咨询",
options:{},
scrollTopRecord:0,
},
onLoad(options) {
let user = uni.getStorageSync('user');
console.log(user)
if(user.is_waitstaff !== 1 && user.is_waitstaff !== 3){
this.getToken(options.id, options.type ?? 1);
}else if(options.is_seize == 1){
this.connectChat(options.id);
}else{
if(options.type ==='2'){
this.getToken(options.id, options.type ?? 1);
}else{
this.getWaitToken(options.id)
}
}
onLoad(options) {
this.options = options;
// this.switchConnect(options);
},
onReady() {
uni.$m.uniSysInfo().then(res => {
if(!res?.windowHeight){
uni.$m.uniSysInfo(true).then(res => {
this.windowHeihgt = res.windowHeight;
});
}
this.windowHeihgt = res.windowHeight;
});
},
methods: {
testScroll(e){
this.scrollTopRecord = e.detail.scrollTop
},
switchConnect(options) {
let user = uni.getStorageSync('user');
console.log(user)
if (user.is_waitstaff !== 1 && user.is_waitstaff !== 3) {
this.getToken(options.id, options.type ?? 1);
} else if (options.is_seize == 1) {
this.connectChat(options.id);
} else {
if (options.type === '2') {
this.getToken(options.id, options.type ?? 1);
} else {
this.getWaitToken(options.id)
}
}
},
scrollToBottom() {
this.scrollTop = this.scrollTopRecord;
this.$nextTick(() => {
uni.createSelectorQuery().in(this).select('#scroll-view-content').boundingClientRect((res) => {
let top = res.height - this.windowHeihgt + 500;
console.log(res)
console.log(top)
let top = res.height + 500;
console.log('res',res)
console.log('top',top)
if (top > 0) {
this.scrollTop = top;
}
@ -132,50 +147,36 @@ export default {
this.chat_id = data.data.chat_id;
this.type = data.data.type;
this.user_id = data.data.user_id;
uni.connectSocket({
url: data.data.chat_url
});
this.connectSocket(data)
})
},
connectSocket(data) {
uni.connectSocket({
url: data.data.chat_url
});
uni.onSocketOpen((res) => {
this.ableChat = true;
console.log('WebSocket连接已打开!');
});
uni.onSocketOpen((res) => {
this.ableChat = true;
console.log('WebSocket连接已打开!');
});
uni.onSocketClose((res) => {
this.ableChat = false;
console.log('WebSocket 已关闭!');
});
uni.onSocketClose((res) => {
this.ableChat = false;
console.log('WebSocket 已关闭!');
});
uni.onSocketMessage((res) => {
this.newMessage(res);
})
uni.onSocketMessage((res) => {
this.newMessage(res);
})
},
connectChat(chat_id) {
seizeChat({chat_id:chat_id}).then(data => {
}, connectChat(chat_id) {
seizeChat({chat_id: chat_id}).then(data => {
this.chat_id = data.data.chat_id;
this.type = data.data.type;
this.user_id = data.data.user_id;
uni.connectSocket({
url: data.data.chat_url
});
uni.onSocketOpen((res) => {
this.ableChat = true;
console.log('WebSocket连接已打开!');
});
uni.onSocketClose((res) => {
this.ableChat = false;
console.log('WebSocket 已关闭!');
});
uni.onSocketMessage((res) => {
this.newMessage(res);
})
this.connectSocket(data);
}).catch(err => {
api.error(err.msg).then(() => {
uni.redirectTo({url:'/pages/Index/index'})
uni.redirectTo({url: '/pages/Index/index'})
return;
});
@ -186,23 +187,7 @@ export default {
this.chat_id = data.data.chat_id;
this.type = data.data.type;
this.user_id = data.data.user_id;
uni.connectSocket({
url: data.data.chat_url
});
uni.onSocketOpen((res) => {
this.ableChat = true;
console.log('WebSocket连接已打开!');
});
uni.onSocketClose((res) => {
this.ableChat = false;
console.log('WebSocket 已关闭!');
});
uni.onSocketMessage((res) => {
this.newMessage(res);
})
this.connectSocket(data)
})
},
newMessage(messages) {
@ -211,7 +196,7 @@ export default {
log(messages);
if (messages.do_type === 'init') {
this.messageList = messages.data;
uni.$u.sleep(200).then(()=>{
uni.$u.sleep(200).then(() => {
this.scrollToBottom();
})
@ -258,7 +243,7 @@ export default {
uni.hideLoading();
this.message = '';
}
uni.$u.sleep(200).then(()=>{
uni.$u.sleep(200).then(() => {
this.scrollToBottom();
})
}
@ -272,8 +257,21 @@ export default {
} catch {
}
},
onShow() {
console.log('onShow')
if(!this.ableChat){
this.switchConnect(this.options);
}
},
onHide(){
try{
uni.closeSocket({
reason:'页面被隐藏',
});
}catch (err){}
},
created() {
this.windowHeihgt = api.navHeight().windowHeihgt;
// this.windowHeihgt = api.navHeight().windowHeihgt;
}
}
</script>

13
pages/OrderPage/index.vue

@ -341,9 +341,9 @@
</view>
</view>
</uv-popup>
<uv-popup ref="doDatePopup" mode="bottom" round="20" closeable safeAreaInsetTop safeAreaInsetBottom custom-style="position:relative;">
<view class="title row" style="position:absolute;justify-content: center;width:100%;top:20rpx;left:0;">
<view class="value col" style="position: relative;text-align: center;">
<uv-popup ref="doDatePopup" mode="bottom" round="20" closeable :safeAreaInsetTop="false" safeAreaInsetBottom custom-style="position:relative;">
<view class="title row" style="position:static;justify-content: center;width:100%;top:20rpx;left:0;height:80rpx;align-items: center;">
<view class="value col" style="position: relative;text-align: center;display:flex;align-items: center;justify-content: center;">
<!-- <u-text text="协议内容" size="38" color="#191919"></u-text>-->
<view class="" style="position: relative;display:inline-block;">
@ -535,10 +535,13 @@
},
onShow(){
if(uni.getStorageSync('token')){
uni.$m.navTo('/pages/Login/index');
if(!uni.getStorageSync('token')){
this.navTo('/pages/Login/index');
}
console.log(this.isPay)
console.log(this.sign)
if(!this.isPay && this.sign === ''){
console.log(123123)
try {
this.getDefaultAddress();
}catch (e){

1
uni.scss

@ -14,6 +14,7 @@
/* 颜色变量 */
@import 'uview-plus/theme.scss';
@import '@/uni_modules/m-view/theme/theme.scss';
/* 行为相关颜色 */
$uni-color-primary: #007aff;
$uni-color-success: #4cd964;

21
uni_modules/m-view/LICENSE

@ -0,0 +1,21 @@
MIT License
Copyright (c) 2020 www.mview.com
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

3
uni_modules/m-view/changelog.md

@ -1,2 +1 @@
## 1.0.0
m-view
# mview

125
uni_modules/m-view/components/m-avatar/m-avatar.vue

@ -0,0 +1,125 @@
<template>
<view class="m-avatar-container">
<slot name="avatar">
<button v-if="!privateStatus && choose" open-type="chooseAvatar" class="avatarBtn" @chooseavatar="chooseAvatar" :style="{
width:size,height:size
}">
<image class="img" mode="aspectFill" :src="avatarUrl" :style="{width:size,height:size,borderRadius:radius}" />
</button>
<button v-else class="avatarBtn" :style="{
width:size,height:size
}" @click="choose ? needPrivate : ''">
<image class="img" mode="aspectFill" :src="avatarUrl" :style="{width:size,height:size,borderRadius:radius}" />
</button>
<!-- <image mode="aspectFill" :src="src" ></image>-->
</slot>
</view>
</template>
<script>
/**
* Sticky 头像
* @description 警告提示展现需要关注的信息
*
* @property {String} title 显示的文字
*/
const base64Avatar =
"";
export default {
name: 'm-avatar',
props:{
src:{
type:String,
default:''
},
defaultImage:{
type:String,
default: ''
},
size:{
type:String,
default: '150rpx'
},
choose:{
type:Boolean,
default: true
},
radius:{
type:String,
default: '50%'
}
},
data() {
return {
avatar:'',
privateStatus:true,
avatarUrl:'',
}
},
watch: {
// srcavatarUrlsrc
// props
src: {
immediate: true,
handler(newVal) {
this.avatarUrl = newVal
// srcerrorsrc''
if(!newVal) {
this.errorHandler()
}
}
}
},
computed: {
},
methods: {
errorHandler() {
this.avatarUrl = base64Avatar;
},
chooseAvatar(v) {
uni.$m.uploadFile(v.detail.avatarUrl).then(res => {
this.avatarUrl = res.show_path;
this.$emit('change',this.avatarUrl)
})
},
checkPrivacy(){
// #ifdef MP-WEIXIN
uni.getPrivacySetting({
success: res => {
console.log("是否需要授权:", res.needAuthorization, "隐私协议的名称为:", res.privacyContractName)
this.privateStatus = res.needAuthorization;
},
fail: () => {
},
complete: () => {},
})
// #endif
},
needPrivate(){
this.$emit('needPrivate',true);
},
},
mounted() {
this.checkPrivacy();
},
}
</script>
<style lang="scss" scoped>
.safe-area{
width:100%;
}
.m-sticky-container{
width:100%;
z-index: 997;
}
button{
padding:0;
background:none;
&::after{
border:none;
}
}
</style>

17
uni_modules/m-view/components/m-avatar/props.js

@ -0,0 +1,17 @@
export default {
props: {
// 是否占位
safeArea: {
type: Boolean,
default: true
},
type: {
type: String,
default: 'fixed'
},
offsetTop: {
type: String,
default: '0'
},
}
}

106
uni_modules/m-view/components/m-button/m-button.vue

@ -0,0 +1,106 @@
<script setup>
import {computed,defineProps} from "vue";
/**
* 按钮
* @description
*
* @property {String} title 显示的文字
*/
const emit = defineEmits(['click']);
function clickHandle(event){
emit('click',event);
}
const props = defineProps({
containerPadding:{
type:[String,Number],
default:'10px'
},
buttonPadding:{
type:[String,Number],
default:'10px'
},
textStyle:{
type:[Object],
default:''
},
justifyAlign:{
type:String,
default:'center'
},
verticalAlign:{
type:String,
default:'center'
},
border:{
type:String,
default:'1px solid #999999;'
},
buttonColor:{
type:String,
},
radius:{
type:[String,Number],
default:'10px'
},
text:{
type:String,
default:'确认'
},
customStyle:{
type:[String,Object],
default:{}
}
});
const containerStyle = computed(() => {
return {
'padding':props.containerPadding,
'justify-content':props.justifyAlign,
'align-items':props.verticalAlign,
}
});
const buttonStyle = computed(() => {
return {
...uni.$m.addStyle(props.customStyle),
'border':props.border,
'border-radius':props.radius,
'background-color':props.buttonColor,
'padding':props.buttonPadding,
}
});
</script>
<template>
<view class="m-button-container" @click="clickHandle" style="display: flex;" :style="containerStyle">
<view class="button" :style="buttonStyle">
<view class="m-button-pre">
<slot name="pre"></slot>
</view>
<view class="m-button-title" :style="textStyle">
<m-text
:text="text"
:size="textStyle.size"
:color="textStyle.color"
:line-height="textStyle.lineHeight"
:align="textStyle.align || 'center'"
:display="textStyle.display"
></m-text>
</view>
<view class="m-button-pre">
<slot name="after"></slot>
</view>
</view>
</view>
</template>
<style lang="scss">
.button{
flex:0 0 1;
width:100%;
text-align: center;
}
</style>

44
uni_modules/m-view/components/m-checkbox/m-checkbox-group.vue

@ -0,0 +1,44 @@
<script setup>
/**
* 复选框组
* @description
*
* @property {String} title 显示的文字
*/
import {ref, provide} from "vue";
const props = defineProps({
checkData:{type:[Object,Array],default:[]},
labelSize:{type:[String,Number],default:'26rpx'},
size:{type:[String,Number],default:'32rpx'},
activeColor:{type:String,default:'#578CF5'},
inActiveColor:{type:String,default:'transparent'},
borderInActiveColor:{type:String,default:'#999'},
labelStyle:{type:Object,default:{}}
});
const emit = defineEmits(['change','update:modelValue']);
const checked = ref([]);
provide('checkboxGroup',{
emit:emit,
checked:checked,
props:props,
boxStyle:boxStyle
});
function boxStyle (index) {
return {
borderColor:(checked.value.indexOf(index) !== -1)?props.activeColor:props.unActiveColor,
backgroundColor:(checked.value.indexOf(index) !== -1)?props.activeColor:'transparent'
}
}
</script>
<template>
<slot></slot>
</template>
<style lang="scss" scoped>
</style>

79
uni_modules/m-view/components/m-checkbox/m-checkbox.vue

@ -0,0 +1,79 @@
<script setup>
/**
* 复选框
* @description
*
* @property {String} title 显示的文字
*/
import {inject} from "vue";
//props
const props = defineProps({
checkData:{type:[Object,Array],default:[]},
labelSize:{type:[String,Number],default:'26rpx'},
size:{type:[String,Number],default:'32rpx'},
activeColor:{type:String,default:'#578CF5'},
inActiveColor:{type:String,default:'transparent'},
borderInActiveColor:{type:String,default:'#999'},
labelStyle:{type:Object,default:{}},
label:{type:String,default:''},
value:{type:Number,default:''}
});
const parent = inject('checkboxGroup');
//使checkbox
const emit = parent.emit || defineEmits(['change','update:modelValue']);
const checked = parent.checked;
//
function checkedHandle(label,value){
const posIndex = checked.value.indexOf(value)
if(posIndex !== -1){
checked.value.splice(posIndex,1);
}else{
checked.value.push(value);
}
//
emit('update:modelValue',checked.value);
//label value checked |
emit('change',{label:label,value:value,checked:(posIndex === -1)});
}
function boxStyle(param){
return parent.boxStyle(param);
}
</script>
<template>
<view class="m-checkbox-group">
<view class="m-checkbox">
<view class="item row align-center" @click="checkedHandle(label,value)">
<view class="box border-box mr-1" :style="boxStyle(value)">
<slot :active="checked.indexOf(value) !== -1">
<view class="icon" :style="{backgroundColor:(checked.indexOf(value) !== -1)?activeColor:inActiveColor}"></view>
</slot>
</view>
<view class="label"><m-text :text="label" :size="labelStyle.size" :line-height="labelStyle.lingHeight" :color="labelStyle.color"></m-text></view>
</view>
</view>
</view>
</template>
<style lang="scss" scoped>
.box{
width: $m-checkbox-size;
height: $m-checkbox-size;
border:3rpx solid;
border-radius:20%;
&.circle{
border-radius:50%;
}
.icon{
width:100%;
height:100%;
border-radius:50%;
background:transparent;
}
}
</style>

120
uni_modules/m-view/components/m-form-item/m-form-item.vue

@ -0,0 +1,120 @@
<script setup>
import {computed, inject, onMounted, ref, watch} from "vue";
import image from "@/config/image";
const imagesPublic = image.Public;
const props = defineProps({
label: {
type: String,
default: '',
},
name: {
type: String,
default: '',
},
align: {
type: String,
default: 'between',// vertical || between
},
itemStyle: {
type: [Object, String],
default: {}
},
labelStyle: {
type: [Object, String],
default: {}
},
inputStyle: {
type: [Object, String],
default: {}
},
customOption: {
type: [Object],
default: {}
},
border: {
type: [Boolean],
default: true
},
required: {
type: [Boolean],
default: false
},
prop: {
type: [String],
default: ''
},
});
const formItemStyle = computed(() => {
return {
paddingBottom: ((props.align === 'vertical') ? '0' : '35rpx'),
borderBottom: props.border ? '2rpx solid #EEEEEE' : 'none',
...uni.$m.addStyle(props.itemStyle),
}
})
const labelStyle = computed(() => {
return {
paddingLeft: (props.inputType === 'textarea' || props.inputType === 'upload') ? 0 : '35rpx',
paddingRight: (props.inputType === 'textarea' || props.inputType === 'upload') ? 0 : '35rpx',
...uni.$m.addStyle(props.labelStyle),
}
})
const inputStyle = computed(() => {
const padding = ((props.inputType === 'textarea' || props.inputType === 'upload') && type === 'form') ? 0 : '35rpx'
return {
paddingLeft: padding,
paddingRight: padding,
...uni.$m.addStyle(props.inputStyle),
}
})
const parent = inject('form')
onMounted(() => {
if(parent && props.required){
parent.rule.value.push({label:props.label || props.name,value:props.prop});
}
});
</script>
<template>
<view class="form-item row justify-between" :style="formItemStyle">
<view class="label border-box" :style="labelStyle" :class="{'vertical':(align === 'vertical')}">
<view class="required" v-if="required">
<m-text text="*" size="32rpx" align="center" line-height="50rpx" color="#FF3838"></m-text>
</view>
<m-text :text="label" size="32rpx" line-height="50rpx" color="#020b18"></m-text>
</view>
<view class="input border-box" :style="inputStyle" :class="{'vertical':(align === 'vertical')}">
<slot></slot>
</view>
</view>
</template>
<style lang="scss">
.form-item {
padding: 35rpx 0;
position:relative;
.label{
position: relative;
}
.required{
position: absolute;
width:35rpx;
height:100%;
top:8rpx;
left:0;
}
&.bottom {
padding-bottom: 10rpx;
}
}
.vertical {
flex: 0 0 100%;
padding-bottom: 10rpx;
}
</style>

79
uni_modules/m-view/components/m-form/m-form.vue

@ -0,0 +1,79 @@
<script setup>
import {nextTick, provide, ref, watch} from "vue";
const props = defineProps({
type:{
type:String,
default:'form'
},
model:{
type:Object,
default:{}
},
isRead:{
type:Boolean,
default:false,
},
disabled:{
type:Boolean,
default:false,
}
})
watch(() => props.type,()=>{
uni.$emit('formType',props.type);
});
const rule = ref([]);
provide('form',{
rule:rule,
props:props
});
function verify(){
try {
// nextTick(() => {
return new Promise((resolve, reject) => {
rule.value.some(item => {
if(!props.model[item.value]){
uni.$m.error(item.label+'未填写')
reject(item)
return true;
}
})
resolve(true);
}).catch(e => {
console.log('e',e)
})
// })
}catch (err) {
console.log('err',err)
}
}
defineExpose({
verify
})
</script>
<template>
<view class="form border-box" style="position:relative;z-index:2;background-color:#fff;padding-top:0;">
<slot></slot>
</view>
</template>
<style lang="scss">
.form-item {
padding: 35rpx 0;
&.bottom {
padding-bottom: 10rpx;
}
}
.vertical{
flex: 0 0 100%;
padding-bottom:10rpx;
}
</style>

55
uni_modules/m-view/components/m-header/m-header.vue

@ -0,0 +1,55 @@
<template>
<view class="mz-header-container" :style="{top:(offsetTop === '0')?offSetDefault:offsetTop,position:type}">
<view class="safe-area-top" :style="{height:safeHeight,display:((type !== 'sticky') || 'none')}"></view>
<slot>
</slot>
</view>
</template>
<script>
import props from './props.js';
/**
* header 头部
* @description 警告提示展现需要关注的信息
*
* @property {String} title 显示的文字
*/
export default {
name: 'mz-header',
mixins: [props],
data() {
return {
show: true,
safeHeight:''
}
},
computed: {
offSetDefault(){
return uni.getSystemInfoSync().statusBarHeight+44+'px';
},
},
methods: {
getSafeHeight(){
new Promise((resolve,reject) => {
resolve(uni.getSystemInfoSync().statusBarHeight);
}).then((height) => {
this.safeHeight = height+'px';
})
},
},
mounted() {
this.getSafeHeight();
},
}
</script>
<style lang="scss" scoped>
.safe-area{
width:100%;
}
.mz-header-container{
width:100%;
z-index: 997;
}
</style>

17
uni_modules/m-view/components/m-header/props.js

@ -0,0 +1,17 @@
export default {
props: {
// 是否占位
safeArea: {
type: Boolean,
default: true
},
type: {
type: String,
default: 'fixed'
},
offsetTop: {
type: String,
default: '0'
},
}
}

209
uni_modules/m-view/components/m-input/m-input.vue

@ -0,0 +1,209 @@
<script setup>
import {computed} from "vue";
const props = defineProps({
cursor: {
type: Number,
default: 0
},
innerValue: {
type: String,
default: ''
},
autoBlur: {
type: Boolean,
default: false
},
disabled: {
type: Boolean,
default: false
},
readonly: {
type: Boolean,
default: false
},
maxlength: {
type: Number,
default: 255
},
placeholder: {
type: String,
default: '请输入'
},
placeholderStyle: {
type: String,
default: ''
},
placeholderClass: {
type: String,
default: ''
},
confirmType: {
type: String,
default: ''
},
confirmHold: {
type: Boolean,
default: false
},
holdKeyboard: {
type: Boolean,
default: false
},
cursorSpacing: {
type: Number,
default: 0
},
adjustPosition: {
type: Boolean,
default: true
},
selectionEnd: {
type: Number,
default: -1
},
selectionStart: {
type: Number,
default: -1
},
password: {
type: Boolean,
default: false
},
type: {
type: String,
default: 'input'
},
ignoreCompositionEvent: {
type: Boolean,
default: true
},
inputStyle: {
type: [Object, String],
default: {}
},
shape: {
type: String,
default: 'circle'
},
fontSize: {
type: String,
default: '28rpx'
},
border: {
type: String,
default: 'surround'//surround | bottom | none
},
borderColor: {
type: String,
default: '#999'
},
padding: {
type: String,
default: '30rpx'
},
customStyle: {
type: [String,Object],
default: {}
},
});
const emit = defineEmits([
'update:modelValue',
'change',
'blur',
'focus',
'confirm',
'keyBoardHeightChange',
'input',
]);
class EventGroup {
onFocus(e) {
emit('focus', e)
emit('update:modelValue', e)
}
onConfirm(e) {
emit('confirm', e)
emit('update:modelValue', e)
}
onKeyBoardHeightChange() {
emit('keyBoardHeightChange', e)
}
onBlur(e) {
emit('blur', e)
emit('update:modelValue', e)
}
onInput(e) {
emit('input', e)
emit('update:modelValue', e)
}
focus() {
}
}
const InputEvent = new EventGroup();
const inputStyle = computed(() => {
const style = {};
if(props.border === 'surround')style.border = '2rpx solid '+props.borderColor;
if(props.border === 'bottom')style.borderBottom = '2rpx solid '+props.borderColor;
return {
padding: props.padding,
border: '2rpx solid #666',
borderRadius: props.shape === 'circle'?'100px':'4px',
...style,
...uni.$m.addStyle(props.customStyle)
}
});
</script>
<template>
<view class="mz-input-container">
<view class="input">
<input
class="u-input__content__field-wrapper__field"
:style="[inputStyle]"
:type="type"
:focus="focus"
:cursor="cursor"
:value="innerValue"
:auto-blur="autoBlur"
:disabled="disabled || readonly"
:maxlength="maxlength"
:placeholder="placeholder"
:placeholder-style="placeholderStyle"
:placeholder-class="placeholderClass"
:confirm-type="confirmType"
:confirm-hold="confirmHold"
:hold-keyboard="holdKeyboard"
:cursor-spacing="cursorSpacing"
:adjust-position="adjustPosition"
:selection-end="selectionEnd"
:selection-start="selectionStart"
:password="password || type === 'password' || undefined"
:ignoreCompositionEvent="ignoreCompositionEvent"
@input="InputEvent.onInput"
@blur="InputEvent.onBlur"
@focus="InputEvent.onFocus"
@confirm="InputEvent.onConfirm"
@keyboardheightchange="InputEvent.onKeyBoardHeightChange"
>
</view>
</view>
</template>
<style lang="scss">
.mz-input-container {
}
</style>

100
uni_modules/m-view/components/m-private/m-private.vue

@ -0,0 +1,100 @@
<template>
<uv-popup ref="popup" type="center" :closeOnClickOverlay="false" :is-mask-click="false" round="20rpx" mode="bottom"
custom-style="padding:20rpx">
<view class="popup-box" style="padding:40rpx">
<view style="padding:20rpx 0;">
<m-text bold text="用户隐私保护提示" color="#020b18" size="32rpx"></m-text>
</view>
<view class="row">
<view class="">
<m-text color="#020b18" line-height="50rpx" text="用户隐私保护提示" size="28rpx"></m-text>
</view>
<view class="">
<m-text color="#178AF2" size="28rpx" line-height="50rpx" text="《小程序隐私保护指引》"
@click="openPrivacyContract"></m-text>
</view>
<view class="">
<m-text color="#020b18" size="28rpx" line-height="50rpx"
text="当您点击同意并开始使用产品服务时,即表示你已理解并同意该条款内容,该条款将对您产生法律约束力。如您拒绝,将无法更好的体验产品。"></m-text>
</view>
</view>
<view class="row" style="padding:30rpx;display:flex;box-sizing: border-box;">
<view class="col-6" style="padding:15rpx;flex:0 0 50%;box-sizing: border-box;">
<button @click="closePrivacyAuthorization">
<m-text text="拒绝" size="32rpx" line-height="100rpx" align="center" color="#666"></m-text>
</button>
</view>
<view class="col-6" style="padding:15rpx;flex:0 0 50%;box-sizing: border-box;">
<button id="agree-btn" open-type="agreePrivacyAuthorization" color="#45C55E" class="weui-btn agree"
@agreeprivacyauthorization="handleAgreePrivacyAuthorization" :style="{backgroundColor:buttonColor}">
<m-text text="同意" size="32rpx" line-height="100rpx" align="center" color="#fff"></m-text>
</button>
</view>
</view>
</view>
</uv-popup>
</template>
<script>
/**
* Sticky 头像
* @description 警告提示展现需要关注的信息
*
* @property {String} title 显示的文字
*/
export default {
name: 'm-private',
props: {
src: {
type: String,
default: ''
},
defaultImage: {
type: String,
default: ''
},
size: {
type: String,
default: '150rpx'
},
buttonColor: {
type: String,
default: '#45C55E'
}
},
data() {
return {
avatar: '',
}
},
computed: {},
methods: {
open() {
this.$refs.popup.open();
},
openPrivacyContract() {
uni.openPrivacyContract({});
},
closePrivacyAuthorization() {
this.$refs.popup.close();
},
handleAgreePrivacyAuthorization() {
this.$refs.popup.close();
this.$emit('agreePrivate',true);
}
},
mounted() {
},
}
</script>
<style lang="scss" scoped>
button.agree{
//background-color:#45C55E;
color:#ffffff;
}
</style>

17
uni_modules/m-view/components/m-private/props.js

@ -0,0 +1,17 @@
export default {
props: {
// 是否占位
safeArea: {
type: Boolean,
default: true
},
type: {
type: String,
default: 'fixed'
},
offsetTop: {
type: String,
default: '0'
},
}
}

106
uni_modules/m-view/components/m-radio/m-radio.vue

@ -1,29 +1,91 @@
<script setup>
import {ref} from "vue";
const props = defineProps({
modelValue: {
type: String,
default: ''
},
inputSize: {
type: String,
default: '28rpx'
},
radioAlign: {
type: [String],
default: 'row'
},
radioData: {
type: Array,
default: [],
},
labelName: {
type: String,
default: 'label',
},
valueName: {
type: String,
default: 'value',
},
activeColor: {
type: String,
default: '#BE1414',
},
unActiveColor: {
type: String,
default: '#333333',
},
});
const emit = defineEmits(['change', 'update:modelValue']);
const radioChecked = ref('');
function radioChange(name) {
radioChecked.value = name;
emit('update:modelValue', name);
emit('change', name);
}
</script>
<template>
<view>
<slot>
<view class="radio-group row"
style="padding:22rpx 0;display: flex;justify-content:flex-start;">
<view
class="radio flex align-center"
:class="{
col:(radioAlign === 'row'),
'col-12':(radioAlign !== 'row'),
'pt-1':(radioAlign !== 'row'),
'pb-2':(radioAlign !== 'row'),
}"
v-for="item in radioData"
@click="radioChange(item[valueName])"
>
<view class="checkbox border-box" style="margin-right:10rpx;">
<view
class="border-box"
style="width:32rpx;height:32rpx;border-radius: 50%;padding:6rpx;"
:style="{border:modelValue === item[valueName]?'3rpx solid #BE1414':'3rpx solid rgb(51,51,51)'}"
>
<view
class="border-box"
style="width:100%;height:100%;border-radius: 50%;"
:style="{backgroundColor:(modelValue === item[valueName]?activeColor:'transparent')}"
>
</slot>
</view>
</view>
</view>
<view class="label">
<m-text :text="item[labelName]" size="32rpx" :color="modelValue === item[valueName]?activeColor:unActiveColor"
custom-style="white-space:nowrap;"></m-text>
</view>
</view>
</view>
</view>
</template>
<script>
/**
* 单选
* @description
*
* @property {String} title 显示的文字
*/
export default {
name: 'm-radio',
methods: {
},
mounted() {
},
}
</script>
<style lang="scss" scoped>
<style scoped lang="scss">
</style>
</style>

41
uni_modules/m-view/components/m-safe-bottom/m-safe-bottom.vue

@ -0,0 +1,41 @@
<template>
<view style="width:100%;" :style="{height:height}"></view>
</template>
<script>
export default {
name: 'm-private',
props: {
src: {
type: String,
default: ''
},
defaultImage: {
type: String,
default: ''
},
size: {
type: String,
default: '150rpx'
}
},
data() {
return {
height: '0',
}
},
computed: {},
methods: {},
mounted() {
uni.$m.uniSysInfo().then(response => {
this.height = response.safeAreaInsetBottom + 'px';
})
}
}
</script>
<style lang="scss" scoped>
page {
width: 100%;
}
</style>

17
uni_modules/m-view/components/m-safe-bottom/props.js

@ -0,0 +1,17 @@
export default {
props: {
// 是否占位
safeArea: {
type: Boolean,
default: true
},
type: {
type: String,
default: 'fixed'
},
offsetTop: {
type: String,
default: '0'
},
}
}

41
uni_modules/m-view/components/m-safe-top/m-safe-top.vue

@ -0,0 +1,41 @@
<template>
<view style="width:100vw;" :style="{height:height}"></view>
</template>
<script>
export default {
name: 'm-private',
props: {
src: {
type: String,
default: ''
},
defaultImage: {
type: String,
default: ''
},
size: {
type: String,
default: '150rpx'
}
},
data() {
return {
height: '0',
}
},
computed: {},
methods: {},
mounted() {
uni.$m.uniSysInfo().then(response => {
this.height = response.sysAndMenuHeight + 'px';
})
}
}
</script>
<style lang="scss" scoped>
page {
width: 100%;
}
</style>

17
uni_modules/m-view/components/m-safe-top/props.js

@ -0,0 +1,17 @@
export default {
props: {
// 是否占位
safeArea: {
type: Boolean,
default: true
},
type: {
type: String,
default: 'fixed'
},
offsetTop: {
type: String,
default: '0'
},
}
}

79
uni_modules/m-view/components/m-sticky/m-sticky.vue

@ -0,0 +1,79 @@
<template>
<view class="safe-area" :style="{height:safeHeight,display:((type !== 'sticky') ? 'block' : 'none')}"></view>
<view class="m-sticky-container" :style="containerStyle">
<slot />
</view>
</template>
<script>
import props from './props.js';
/**
* Sticky 吸顶
* @description 警告提示展现需要关注的信息
*
* @property {String} title 显示的文字
*/
export default {
name: 'm-sticky',
mixins: [props],
data() {
return {
show: true,
safeHeight:'',
offsetHeightValue:'0',
}
},
computed: {
offSetDefault(){
let offsetHeight = uni.getSystemInfoSync().statusBarHeight;
// #ifdef MP-WEIXIN
offsetHeight += 40;
// #endif
return offsetHeight;
},
containerStyle() {
let style = {
position: this.type,
zIndex: this.zIndex
}
if(this.mode === 'bottom'){
style.bottom = this.offsetHeightValue;
}else{
style.top = this.offsetHeightValue;
}
return style;
}
},
methods: {
getSafeHeight(){
new Promise((resolve) => {
uni.$m.sleep(10).then(() => {
const info = uni.createSelectorQuery().in(this).select('.m-sticky-container');
info.boundingClientRect(function (data) {
resolve(data.height);
}).exec(function (res) {});
});
}).then((height) => {
this.safeHeight = height+'px';
this.offsetHeightValue = (parseFloat(
uni.$m.getPx((this.mode === 'top') ?(this.top || this.offSetDefault) :0,false))
+ parseFloat(uni.$m.getPx(this.offset,false))+'px')
this.$emit('change',true);
})
},
},
mounted() {
this.getSafeHeight();
},
}
</script>
<style lang="scss" scoped>
.safe-area{
width:100%;
}
.m-sticky-container{
width:100%;
}
</style>

28
uni_modules/m-view/components/m-sticky/props.js

@ -0,0 +1,28 @@
export default {
props: {
// 是否占位
safeArea: {
type: Boolean,
default: true
},
type: {
type: String,
default: 'fixed'
},
offset: {
type: String,
default: '0'
},
zIndex: {
type: String,
default: '997'
},
mode:{
type: String,
default: 'top'
},
top:{
type: String,
},
}
}

71
uni_modules/m-view/components/m-text/m-text.vue

@ -0,0 +1,71 @@
<script setup>
import {computed} from "vue";
const props = defineProps({
text:{
type:String,
default:''
},
align:{
type:String,
default:'left'
},
size:{
type:[String,Number],
default:'16px'
},
bold:{
type:[String,Number],
default:'normal'
},
color:{
type:String,
default:'#000000'
},
lineHeight:{
type:[String,Number],
default:'16px'
},
display:{
type:String,
default:'block'
},
lines:{
type:Number,
},
customStyle:{
type:[String,Object],
default:{}
}
})
const textStyle = computed(() => {
return {
...uni.$m.addStyle(props.customStyle),
'text-align':props.align,
'font-size':props.size,
'font-weight':props.bold,
'line-height':props.lineHeight,
'color':props.color,
'display':props.display,
}
})
const emit = defineEmits(['click'])
function handleClick(e){
emit('click',e);
}
</script>
<template>
<view class="m-text-container" :class="[lines && `u-line-${lines}`]" :style="textStyle" @click="handleClick">
<slot name="default">
{{text}}
</slot>
</view>
</template>
<style scoped lang="scss">
</style>

26
uni_modules/m-view/index.js

@ -0,0 +1,26 @@
import api from './lib/functions.js'
import tools from './lib/tools.js'
import service from "@/uni_modules/m-view/lib/request";
const $m = {
...api,
...tools,
service
}
// $m挂载到uni对象上
uni.$m = $m
const install = (Vue) => {
// #ifndef APP-NVUE
// 只有vue,挂载到Vue.prototype才有意义,因为nvue中全局Vue.prototype和Vue.mixin是无效的
Vue.config.globalProperties.$m = $m
// 暂时兼容vue2的$nextTick写法
Vue.config.globalProperties.$nextTick = (cb) => {
cb();
}
// #endif
}
export default {
install
}

1
uni_modules/m-view/index.scss

@ -0,0 +1 @@
@import "./theme/common.scss";

8
uni_modules/m-view/lib/api.js

@ -0,0 +1,8 @@
import {host} from "../../../api/config.js";
export function getSts(data) {
return uni.$m.service({
url: `${host}/user/agree-privacy`,
method: "post",
});
}

598
uni_modules/m-view/lib/functions.js

@ -0,0 +1,598 @@
// #ifdef MP-WEIXIN
import log from "../../../utils/log";
const QQMapWX = require('./qqmap-wx-jssdk.min.js');
import * as QQMapwx from './/qqmap-wx-jssdk.min.js';
var qqMapSdk;
// #endif
// import {getSTS} from "@/api/other";
import {imgHost, host, oss} from '@/config/host.js'
import {
INDEX_PATH,
LOGIN_PATH,
OTHER_FIELD,
PAGEINFO_FIELD,
TOKEN_FIELD,
UPLOAD_PATH,
USERINFO_FIELD
} from '@/config/config.js'
import store from "@/store";
import {LIMIT_FIELD} from "../../../config/config";
// import store from "@/store";
// const store = useStore();
export default {
loginModalState: true,
mapLocation: () => {
return new Promise((relove, reject) => {
wx.authorize({
scope: 'scope.userFuzzyLocation',
success(res) {
if (res.errMsg == 'authorize:ok') {
wx.getFuzzyLocation({
type: 'wgs84',
success(re) {
console.log(re);
uni.chooseLocation({
latitude: re.latitude,
longitude: re.longitude,
success: function (data) {
console.log(data);
let obj = {
latitude: data.latitude,
longitude: data.longitude,
address: data.address
}
if (data.errMsg == 'chooseLocation:ok') {
qqMapSdk = new QQMapWX({
key: 'K65BZ-LULE4-744UV-X4EIA-27RA5-A4FWT'
})
qqMapSdk.reverseGeocoder({
location: {
latitude: data.latitude,
longitude: data.longitude
},
success: function (mapres) {
console.log(mapres);
if (mapres.message == 'query ok') {
obj.name = mapres.result.formatted_addresses.recommend
obj.city = mapres.result.address_component.city
relove(obj)
}
}
})
}
}
});
}
})
}
},
fail(err) {
reject(err);
}
})
})
},
getLocation(details) {
return new Promise((relove, reject) => {
qqMapSdk = new QQMapWX({
key: 'K65BZ-LULE4-744UV-X4EIA-27RA5-A4FWT'
})
qqMapSdk.geocoder({
//获取表单传入地址
address: details.activedeta.city + details.activedeta.address, //地址参数,例:固定地址,address: '北京市海淀区彩和坊路海淀西大街74号'
success: function (res) {//成功后的回调
console.log(res);
relove(res.result)
},
fail: function (error) {
console.error(error);
},
complete: function (res) {
console.log(res);
}
})
})
},
urlToObj: (url) => {
let obj = {}
let str = url.slice(url.indexOf('?') + 1)
let arr = str.split('&')
for (let j = arr.length, i = 0; i < j; i++) {
let arr_temp = arr[i].split('=')
obj[arr_temp[0]] = arr_temp[1]
}
return obj
},
uploadFile: (file) => {
// console.log(file);
return new Promise((resolve, reject) => {
uni.showLoading({
title: '上传中'
}).then()
uni.uploadFile({
url: host + UPLOAD_PATH, //仅为示例,非真实的接口地址
filePath: file,
name: 'file',
success: (uploadFileRes) => {
uni.hideLoading()
resolve(JSON.parse(uploadFileRes.data).data)
},
fail: (err => {
reject(err);
})
});
})
},
getImgUrl: (url) => {
if (url) {
if (url.indexOf('http') === -1) {
return imgHost + url
} else {
return url
}
} else {
return ''
}
},
getImgUrlToHttps: (url) => {
if (url) {
if (url.indexOf('https') == -1) {
console.log(url.replace('http', 'https'))
return url.replace('http', 'https')
} else {
return url
}
} else {
return ''
}
},
/**
* 失败
*/
error: (msg) => {
return new Promise((relove, reject) => {
uni.showToast({
title: msg,
icon: "none"
})
setTimeout(() => {
relove();
}, 1500)
})
},
/**
* 成功提示
*/
success: (msg) => {
return new Promise((resolve, reject) => {
uni.showToast({
title: msg,
icon: "success"
})
setTimeout(() => {
resolve();
}, 1500)
})
},
/**
* 确认对话框
* @param {Object} contents
* @param {Object} callback
*/
confirm: (contents) => {
return new Promise((relove, reject) => {
uni.showModal({
title: '提示',
content: contents,
success: function (res) {
if (res.confirm) {
relove();
}
}
});
})
},
payment: (param) => {
return new Promise((relove, reject) => {
uni.requestPayment({
provider: 'wxpay',
timeStamp: param.timeStamp,
nonceStr: param.nonceStr,
package: param.package,
signType: param.signType,
paySign: param.paySign,
success: function (res) {
relove(res);
},
fail: function (err) {
console.log(err);
api.error("支付失败");
reject(err);
}
});
})
},
navHeight: () => {
let system, menu, systemBar, navigationBar, navPaddingTop, navHeight, headerPadding;
//获取系统信息
try {
wx.getSystemInfo({
success: res => {
this.system = res
}
})
// log('res',this.system);
//获取胶囊信息
this.menu = wx.getMenuButtonBoundingClientRect()
this.systemBar = this.system.statusBarHeight //状态栏高度
this.navigationBar = (this.menu.top - this.system.statusBarHeight) * 2 + this.menu.height //胶囊高度
this.navPaddingTop = this.systemBar;
this.navHeight = this.menu.height;
this.headerPadding = (this.menu.top - this.systemBar);
this.systemBarHeight = (this.navPaddingTop + this.navHeight + (this.headerPadding * 2))
} catch (e) {
//默认高度,刘海屏
this.navPaddingTop = 43;
this.navHeight = 32;
this.headerPadding = 4;
}
return {
navPaddingTop: this.navPaddingTop,
navHeight: this.navHeight,
headerPadding: this.headerPadding,
systemBarHeight: this.systemBarHeight,
windowHeihgt: (this.system.windowHeight - this.systemBarHeight)
};
},
wxSystemInfo: () => {
let system;
//获取系统信息
try {
wx.getSystemInfo({
success: res => {
this.system = res
}
})
} catch (e) {
}
return {
system: this.system
};
},
uniSysInfo: (flush = false) => {
return new Promise((resolve, reject) => {
if (uni.getStorageSync('sysInfo') && !flush) {
resolve(uni.getStorageSync('sysInfo'));
} else {
//#ifdef MP-WEIXIN
let menuButtonInfo = uni.getMenuButtonBoundingClientRect();
// #endif
//#ifdef APP-PLUS
let menuButtonInfo = {height:0,top:0};
// #endif
let sysInfo = uni.getSystemInfoSync();
let sysMenuHeight = sysInfo.statusBarHeight;
let sysWindowHeight = sysInfo.windowHeight;
let menuButtonHeight = menuButtonInfo.height
let menuButtonPadding = menuButtonInfo.top - sysMenuHeight;
let sysAndMenuHeight = sysMenuHeight + 44;// (sysMenuHeight+menuButtonHeight+(menuButtonPadding*2));
let windowHeight = (sysWindowHeight - sysAndMenuHeight);
let safeAreaInsetBottom = (sysInfo.safeAreaInsets.bottom);
let data = {
sysMenuHeight: sysMenuHeight,
menuButtonHeight: menuButtonHeight,
menuButtonPadding: menuButtonPadding,
sysAndMenuHeight: sysAndMenuHeight,
windowHeight: windowHeight,
safeAreaInsetBottom: safeAreaInsetBottom,
}
uni.setStorageSync('sysInfo', data);
resolve(data);
}
})
},
uploadOssFile: (file) => {
return new Promise((resolve, reject) => {
// getSTS({}).then(res => {
// // const client = new OSS(res.data);
// // console.log(file);
// uni.showLoading({
// title:'上传中'
// })
// let timer = new Date();
// let address = timer.getFullYear() + '' + (timer.getMonth() + 1) + '' + timer.getDate();
// address = 'upload/' + address + '/';
// var imageSrc = file;
// let str = file.substr(file.lastIndexOf('.'));
// let nameStr = address + timer.getTime() + str;
// uni.uploadFile({
// url: `${oss}`, //仅为示例,非真实的接口地址
// filePath:file,
// name: 'file',
// header: {
// "content-type": "application/json;charset=utf-8",
// },
// formData:Object.assign({
// name:nameStr,
// key:nameStr,
// success_action_status: '200',
// },res.data),
// success: (uploadFileRes) => {
// uni.hideLoading()
// resolve ({show_path:oss+'/' + nameStr});
// // relove(JSON.parse(uploadFileRes.data).data)
// // console.log(JSON.parse(uploadFileRes.data).data);
// },
// fail:(res) =>{
// uni.hideLoading()
// console.log(res)
// return {};
// }
// });
// });
})
},
logout: (reLaunch = true, confirm = true) => {
return new Promise((resolve, reject) => {
if (confirm) {
uni.$m.confirm('是否退出登录').then(res => {
uni.$m.doLogout(reLaunch)
resolve();
})
} else {
uni.$m.doLogout(reLaunch);
resolve();
}
})
},
doLogout: (reLaunch = true) => {
uni.showLoading({title: '退出中'}).then();
uni.removeStorageSync('token');
// store.commit('userInfo',{});
// store.commit('userExtends',{});
// uni.$m.uniSysInfo().then(() => {
if (reLaunch) {
uni.reLaunch({url: INDEX_PATH}).then();
}
uni.hideLoading();
// }).catch(err => {
// uni.hideLoading();
// })
},
navTo: (url, token = false) => {
uni.$m.checkToken(token, () => {
uni.navigateTo({
url: url
}).then();
});
},
navBack: () => {
uni.navigateBack({
delta:1,
success:() => {
console.log('backSuccess')
},
fail: (err) => {
uni.navigateTo({
url: INDEX_PATH
}).then()
}
});
},
switchTo: (url, token = false) => {
uni.$m.checkToken(token, () => {
uni.switchTab({
url: url
}).then();
});
},
redTo: (url, token = false) => {
uni.$m.checkToken(token, () => {
uni.redirectTo({
url: url
}).then();
});
},
checkToken: (token, callback = ()=>{}) => {
if (token) {
if (uni.getStorageSync('token')) {
callback(true)
} else {
uni.navigateTo({
url: LOGIN_PATH
}).then();
}
} else {
callback(true)
}
},
getFileType: (name) => {
let str = name.substr(name.lastIndexOf('.') + 1).toUpperCase();
const imageType = ['JPG', 'JPEG', 'PNG', 'GIF', 'BMP', 'TIF', 'TGA', 'DDS', 'SVG', 'EPS', 'RAW', 'PSD'];
const videoType = ['MP4', 'MOV', 'WMV', 'FLV', 'AVI', 'AVCHD', 'WEBM', 'MKV'];
if (imageType.indexOf(str) !== -1) {
return 'image';
} else if (videoType.indexOf(str) !== -1) {
return 'video';
} else {
return 'none';
}
},
sleep: (time) => {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve();
}, time);
});
},
getDomHeight: (domName) => {
return new Promise((resolve) => {
uni.$m.sleep(10).then(() => {
const info = uni.createSelectorQuery().select(domName);
info.boundingClientRect(function (data) {
console.log(data)
resolve(data.height);
}).exec(function (res) {
});
});
})
},
isTop: (data) => {
return data.scrollTop <= 20;
},
//设置用户信息
setUserInfo: (data) => {
uni.setStorageSync(TOKEN_FIELD, data[TOKEN_FIELD]);
if(USERINFO_FIELD){
uni.setStorageSync(USERINFO_FIELD, data[USERINFO_FIELD]);
}
if(OTHER_FIELD){
uni.setStorageSync(OTHER_FIELD, data[OTHER_FIELD]);
}
uni.$emit('userInfoUpdate',data[USERINFO_FIELD]);
store.commit('userInfo', data[USERINFO_FIELD]);
},
// 获取用户信息
getUserInfo: (field = null) => {
const user = uni.getStorageSync(USERINFO_FIELD);
return field ? (user[field] || '') : user;
},
// 获取token
getToken: () => {
return uni.getStorageSync(TOKEN_FIELD);
},
//微信网页登录
wechatWeb: (request, params) => {
request(params).then(res => {
uni.$m.setUserInfo(res.data);
uni.$m.success('登录成功')
setTimeout(() => {
if (params.url) {
uni.reLaunch({url: params.url}).then();
} else {
uni.reLaunch({url: INDEX_PATH}).then();
}
}, 2000)
});
},
setPageInfo: (pageName, data) => {
let pageInfo = uni.getStorageSync(PAGEINFO_FIELD) || {};
pageInfo[pageName] = data;
uni.setStorageSync(PAGEINFO_FIELD, pageInfo);
},
getPageInfo: (pageName) => {
let pageInfo = uni.getStorageSync(PAGEINFO_FIELD) || {};
return pageInfo[pageName];
},
list: (request, lists, pages, status) => {
return new Promise((resolve, reject) => {
if (status.value === 'nomore') {
reject(false);
return false;
}
status.value = 'loading';
resolve(request(pages).then(res => {
if (res.data.length === 0) {
status.value = 'nomore'
return false;
} else {
lists.value = lists.value.concat(res.data)
if (res.data.length < parseFloat(pages[LIMIT_FIELD] + '')) {
status.value = 'nomore'
} else {
status.value = 'loadmore'
}
}
pages.page++;
}).catch((err) => {
status.value = 'loadmore'
}));
})
},
copy: (data) => {
return new Promise((resolve, reject) => {
uni.setClipboardData({
data: data,
success: function () {
resolve();
},
fail: function () {
reject();
}
});
})
},
callPhone: (phone = '') => {
if (!phone) {
uni.$m.error('手机号不能为空');
return false;
}
uni.makePhoneCall({
phoneNumber: phone
});
},
callPosition: (lat, lng,name) => {
const latitude = parseFloat(lat);
const longitude = parseFloat(lng);
uni.openLocation({
latitude: latitude,
longitude: longitude,
name: name,
})
},
getAddressLocation(address) {
return new Promise((relove, reject) => {
qqMapSdk = new QQMapWX({
key: 'K65BZ-LULE4-744UV-X4EIA-27RA5-A4FWT'
})
qqMapSdk.geocoder({
//获取表单传入地址
address: address, //地址参数,例:固定地址,address: '北京市海淀区彩和坊路海淀西大街74号'
success: function (res) {//成功后的回调
console.log(res);
relove(res.result)
},
fail: function (error) {
console.error(error);
},
complete: function (res) {
console.log(res);
}
})
})
},
async logoutYW(){
await uni.$m.confirm('确认退出登录?');
await uni.showLoading({title: '退出中'});
await uni.removeStorageSync('token');
uni.hideLoading();
uni.reLaunch(LOGIN_PATH)
}
};

3
uni_modules/m-view/lib/qqmap-wx-jssdk.min.js vendored

File diff suppressed because one or more lines are too long

74
uni_modules/m-view/lib/request.js

@ -0,0 +1,74 @@
import {TOKEN_FIELD} from '@/config/config.js'
// import {refreshToken} from "../../../api";
const service = (params,showToast = true,type='default') => {
return new Promise((resolve, reject) => {
let token=uni.getStorageSync(TOKEN_FIELD)
// let token = uni.getStorageSync('token')
// let now = Math.round(new Date() / 1000);
// if(now > token.token_exp){
// refreshToken();
// }
uni.request({
url: params.url,
data: params.data,
method: params.method,
dataType: "json",
header: {
"content-type": "application/json;charset=utf-8",
"Authorization":"Bearer "+ token.token
}
}).then(res => {
if (res.statusCode === 200) {
let data = res.data;
switch (data.code) {
case 200:
resolve(data);
break;
case 202:
if(showToast){
uni.hideLoading()
uni.$m.error(data.msg).then(() => {
reject(data);
});
}else{
reject(data);
}
break;
case 203:
resolve(data);
break;
case 1001:
uni.setStorageSync('token',data.data)
if(type === 'default'){
console.log('default')
resolve(service(params))
}else{
console.log('refresh')
resolve();
}
break;
case 1002:
if(showToast){
uni.hideLoading()
uni.$m.error(data.msg);
}
reject(data);
break;
default:
reject(data);
break;
}
} else {
uni.$m.error('连接服务器失败')
}
})
})
}
export default service;

215
uni_modules/m-view/lib/tools.js

@ -0,0 +1,215 @@
export default {
/**
* 添加单位
* @param num
* @param unit
* @returns {string}
*/
addUnit(num, unit = 'px') {
return (/px$/.test(num)) ? num : num + unit;
},
/**
* 转换成px
* 和upx2px区别是可带单位
* @param num
* @returns {string}
*/
getPx(num, isUnit = true) {
return this.addUnit((
/(rpx|upx)$/.test(num))
? uni.upx2px(parseFloat(num))
: parseFloat(num), isUnit ? 'px' : '');
},
/**
* 样式转换格式
* @param customStyle
* @param target
* @returns {{}|*|string|string}
*/
addStyle(customStyle, target = 'object') {
// 字符串转字符串,对象转对象情形,直接返回
if (this.empty(customStyle) || typeof (customStyle) === 'object' && target === 'object' || target === 'string' &&
typeof (customStyle) === 'string') {
return customStyle
}
// 字符串转对象
if (target === 'object') {
// 去除字符串样式中的两端空格(中间的空格不能去掉,比如padding: 20px 0如果去掉了就错了),空格是无用的
customStyle = this.trim(customStyle)
// 根据";"将字符串转为数组形式
const styleArray = customStyle.split(';')
const style = {}
// 历遍数组,拼接成对象
for (let i = 0; i < styleArray.length; i++) {
// 'font-size:20px;color:red;',如此最后字符串有";"的话,会导致styleArray最后一个元素为空字符串,这里需要过滤
if (styleArray[i]) {
const item = styleArray[i].split(':')
style[this.trim(item[0])] = this.trim(item[1])
}
}
return style
}
// 这里为对象转字符串形式
let string = ''
for (const i in customStyle) {
// 驼峰转为中划线的形式,否则css内联样式,无法识别驼峰样式属性名
const key = i.replace(/([A-Z])/g, '-$1').toLowerCase()
string += `${key}:${customStyle[i]};`
}
// 去除两端空格
return this.trim(string)
},
/**
* 判断是否为空
*/
empty(value) {
switch (typeof value) {
case 'undefined':
return true
case 'string':
if (value.replace(/(^[ \t\n\r]*)|([ \t\n\r]*$)/g, '').length == 0) return true
break
case 'boolean':
if (!value) return true
break
case 'number':
if (value === 0 || isNaN(value)) return true
break
case 'object':
if (value === null || value.length === 0) return true
for (const i in value) {
return false
}
return true
}
return false
},
/**
* @description 去除空格
* @param str
* @param pos
*/
trim(str, pos = 'both') {
str = String(str)
if (pos == 'both') {
return str.replace(/^\s+|\s+$/g, '')
}
if (pos == 'left') {
return str.replace(/^\s*/, '')
}
if (pos == 'right') {
return str.replace(/(\s*$)/g, '')
}
if (pos == 'all') {
return str.replace(/\s+/g, '')
}
return str
},
/**
* @description 格式化时间
* @param {String|Number} dateTime 需要格式化的时间戳
* @param {String} fmt 格式化规则 yyyy:mm:dd|yyyy:mm|yyyy年mm月dd日|yyyy年mm月dd日 hh时MM分等,可自定义组合 默认yyyy-mm-dd
* @returns {string} 返回格式化后的字符串
*/
timeFormat(dateTime = null, formatStr = 'yyyy-mm-dd') {
let date
// 若传入时间为假值,则取当前时间
if (!dateTime) {
date = new Date()
}
// 若为unix秒时间戳,则转为毫秒时间戳(逻辑有点奇怪,但不敢改,以保证历史兼容)
else if (/^\d{10}$/.test(dateTime.toString().trim())) {
date = new Date(dateTime * 1000)
}
// 若用户传入字符串格式时间戳,new Date无法解析,需做兼容
else if (typeof dateTime === 'string' && /^\d+$/.test(dateTime.trim())) {
date = new Date(Number(dateTime))
}
// 其他都认为符合 RFC 2822 规范
else {
// 处理平台性差异,在Safari/Webkit中,new Date仅支持/作为分割符的字符串时间
date = new Date(
typeof dateTime === 'string'
? dateTime.replace(/-/g, '/')
: dateTime
)
}
const timeSource = {
'y': date.getFullYear().toString(), // 年
'm': (date.getMonth() + 1).toString().padStart(2, '0'), // 月
'd': date.getDate().toString().padStart(2, '0'), // 日
'h': date.getHours().toString().padStart(2, '0'), // 时
'M': date.getMinutes().toString().padStart(2, '0'), // 分
's': date.getSeconds().toString().padStart(2, '0') // 秒
// 有其他格式化字符需求可以继续添加,必须转化成字符串
}
for (const key in timeSource) {
const [ret] = new RegExp(`${key}+`).exec(formatStr) || []
if (ret) {
// 年可能只需展示两位
const beginIndex = key === 'y' && ret.length === 2 ? 2 : 0
formatStr = formatStr.replace(ret, timeSource[key].slice(beginIndex))
}
}
return formatStr
},
deepClone(data){
return JSON.parse(JSON.stringify(data));
},
downloadFile(url,name){
new Promise((resolve, reject) => {
let dTask = plus.downloader.createDownload(url, {
filename: 'file://storage/emulated/0/阅文系统/' + name
}, (d, status) => {
//d为下载的文件对象
if (status === 200) {
resolve('file://storage/emulated/0/阅文系统/' + name);
} else {
reject();
}
})
dTask.start();
})
},
getDirFile(name){
return new Promise((resolve, reject) => {
plus.io.requestFileSystem(plus.io.PRIVATE_DOC, function (fs) {
fs.root.getDirectory('阅文系统/' + name, {}, function (entry) {
entry.createReader().readEntries(function (entries) {
resolve(entries);
}, function (err) {
console.log('entries', err)
reject();
});
}, function (err) {
console.log('entry', err)
reject();
});
}, function (err) {
console.log('fs', err)
reject();
})
})
},
checkFile(name){
return new Promise((resolve, reject) => {
plus.io.requestFileSystem(plus.io.PRIVATE_DOC, function (fs) {
fs.root.getFile('阅文系统/' + name, {}, function (entry) {
console.log(entry)
resolve();
}, function (err) {
console.log('entry', err)
reject();
});
}, function (err) {
console.log('fs', err)
reject();
})
})
},
};

50
uni_modules/m-view/package.json

@ -1,18 +1,18 @@
{
"id": "m-view",
"displayName": "m-view 卡片",
"version": "1.0.1",
"description": "m-view",
"id": "mview",
"name": "mview",
"displayName": "mview",
"version": "3.1.36",
"description": "mview",
"keywords": [
"卡片"
"mview"
],
"repository": "",
"repository": "no",
"engines": {
"HBuilderX": "^3.1.0"
},
"dcloudext": {
"type": "component-vue",
"sale": {
"dcloudext": {
"sale": {
"regular": {
"price": "0.00"
},
@ -21,19 +21,18 @@
}
},
"contact": {
"qq": ""
"qq": "1416956117"
},
"declaration": {
"ads": "无",
"data": "插件不采集任何数据",
"data": "",
"permissions": "无"
},
"npmurl": ""
"npmurl": "no",
"type": "component-vue"
},
"uni_modules": {
"dependencies": [
],
"dependencies": [],
"encrypt": [],
"platforms": {
"cloud": {
@ -42,7 +41,7 @@
},
"client": {
"Vue": {
"vue2": "y",
"vue2": "n",
"vue3": "y"
},
"App": {
@ -67,17 +66,20 @@
"阿里": "y",
"百度": "y",
"字节跳动": "y",
"QQ": "y",
"钉钉": "u",
"快手": "u",
"飞书": "u",
"京东": "u"
"QQ": "y"
},
"快应用": {
"华为": "u",
"联盟": "u"
"华为": "y",
"联盟": "y"
}
}
}
},
"dependencies": {
"clipboard": "^2.0.11",
"dayjs": "^1.11.3"
},
"publishConfig": {
"registry": "no"
}
}
}

3
uni_modules/m-view/readme.md

@ -1,3 +0,0 @@
## 常用功能封装
> **组件名:m-view**

142
uni_modules/m-view/theme/common.scss

@ -0,0 +1,142 @@
view,scroll-view{
box-sizing:border-box;
}
.transition{
transition-duration: .2s;
transition-timing-function: ease-in-out;
transition-delay: .2s;
-webkit-transition-duration: .2s;
-webkit-transition-timing-function: ease-in-out;
-webkit-transition-delay: .2s;
}
.body-background{
position: fixed;
top:0;
left:0;
width:100%;
height:100%;
background:{
size:100% 100%;
position: center center;
repeat:no-repeat;
}
z-index: -1;
}
.border-box{
box-sizing: border-box;
}
.flex{
display:flex;
}
.flex-column{
flex-direction: column;
}
.row{
box-sizing: border-box;
width:100%;
display:flex;
flex-wrap: wrap;
}
.col-12{
box-sizing: border-box;
flex: 0 0 100%;
}
.col-8{
box-sizing: border-box;
flex: 0 0 66.66%;
}
.col-6{
box-sizing: border-box;
flex: 0 0 50%;
}
.col-4{
box-sizing: border-box;
flex: 0 0 33.33%;
}
.col-3{
box-sizing: border-box;
flex: 0 0 25%;
}
.col-2{
box-sizing: border-box;
flex: 0 0 16.667%;
}
.col-1{
box-sizing: border-box;
flex: 0 0 8.333%;
}
.col{
flex:1;
}
.col-auto{
flex:auto;
}
.align-center{
align-items: center;
}
.justify-center{
justify-content: center;
}
.justify-between{
justify-content: space-between;
}
.text-center{
text-align: center;
}
@for $i from 1 through 5 {
.mb-#{$i} {
margin-bottom: #{$i}0rpx;
}
.mr-#{$i} {
margin-right: #{$i}0rpx;
}
.ml-#{$i} {
margin-left: #{$i}0rpx;
}
.mt-#{$i} {
margin-top: #{$i}0rpx;
}
.pb-#{$i} {
padding-bottom: #{$i}0rpx;
}
.pr-#{$i} {
padding-right: #{$i}0rpx;
}
.pl-#{$i} {
padding-left: #{$i}0rpx;
}
.pt-#{$i} {
padding-top: #{$i}0rpx;
}
}
@for $i from 1 through 5 {
.m-line-#{$i} {
/* #ifdef APP-NVUE */
// nvue下可以直接使用lines属性这是weex特有样式
lines: $i;
text-overflow: ellipsis;
overflow: hidden;
flex: 1;
/* #endif */
/* #ifndef APP-NVUE */
// vue下单行和多行显示省略号需要单独处理
@if $i == '1' {
overflow: hidden;
white-space: nowrap;
text-overflow: ellipsis;
} @else {
display: -webkit-box!important;
overflow: hidden;
text-overflow: ellipsis;
word-break: break-all;
-webkit-line-clamp: $i;
-webkit-box-orient: vertical!important;
}
/* #endif */
}
}

6
uni_modules/m-view/theme/theme.scss

@ -0,0 +1,6 @@
// 此文件为主题变量这些变量目前只能通过uni.scss引入才有效另外由于
// uni.scss中引入的样式会同时混入到全局样式文件和单独每一个页面的样式中造成微信程序包太大
// 故uni.scss只建议放scss变量名相关样式其他的样式可以通过main.js或者App.vue引入
$m-checkbox-size: 32rpx !default;
$m-main-color: #3c9cff !default;
$m-assistant-color: #398ade !default;

2
utils/functions.js

@ -346,7 +346,7 @@ const api = {
uni.clearStorageSync();
store.commit('userInfo',{});
store.commit('userExtends',{});
api.uniSysInfo().then(() => {
uni.$m.uniSysInfo().then(() => {
if(reLaunch){
uni.reLaunch({url:'/pages/Index/index'});
}

Loading…
Cancel
Save