9 changed files with 2513 additions and 0 deletions
@ -0,0 +1,62 @@ |
|||||||
|
<template> |
||||||
|
<view class="PullDownBox"> |
||||||
|
<view @click="handleShowPullDown" class="Pulldown-title-container"> |
||||||
|
<!-- 标题显示内容 --> |
||||||
|
<view class="Pulldown-title"> |
||||||
|
<slot name="title"></slot> |
||||||
|
</view> |
||||||
|
<u-icon :class="{'icon': true,'normal':isShowPullDownBox, 'active': !isShowPullDownBox}" name="arrow-down" |
||||||
|
color="#2979ff" size="28"></u-icon> |
||||||
|
</view> |
||||||
|
|
||||||
|
<view class=""> |
||||||
|
<template v-if="isShowPullDownBox"> |
||||||
|
<slot name="content"></slot> |
||||||
|
</template> |
||||||
|
</view> |
||||||
|
</view> |
||||||
|
|
||||||
|
|
||||||
|
</template> |
||||||
|
|
||||||
|
<script setup lang="ts"> |
||||||
|
import { ref } from 'vue'; |
||||||
|
|
||||||
|
/** 是否显示下拉框 */ |
||||||
|
const isShowPullDownBox = ref(false) |
||||||
|
|
||||||
|
/** 是否显示下拉内容 */ |
||||||
|
const handleShowPullDown = () => { |
||||||
|
isShowPullDownBox.value = !isShowPullDownBox.value |
||||||
|
} |
||||||
|
</script> |
||||||
|
|
||||||
|
<style lang="scss" scoped> |
||||||
|
.PullDownBox { |
||||||
|
background: #fff; |
||||||
|
margin-bottom: 10upx; |
||||||
|
padding: 10upx; |
||||||
|
} |
||||||
|
|
||||||
|
.Pulldown-title-container { |
||||||
|
display: flex; |
||||||
|
} |
||||||
|
|
||||||
|
.Pulldown-title { |
||||||
|
flex: 1; |
||||||
|
} |
||||||
|
|
||||||
|
.icon { |
||||||
|
padding: 0 10upx; |
||||||
|
flex: none; |
||||||
|
transition: all 0.2s; |
||||||
|
} |
||||||
|
|
||||||
|
.normal { |
||||||
|
transform: rotate(0deg); |
||||||
|
} |
||||||
|
|
||||||
|
.active { |
||||||
|
transform: rotate(180deg); |
||||||
|
} |
||||||
|
</style> |
@ -0,0 +1,87 @@ |
|||||||
|
<template> |
||||||
|
<!-- @scrolltolower触底事件,@scroll滚动事件 --> |
||||||
|
<scroll-view scroll-y="true" :style="{height: containerHeight+'px'}" @scroll="scrollTop = $event.detail.scrollTop" |
||||||
|
@scrolltolower="$emit('@scrolltolower')"> |
||||||
|
<!-- 监听滚动事件使用passive修饰符 --> |
||||||
|
<view :style="paddingStyle"> |
||||||
|
<!-- key使用index,可避免多次渲染该dom --> |
||||||
|
<view id="box" v-for="(item, index) in showList" :key="index"> |
||||||
|
<!-- 使用作用域插槽,将遍历后的数据item和index传递出去 --> |
||||||
|
<slot :item="item" :$index="index"></slot> |
||||||
|
</view> |
||||||
|
</view> |
||||||
|
</scroll-view> |
||||||
|
</template> |
||||||
|
|
||||||
|
<script> |
||||||
|
export default { |
||||||
|
name: "App", |
||||||
|
props: { |
||||||
|
// 所有数据 |
||||||
|
allList: { |
||||||
|
type: Array, |
||||||
|
default () { |
||||||
|
return [] |
||||||
|
} |
||||||
|
}, |
||||||
|
// 容器高度 |
||||||
|
containerHeight: { |
||||||
|
type: Number, |
||||||
|
default: 0 |
||||||
|
}, |
||||||
|
}, |
||||||
|
data() { |
||||||
|
return { |
||||||
|
oneHeight: 200, // 单个元素的高度,默认200.在mounted中会再次回去单个元素高度 |
||||||
|
scrollTop: 0 // 滚动距离 |
||||||
|
}; |
||||||
|
}, |
||||||
|
async mounted() { |
||||||
|
// 计算单个元素的高度 |
||||||
|
await this.$nextTick(() => { |
||||||
|
|
||||||
|
}) |
||||||
|
|
||||||
|
// uniapp中获取高度的兼容写法 |
||||||
|
const query = uni.createSelectorQuery(); |
||||||
|
console.log('query.select :>> ', query.select('#box').boundingClientRect); |
||||||
|
query.select('#box').boundingClientRect(data => { |
||||||
|
console.log('data :>> ', data); |
||||||
|
this.oneHeight = data.height |
||||||
|
}).exec(); |
||||||
|
}, |
||||||
|
computed: { |
||||||
|
// 一屏多少条数据 |
||||||
|
showNum() { |
||||||
|
return ~~(this.containerHeight / this.oneHeight) + 2; |
||||||
|
}, |
||||||
|
// 开始索引 |
||||||
|
startIndex() { |
||||||
|
// 可见区域,第一个元素的index |
||||||
|
const curIndex = ~~(this.scrollTop / this.oneHeight) |
||||||
|
// console.log(this.showNum, this.oneHeight) |
||||||
|
// 渲染区域第一个元素的index,这里缓冲区域的列表条数使用的是this.showNum |
||||||
|
return curIndex < this.showNum ? 0 : curIndex - this.showNum |
||||||
|
}, |
||||||
|
// 渲染元素最后的index |
||||||
|
endIndex() { |
||||||
|
// 可见区域,第一个index |
||||||
|
const curIndex = ~~(this.scrollTop / this.oneHeight) |
||||||
|
let end = curIndex + this.showNum * 3; // 2倍是需要预留缓冲区域 |
||||||
|
let len = this.allList.length |
||||||
|
return end >= len ? len : end; // 结束元素大于所有元素的长度时,就取元素长度 |
||||||
|
}, |
||||||
|
// 需要渲染的数据 |
||||||
|
showList() { |
||||||
|
return this.allList.slice(this.startIndex, this.endIndex) |
||||||
|
}, |
||||||
|
// 空白占位的高度 |
||||||
|
paddingStyle() { |
||||||
|
return { |
||||||
|
paddingTop: this.startIndex * this.oneHeight + 'px', |
||||||
|
paddingBottom: (this.allList.length - this.endIndex) * this.oneHeight + 'px' |
||||||
|
} |
||||||
|
} |
||||||
|
}, |
||||||
|
}; |
||||||
|
</script> |
@ -0,0 +1,535 @@ |
|||||||
|
<template> |
||||||
|
<BasicContainer ref="basicContainer" :option="option"> |
||||||
|
<template #head> |
||||||
|
<image mode="widthFix" class="bgimg" src="/pagesHome/static/bgby.png"></image> |
||||||
|
<view class="mabxtop"> |
||||||
|
<view> |
||||||
|
<view> |
||||||
|
<view>{{(detauser?.trayName)||'暂无数据'}}</view> |
||||||
|
<view>托盘名称</view> |
||||||
|
</view> |
||||||
|
<view> |
||||||
|
<view>{{(detauser?.trayCode)||'暂无数据'}}</view> |
||||||
|
<view>托盘码</view> |
||||||
|
</view> |
||||||
|
</view> |
||||||
|
|
||||||
|
<view> |
||||||
|
<view> |
||||||
|
<view>{{(details.datelist.length)||0}}</view> |
||||||
|
<view>订单在托数量</view> |
||||||
|
</view> |
||||||
|
<view> |
||||||
|
<view>{{detauser?.trayTypeName || '暂无数据'}}</view> |
||||||
|
<view>打托方式</view> |
||||||
|
</view> |
||||||
|
</view> |
||||||
|
<view> |
||||||
|
|
||||||
|
<view> |
||||||
|
<view class="printBtn" @click.stop.prevent="handleGetSynchronousData">同步托盘</view> |
||||||
|
</view> |
||||||
|
</view> |
||||||
|
</view> |
||||||
|
</template> |
||||||
|
|
||||||
|
<template #body> |
||||||
|
<scroll-view class="scvmabx" :style="{height: details.scrollHeight}" scroll-y="true"> |
||||||
|
<uni-table v-show="!details.isShowSelect" border stripe emptyText="暂无更多数据" |
||||||
|
@selection-change="selectionChange"> |
||||||
|
<!-- 表头行 --> |
||||||
|
<uni-tr> |
||||||
|
<uni-th align="left">订单号</uni-th> |
||||||
|
<uni-th align="left">运单号</uni-th> |
||||||
|
<uni-th align="left">包条码</uni-th> |
||||||
|
<uni-th align="center">数量</uni-th> |
||||||
|
</uni-tr> |
||||||
|
<!-- 表格数据行 --> |
||||||
|
<block v-for="item in details.datelist"> |
||||||
|
<uni-tr> |
||||||
|
<!-- 订单号 --> |
||||||
|
<uni-td>{{item.orderCode}}</uni-td> |
||||||
|
<!-- 运单号 --> |
||||||
|
<uni-td>{{item.waybillNo}}</uni-td> |
||||||
|
<!-- 包条码 --> |
||||||
|
<uni-td>{{item.orderPackageCode}}</uni-td> |
||||||
|
<uni-td align="center">{{item.num}}</uni-td> |
||||||
|
</uni-tr> |
||||||
|
</block> |
||||||
|
</uni-table> |
||||||
|
|
||||||
|
<uni-table v-show="details.isShowSelect" ref="uniTable1" border stripe type="selection" emptyText="暂无更多数据" |
||||||
|
@selection-change="selectionChange"> |
||||||
|
<!-- 表头行 --> |
||||||
|
<uni-tr> |
||||||
|
<uni-th align="left">订单号</uni-th> |
||||||
|
<uni-th align="left">运单号</uni-th> |
||||||
|
<uni-th align="left">包条码</uni-th> |
||||||
|
<uni-th align="center">数量</uni-th> |
||||||
|
</uni-tr> |
||||||
|
<!-- 表格数据行 --> |
||||||
|
<block v-for="item in details.datelist"> |
||||||
|
<uni-tr> |
||||||
|
<!-- 订单号 --> |
||||||
|
<uni-td>{{item.orderCode}}</uni-td> |
||||||
|
<!-- 运单号 --> |
||||||
|
<uni-td>{{item.waybillNo}}</uni-td> |
||||||
|
<!-- 包条码 --> |
||||||
|
<uni-td>{{item.orderPackageCode}}</uni-td> |
||||||
|
<uni-td align="center">{{item.num}}</uni-td> |
||||||
|
</uni-tr> |
||||||
|
</block> |
||||||
|
|
||||||
|
</uni-table> |
||||||
|
|
||||||
|
|
||||||
|
<!-- 底部站位 --> |
||||||
|
<view class="footer"></view> |
||||||
|
<view class="submitBtn" @click="handleSubmit">提交同步</view> |
||||||
|
<!-- <button type="default" @click="moni">moni</button> --> |
||||||
|
</scroll-view> |
||||||
|
|
||||||
|
</template> |
||||||
|
</BasicContainer> |
||||||
|
<!-- #ifdef APP --> |
||||||
|
<saomiao2 :ishidestop="scanState !== 0"></saomiao2> |
||||||
|
<!-- #endif --> |
||||||
|
<tiplist ref="tiplists"></tiplist> |
||||||
|
</template> |
||||||
|
|
||||||
|
<script lang="ts" setup> |
||||||
|
import { |
||||||
|
postFindSyncTrayData, |
||||||
|
postFindSyncOldTrayData, |
||||||
|
postSyncOldTrayData |
||||||
|
} from '@/api/user.js' |
||||||
|
import { |
||||||
|
onLoad, |
||||||
|
onShow, |
||||||
|
} from '@dcloudio/uni-app' |
||||||
|
import utils from '@/utils/utils.js' |
||||||
|
import { nextTick, reactive, ref, toRefs } from "vue"; |
||||||
|
// 引入Pinia仓库 |
||||||
|
import useSystemSettingsStore from '@/store/useSystemSettingsStore'; |
||||||
|
import { storeToRefs } from 'pinia'; |
||||||
|
const { scanState } = storeToRefs(useSystemSettingsStore()) |
||||||
|
|
||||||
|
// 组件实例 |
||||||
|
const uniTable1 = ref(null) |
||||||
|
|
||||||
|
const option = reactive({ |
||||||
|
title: '托盘数据同步', |
||||||
|
haveData: true, |
||||||
|
async pullDownRefreshInitPage() { |
||||||
|
details.isShowSelect = false |
||||||
|
details.datelist = [] |
||||||
|
details.detauser = {} |
||||||
|
details.seletionList = [] |
||||||
|
return null |
||||||
|
} |
||||||
|
}) |
||||||
|
|
||||||
|
let details = reactive({ |
||||||
|
trayCode: '', |
||||||
|
scancode: '', |
||||||
|
datelist: [], |
||||||
|
seletionList: [], |
||||||
|
waybillCode: '', |
||||||
|
detauser: {} as any, |
||||||
|
/** 设置滚动容器的高度 */ |
||||||
|
scrollHeight: '', |
||||||
|
/** 是否显示复选框 */ |
||||||
|
isShowSelect: false |
||||||
|
}) |
||||||
|
|
||||||
|
|
||||||
|
const tiplists = ref(null) |
||||||
|
const basicContainer = ref(null) |
||||||
|
|
||||||
|
onLoad(() => { |
||||||
|
utils.ttsspke('请扫描托盘码') |
||||||
|
}) |
||||||
|
|
||||||
|
onShow(async () => { |
||||||
|
// #ifdef APP |
||||||
|
uni.$off('scancodedate') |
||||||
|
uni.$on('scancodedate', function (code) { |
||||||
|
if (code) { |
||||||
|
console.log(code); |
||||||
|
details.scancode = code |
||||||
|
|
||||||
|
scandata() |
||||||
|
} |
||||||
|
}) |
||||||
|
// #endif |
||||||
|
|
||||||
|
await nextTick() |
||||||
|
details.scrollHeight = await utils.getViewDistanceFormTop('.scvmabx') |
||||||
|
}) |
||||||
|
|
||||||
|
async function scandata() { |
||||||
|
// uni.showLoading({ |
||||||
|
// mask: true, |
||||||
|
// // duration: 30000 |
||||||
|
// }) |
||||||
|
try { |
||||||
|
// 表格不显示复选框 |
||||||
|
details.isShowSelect = false |
||||||
|
const res = await postFindSyncTrayData({ trayCode: details.scancode }) |
||||||
|
const { code, data } = res |
||||||
|
if (code !== 200 || !data) return |
||||||
|
details.detauser = data |
||||||
|
details.datelist = data.packageList ? data.packageList : [] |
||||||
|
|
||||||
|
} catch (err) { |
||||||
|
console.log('err :>> ', err); |
||||||
|
//TODO handle the exception |
||||||
|
} finally { |
||||||
|
// await nextTick() |
||||||
|
// uni.hideLoading() |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
const selectionChange = (list) => { |
||||||
|
details.seletionList = list.detail.index |
||||||
|
} |
||||||
|
|
||||||
|
/** 获取同步数据 */ |
||||||
|
const handleGetSynchronousData = async () => { |
||||||
|
if (!details.detauser.trayCode) { |
||||||
|
let content = '请先扫描托盘码' |
||||||
|
uni.showToast({ |
||||||
|
title: content, |
||||||
|
icon: "none" |
||||||
|
}) |
||||||
|
// #ifdef APP |
||||||
|
utils.ttsspke(content) |
||||||
|
// #endif |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
// uni.showLoading({ |
||||||
|
// mask: true |
||||||
|
// }) |
||||||
|
try { |
||||||
|
|
||||||
|
const res = await postFindSyncOldTrayData({ trayCode: details.detauser.trayCode }) |
||||||
|
const { code, data } = res |
||||||
|
if (code !== 200 || !data) return |
||||||
|
details.detauser.trayTypeName = data.trayTypeName |
||||||
|
details.datelist = data.packageList ? data.packageList : [] |
||||||
|
// 表格显示复选框 |
||||||
|
details.isShowSelect = true |
||||||
|
} catch (err) { |
||||||
|
console.log('err :>> ', err); |
||||||
|
//TODO handle the exception |
||||||
|
} finally { |
||||||
|
await nextTick() |
||||||
|
// 开启全选 |
||||||
|
uniTable1.value.selectionAll() |
||||||
|
console.log('111 :>> ', 111); |
||||||
|
// 关闭loading |
||||||
|
// uni.hideLoading() |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** 提交 */ |
||||||
|
const handleSubmit = async () => { |
||||||
|
let content = '' |
||||||
|
if (!details.isShowSelect) content = '请获取同步数据' |
||||||
|
else if (details.seletionList.length <= 0) content = '最少选择一条数据' |
||||||
|
if (content) { |
||||||
|
// #ifdef APP |
||||||
|
utils.ttsspke(content) |
||||||
|
// #endif |
||||||
|
uni.showToast({ |
||||||
|
title: content, |
||||||
|
icon: 'none' |
||||||
|
}) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
try { |
||||||
|
uni.showLoading({ |
||||||
|
mask: true |
||||||
|
}) |
||||||
|
|
||||||
|
console.log('details.seletionList :>> ', details.seletionList); |
||||||
|
const orderPackageCodes = [] |
||||||
|
details.datelist.forEach((val, index) => { |
||||||
|
if (details.seletionList.includes(index)) orderPackageCodes.push(val.orderPackageCode) |
||||||
|
}) |
||||||
|
const submitData = { |
||||||
|
trayCode: details.detauser.trayCode, |
||||||
|
trayType: details.detauser.trayType, |
||||||
|
orderPackageCodes, |
||||||
|
} |
||||||
|
const res = await postSyncOldTrayData(submitData) |
||||||
|
const { code } = res |
||||||
|
if (code !== 200) return |
||||||
|
basicContainer.value.startPullDownRefresh() |
||||||
|
} catch (err) { |
||||||
|
console.log('err :>> ', err); |
||||||
|
//TODO handle the exception |
||||||
|
} finally { |
||||||
|
uni.hideLoading() |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// function moni(){ |
||||||
|
// details.scancode = 'T2966' |
||||||
|
// scandata() |
||||||
|
// } |
||||||
|
const { datelist, detauser } = toRefs(details) |
||||||
|
</script> |
||||||
|
|
||||||
|
<style lang="scss" scoped> |
||||||
|
// 主题色 |
||||||
|
$subjectColor: #d3832a; |
||||||
|
|
||||||
|
.mabxtop { |
||||||
|
position: relative; |
||||||
|
width: 684upx; |
||||||
|
// height: 348upx; |
||||||
|
background: #FFFFFF; |
||||||
|
border-radius: 20upx; |
||||||
|
margin: auto; |
||||||
|
margin-top: 20upx; |
||||||
|
padding: 60upx; |
||||||
|
box-sizing: border-box; |
||||||
|
display: flex; |
||||||
|
flex-direction: column; |
||||||
|
align-items: center; |
||||||
|
justify-content: center; |
||||||
|
|
||||||
|
>view { |
||||||
|
width: 100%; |
||||||
|
display: flex; |
||||||
|
align-items: center; |
||||||
|
justify-content: space-between; |
||||||
|
margin-bottom: 10upx; |
||||||
|
|
||||||
|
&:nth-last-child(1) { |
||||||
|
margin-bottom: 0; |
||||||
|
} |
||||||
|
|
||||||
|
>view { |
||||||
|
&:nth-of-type(1) { |
||||||
|
width: 60%; |
||||||
|
} |
||||||
|
|
||||||
|
&:nth-of-type(2) { |
||||||
|
width: 40%; |
||||||
|
} |
||||||
|
|
||||||
|
display: flex; |
||||||
|
flex-direction: column; |
||||||
|
align-items: flex-start; |
||||||
|
|
||||||
|
>view:nth-of-type(1) { |
||||||
|
font-size: 32upx; |
||||||
|
font-weight: 400; |
||||||
|
color: #020B18; |
||||||
|
word-wrap: break-word; |
||||||
|
word-break: break-all; |
||||||
|
} |
||||||
|
|
||||||
|
>view:nth-of-type(2) { |
||||||
|
font-size: 28upx; |
||||||
|
font-weight: 400; |
||||||
|
color: #90A0AF; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
.printBtn { |
||||||
|
background: var(--subjectColor); |
||||||
|
color: #fff !important; |
||||||
|
// width: 100upx; |
||||||
|
padding: 10upx 20upx; |
||||||
|
text-align: center; |
||||||
|
border-radius: 5upx; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
.bgimg { |
||||||
|
position: fixed; |
||||||
|
left: 0; |
||||||
|
top: 0; |
||||||
|
width: 100%; |
||||||
|
} |
||||||
|
|
||||||
|
.scvmabx { |
||||||
|
width: 684rpx; |
||||||
|
height: 45vh; |
||||||
|
margin: auto; |
||||||
|
margin-top: 20upx; |
||||||
|
|
||||||
|
// 商场名称 |
||||||
|
.marketName { |
||||||
|
display: flex; |
||||||
|
margin-bottom: 20upx; |
||||||
|
align-items: center; |
||||||
|
} |
||||||
|
|
||||||
|
.maxboxs { |
||||||
|
width: 100%; |
||||||
|
display: flex; |
||||||
|
flex-direction: column; |
||||||
|
align-items: center; |
||||||
|
|
||||||
|
.chuanhuo { |
||||||
|
border: #ff0000; |
||||||
|
} |
||||||
|
|
||||||
|
.weiqt { |
||||||
|
position: absolute; |
||||||
|
right: 0; |
||||||
|
top: 0; |
||||||
|
width: 70upx; |
||||||
|
height: 70upx; |
||||||
|
} |
||||||
|
|
||||||
|
.qitao { |
||||||
|
position: absolute; |
||||||
|
right: 0; |
||||||
|
top: 0; |
||||||
|
width: 70upx; |
||||||
|
height: 70upx; |
||||||
|
} |
||||||
|
|
||||||
|
>.items { |
||||||
|
width: 690upx; |
||||||
|
display: flex; |
||||||
|
flex-direction: column; |
||||||
|
align-items: center; |
||||||
|
padding: 20upx 30upx; |
||||||
|
box-sizing: border-box; |
||||||
|
background-color: #ffffff; |
||||||
|
border-radius: 10upx; |
||||||
|
margin-bottom: 30upx; |
||||||
|
position: relative; |
||||||
|
|
||||||
|
&:nth-last-child(1) { |
||||||
|
margin-bottom: 20upx; |
||||||
|
} |
||||||
|
|
||||||
|
.tophd { |
||||||
|
width: 100%; |
||||||
|
display: flex; |
||||||
|
align-items: flex-start; |
||||||
|
justify-content: flex-start; |
||||||
|
padding: 15upx 20upx; |
||||||
|
box-sizing: border-box; |
||||||
|
padding-top: 30upx; |
||||||
|
border-bottom: 4upx solid #EEEEEE; |
||||||
|
|
||||||
|
>image { |
||||||
|
width: 48upx; |
||||||
|
height: 48upx; |
||||||
|
margin-right: 10upx; |
||||||
|
box-sizing: border-box; |
||||||
|
} |
||||||
|
|
||||||
|
>view { |
||||||
|
font-size: 32upx; |
||||||
|
color: #020B18; |
||||||
|
font-weight: 400; |
||||||
|
word-wrap: break-word; |
||||||
|
word-break: break-all; |
||||||
|
|
||||||
|
&:nth-of-type(1) { |
||||||
|
width: 180upx; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
>view:nth-of-type(2) { |
||||||
|
flex: 1; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
.tophd1 { |
||||||
|
width: 100%; |
||||||
|
display: flex; |
||||||
|
box-sizing: border-box; |
||||||
|
padding: 15upx 20upx; |
||||||
|
border-bottom: 4upx solid #EEEEEE; |
||||||
|
|
||||||
|
&>view { |
||||||
|
display: flex; |
||||||
|
flex: 1; |
||||||
|
align-items: center; |
||||||
|
} |
||||||
|
|
||||||
|
image { |
||||||
|
width: 64upx; |
||||||
|
height: 64upx; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
.numbxvie { |
||||||
|
width: 100%; |
||||||
|
display: flex; |
||||||
|
align-items: center; |
||||||
|
justify-content: space-around; |
||||||
|
padding: 15upx 25upx; |
||||||
|
box-sizing: border-box; |
||||||
|
|
||||||
|
.zhon { |
||||||
|
color: $subjectColor; |
||||||
|
} |
||||||
|
|
||||||
|
.yish { |
||||||
|
color: #1197EB; |
||||||
|
} |
||||||
|
|
||||||
|
.weish { |
||||||
|
color: #020B18; |
||||||
|
} |
||||||
|
|
||||||
|
>view { |
||||||
|
display: flex; |
||||||
|
flex-direction: column; |
||||||
|
align-items: center; |
||||||
|
|
||||||
|
>view:nth-of-type(1) { |
||||||
|
font-size: 36upx; |
||||||
|
margin-bottom: 12upx; |
||||||
|
} |
||||||
|
|
||||||
|
>view:nth-of-type(2) { |
||||||
|
font-size: 28upx; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// 底部站位 |
||||||
|
.footer { |
||||||
|
height: 150upx; |
||||||
|
} |
||||||
|
|
||||||
|
// 底部提交按钮 |
||||||
|
.submitBtn { |
||||||
|
width: fit-content; |
||||||
|
position: fixed; |
||||||
|
bottom: 20upx; |
||||||
|
background: var(--subjectColor); |
||||||
|
color: #fff; |
||||||
|
padding: 20upx 50upx; |
||||||
|
border-radius: 5upx; |
||||||
|
left: 50%; |
||||||
|
transform: translateX(-50%); |
||||||
|
} |
||||||
|
|
||||||
|
.uni-table-scroll { |
||||||
|
zoom: 0.9; |
||||||
|
} |
||||||
|
|
||||||
|
.uni-table-td, |
||||||
|
.uni-table-th { |
||||||
|
padding: 10upx; |
||||||
|
} |
||||||
|
</style> |
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,35 @@ |
|||||||
|
import { |
||||||
|
defineStore |
||||||
|
} from 'pinia'; |
||||||
|
import { |
||||||
|
ref |
||||||
|
} from 'vue'; |
||||||
|
|
||||||
|
const useSearchInfoStore = defineStore('useSearchInfoStore', () => { |
||||||
|
/** 搜索的值 */ |
||||||
|
const searchInfo = ref({}) |
||||||
|
|
||||||
|
/** 新增需要缓存的搜索值 */ |
||||||
|
const HANDLE_EDIT_INFO = (pageName, info) => { |
||||||
|
searchInfo.value[pageName] = info |
||||||
|
} |
||||||
|
|
||||||
|
/** 删除需要缓存的搜索值 */ |
||||||
|
const HANDLE_DELETE = (pageName) => { |
||||||
|
delete searchInfo.value[pageName] |
||||||
|
} |
||||||
|
|
||||||
|
/** 清空搜索框的值 */ |
||||||
|
const HANDLE_CLEAR = () => { |
||||||
|
searchInfo.value = {} |
||||||
|
} |
||||||
|
|
||||||
|
return { |
||||||
|
searchInfo, |
||||||
|
HANDLE_EDIT_INFO, |
||||||
|
HANDLE_DELETE, |
||||||
|
HANDLE_CLEAR |
||||||
|
} |
||||||
|
}) |
||||||
|
|
||||||
|
export default useSearchInfoStore |
@ -0,0 +1,60 @@ |
|||||||
|
import { |
||||||
|
defineStore |
||||||
|
} from 'pinia'; |
||||||
|
|
||||||
|
import { |
||||||
|
ref |
||||||
|
} from 'vue'; |
||||||
|
|
||||||
|
import { |
||||||
|
postFindAllAllocationId |
||||||
|
} from '@/api/user.js'; |
||||||
|
|
||||||
|
const useStorageStore = defineStore('useStorageStore', () => { |
||||||
|
/** 存储本仓库位信息 */ |
||||||
|
const storageArr = ref([]) |
||||||
|
|
||||||
|
/** 请求本仓库位信息 */ |
||||||
|
const HANDLE_GETSTORAGEINFO = async () => { |
||||||
|
const response = await postFindAllAllocationId() |
||||||
|
|
||||||
|
const { |
||||||
|
code, |
||||||
|
data |
||||||
|
} = response |
||||||
|
|
||||||
|
if (code !== 200) return |
||||||
|
storageArr.value = data |
||||||
|
} |
||||||
|
|
||||||
|
/** 清空库位信息 */ |
||||||
|
const HANDLE_CLEARSTORAGE = () => { |
||||||
|
storageArr.value.splice(0) |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 检测是否为库位码 |
||||||
|
* @param {string} code 扫描的code码 |
||||||
|
* @return {boolean} 是否为库位码的判断 |
||||||
|
* */ |
||||||
|
const HANDLE_ISSTORAGECODE = (code) => { |
||||||
|
let _flag = false |
||||||
|
for (let item of storageArr.value) { |
||||||
|
if ((item + '') === (code + '')) { |
||||||
|
_flag = true |
||||||
|
// 退出循环
|
||||||
|
break |
||||||
|
} |
||||||
|
} |
||||||
|
return _flag |
||||||
|
} |
||||||
|
|
||||||
|
return { |
||||||
|
storageArr, |
||||||
|
HANDLE_GETSTORAGEINFO, |
||||||
|
HANDLE_CLEARSTORAGE, |
||||||
|
HANDLE_ISSTORAGECODE |
||||||
|
} |
||||||
|
}) |
||||||
|
|
||||||
|
export default useStorageStore |
@ -0,0 +1,313 @@ |
|||||||
|
//
|
||||||
|
// THIS FILE IS AUTOMATICALLY GENERATED! DO NOT EDIT BY HAND!
|
||||||
|
//
|
||||||
|
;(function(global, factory) { |
||||||
|
typeof exports === 'object' && typeof module !== 'undefined' |
||||||
|
? module.exports = factory() |
||||||
|
: typeof define === 'function' && define.amd |
||||||
|
? define(factory) : |
||||||
|
// cf. https://github.com/dankogai/js-base64/issues/119
|
||||||
|
(function() { |
||||||
|
// existing version for noConflict()
|
||||||
|
const _Base64 = global.Base64; |
||||||
|
const gBase64 = factory(); |
||||||
|
gBase64.noConflict = () => { |
||||||
|
global.Base64 = _Base64; |
||||||
|
return gBase64; |
||||||
|
}; |
||||||
|
if (global.Meteor) { // Meteor.js
|
||||||
|
Base64 = gBase64; |
||||||
|
} |
||||||
|
global.Base64 = gBase64; |
||||||
|
})(); |
||||||
|
}((typeof self !== 'undefined' ? self |
||||||
|
: typeof window !== 'undefined' ? window |
||||||
|
: typeof global !== 'undefined' ? global |
||||||
|
: this |
||||||
|
), function() { |
||||||
|
'use strict'; |
||||||
|
|
||||||
|
/** |
||||||
|
* base64.ts |
||||||
|
* |
||||||
|
* Licensed under the BSD 3-Clause License. |
||||||
|
* http://opensource.org/licenses/BSD-3-Clause
|
||||||
|
* |
||||||
|
* References: |
||||||
|
* http://en.wikipedia.org/wiki/Base64
|
||||||
|
* |
||||||
|
* @author Dan Kogai (https://github.com/dankogai)
|
||||||
|
*/ |
||||||
|
const version = '3.6.0'; |
||||||
|
/** |
||||||
|
* @deprecated use lowercase `version`. |
||||||
|
*/ |
||||||
|
const VERSION = version; |
||||||
|
const _hasatob = typeof atob === 'function'; |
||||||
|
const _hasbtoa = typeof btoa === 'function'; |
||||||
|
const _hasBuffer = typeof Buffer === 'function'; |
||||||
|
const _TD = typeof TextDecoder === 'function' ? new TextDecoder() : undefined; |
||||||
|
const _TE = typeof TextEncoder === 'function' ? new TextEncoder() : undefined; |
||||||
|
const b64ch = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='; |
||||||
|
const b64chs = [...b64ch]; |
||||||
|
const b64tab = ((a) => { |
||||||
|
let tab = {}; |
||||||
|
a.forEach((c, i) => tab[c] = i); |
||||||
|
return tab; |
||||||
|
})(b64chs); |
||||||
|
const b64re = /^(?:[A-Za-z\d+\/]{4})*?(?:[A-Za-z\d+\/]{2}(?:==)?|[A-Za-z\d+\/]{3}=?)?$/; |
||||||
|
const _fromCC = String.fromCharCode.bind(String); |
||||||
|
const _U8Afrom = typeof Uint8Array.from === 'function' |
||||||
|
? Uint8Array.from.bind(Uint8Array) |
||||||
|
: (it, fn = (x) => x) => new Uint8Array(Array.prototype.slice.call(it, 0).map(fn)); |
||||||
|
const _mkUriSafe = (src) => src |
||||||
|
.replace(/[+\/]/g, (m0) => m0 == '+' ? '-' : '_') |
||||||
|
.replace(/=+$/m, ''); |
||||||
|
const _tidyB64 = (s) => s.replace(/[^A-Za-z0-9\+\/]/g, ''); |
||||||
|
/** |
||||||
|
* polyfill version of `btoa` |
||||||
|
*/ |
||||||
|
const btoaPolyfill = (bin) => { |
||||||
|
// console.log('polyfilled');
|
||||||
|
let u32, c0, c1, c2, asc = ''; |
||||||
|
const pad = bin.length % 3; |
||||||
|
for (let i = 0; i < bin.length;) { |
||||||
|
if ((c0 = bin.charCodeAt(i++)) > 255 || |
||||||
|
(c1 = bin.charCodeAt(i++)) > 255 || |
||||||
|
(c2 = bin.charCodeAt(i++)) > 255) |
||||||
|
throw new TypeError('invalid character found'); |
||||||
|
u32 = (c0 << 16) | (c1 << 8) | c2; |
||||||
|
asc += b64chs[u32 >> 18 & 63] |
||||||
|
+ b64chs[u32 >> 12 & 63] |
||||||
|
+ b64chs[u32 >> 6 & 63] |
||||||
|
+ b64chs[u32 & 63]; |
||||||
|
} |
||||||
|
return pad ? asc.slice(0, pad - 3) + "===".substring(pad) : asc; |
||||||
|
}; |
||||||
|
/** |
||||||
|
* does what `window.btoa` of web browsers do. |
||||||
|
* @param {String} bin binary string |
||||||
|
* @returns {string} Base64-encoded string |
||||||
|
*/ |
||||||
|
const _btoa = _hasbtoa ? (bin) => btoa(bin) |
||||||
|
: _hasBuffer ? (bin) => Buffer.from(bin, 'binary').toString('base64') |
||||||
|
: btoaPolyfill; |
||||||
|
const _fromUint8Array = _hasBuffer |
||||||
|
? (u8a) => Buffer.from(u8a).toString('base64') |
||||||
|
: (u8a) => { |
||||||
|
// cf. https://stackoverflow.com/questions/12710001/how-to-convert-uint8-array-to-base64-encoded-string/12713326#12713326
|
||||||
|
const maxargs = 0x1000; |
||||||
|
let strs = []; |
||||||
|
for (let i = 0, l = u8a.length; i < l; i += maxargs) { |
||||||
|
strs.push(_fromCC.apply(null, u8a.subarray(i, i + maxargs))); |
||||||
|
} |
||||||
|
return _btoa(strs.join('')); |
||||||
|
}; |
||||||
|
/** |
||||||
|
* converts a Uint8Array to a Base64 string. |
||||||
|
* @param {boolean} [urlsafe] URL-and-filename-safe a la RFC4648 §5 |
||||||
|
* @returns {string} Base64 string |
||||||
|
*/ |
||||||
|
const fromUint8Array = (u8a, urlsafe = false) => urlsafe ? _mkUriSafe(_fromUint8Array(u8a)) : _fromUint8Array(u8a); |
||||||
|
// This trick is found broken https://github.com/dankogai/js-base64/issues/130
|
||||||
|
// const utob = (src: string) => unescape(encodeURIComponent(src));
|
||||||
|
// reverting good old fationed regexp
|
||||||
|
const cb_utob = (c) => { |
||||||
|
if (c.length < 2) { |
||||||
|
var cc = c.charCodeAt(0); |
||||||
|
return cc < 0x80 ? c |
||||||
|
: cc < 0x800 ? (_fromCC(0xc0 | (cc >>> 6)) |
||||||
|
+ _fromCC(0x80 | (cc & 0x3f))) |
||||||
|
: (_fromCC(0xe0 | ((cc >>> 12) & 0x0f)) |
||||||
|
+ _fromCC(0x80 | ((cc >>> 6) & 0x3f)) |
||||||
|
+ _fromCC(0x80 | (cc & 0x3f))); |
||||||
|
} |
||||||
|
else { |
||||||
|
var cc = 0x10000 |
||||||
|
+ (c.charCodeAt(0) - 0xD800) * 0x400 |
||||||
|
+ (c.charCodeAt(1) - 0xDC00); |
||||||
|
return (_fromCC(0xf0 | ((cc >>> 18) & 0x07)) |
||||||
|
+ _fromCC(0x80 | ((cc >>> 12) & 0x3f)) |
||||||
|
+ _fromCC(0x80 | ((cc >>> 6) & 0x3f)) |
||||||
|
+ _fromCC(0x80 | (cc & 0x3f))); |
||||||
|
} |
||||||
|
}; |
||||||
|
const re_utob = /[\uD800-\uDBFF][\uDC00-\uDFFFF]|[^\x00-\x7F]/g; |
||||||
|
/** |
||||||
|
* @deprecated should have been internal use only. |
||||||
|
* @param {string} src UTF-8 string |
||||||
|
* @returns {string} UTF-16 string |
||||||
|
*/ |
||||||
|
const utob = (u) => u.replace(re_utob, cb_utob); |
||||||
|
//
|
||||||
|
const _encode = _hasBuffer |
||||||
|
? (s) => Buffer.from(s, 'utf8').toString('base64') |
||||||
|
: _TE |
||||||
|
? (s) => _fromUint8Array(_TE.encode(s)) |
||||||
|
: (s) => _btoa(utob(s)); |
||||||
|
/** |
||||||
|
* converts a UTF-8-encoded string to a Base64 string. |
||||||
|
* @param {boolean} [urlsafe] if `true` make the result URL-safe |
||||||
|
* @returns {string} Base64 string |
||||||
|
*/ |
||||||
|
const encode = (src, urlsafe = false) => urlsafe |
||||||
|
? _mkUriSafe(_encode(src)) |
||||||
|
: _encode(src); |
||||||
|
/** |
||||||
|
* converts a UTF-8-encoded string to URL-safe Base64 RFC4648 §5. |
||||||
|
* @returns {string} Base64 string |
||||||
|
*/ |
||||||
|
const encodeURI = (src) => encode(src, true); |
||||||
|
// This trick is found broken https://github.com/dankogai/js-base64/issues/130
|
||||||
|
// const btou = (src: string) => decodeURIComponent(escape(src));
|
||||||
|
// reverting good old fationed regexp
|
||||||
|
const re_btou = /[\xC0-\xDF][\x80-\xBF]|[\xE0-\xEF][\x80-\xBF]{2}|[\xF0-\xF7][\x80-\xBF]{3}/g; |
||||||
|
const cb_btou = (cccc) => { |
||||||
|
switch (cccc.length) { |
||||||
|
case 4: |
||||||
|
var cp = ((0x07 & cccc.charCodeAt(0)) << 18) |
||||||
|
| ((0x3f & cccc.charCodeAt(1)) << 12) |
||||||
|
| ((0x3f & cccc.charCodeAt(2)) << 6) |
||||||
|
| (0x3f & cccc.charCodeAt(3)), offset = cp - 0x10000; |
||||||
|
return (_fromCC((offset >>> 10) + 0xD800) |
||||||
|
+ _fromCC((offset & 0x3FF) + 0xDC00)); |
||||||
|
case 3: |
||||||
|
return _fromCC(((0x0f & cccc.charCodeAt(0)) << 12) |
||||||
|
| ((0x3f & cccc.charCodeAt(1)) << 6) |
||||||
|
| (0x3f & cccc.charCodeAt(2))); |
||||||
|
default: |
||||||
|
return _fromCC(((0x1f & cccc.charCodeAt(0)) << 6) |
||||||
|
| (0x3f & cccc.charCodeAt(1))); |
||||||
|
} |
||||||
|
}; |
||||||
|
/** |
||||||
|
* @deprecated should have been internal use only. |
||||||
|
* @param {string} src UTF-16 string |
||||||
|
* @returns {string} UTF-8 string |
||||||
|
*/ |
||||||
|
const btou = (b) => b.replace(re_btou, cb_btou); |
||||||
|
/** |
||||||
|
* polyfill version of `atob` |
||||||
|
*/ |
||||||
|
const atobPolyfill = (asc) => { |
||||||
|
// console.log('polyfilled');
|
||||||
|
asc = asc.replace(/\s+/g, ''); |
||||||
|
if (!b64re.test(asc)) |
||||||
|
throw new TypeError('malformed base64.'); |
||||||
|
asc += '=='.slice(2 - (asc.length & 3)); |
||||||
|
let u24, bin = '', r1, r2; |
||||||
|
for (let i = 0; i < asc.length;) { |
||||||
|
u24 = b64tab[asc.charAt(i++)] << 18 |
||||||
|
| b64tab[asc.charAt(i++)] << 12 |
||||||
|
| (r1 = b64tab[asc.charAt(i++)]) << 6 |
||||||
|
| (r2 = b64tab[asc.charAt(i++)]); |
||||||
|
bin += r1 === 64 ? _fromCC(u24 >> 16 & 255) |
||||||
|
: r2 === 64 ? _fromCC(u24 >> 16 & 255, u24 >> 8 & 255) |
||||||
|
: _fromCC(u24 >> 16 & 255, u24 >> 8 & 255, u24 & 255); |
||||||
|
} |
||||||
|
return bin; |
||||||
|
}; |
||||||
|
/** |
||||||
|
* does what `window.atob` of web browsers do. |
||||||
|
* @param {String} asc Base64-encoded string |
||||||
|
* @returns {string} binary string |
||||||
|
*/ |
||||||
|
const _atob = _hasatob ? (asc) => atob(_tidyB64(asc)) |
||||||
|
: _hasBuffer ? (asc) => Buffer.from(asc, 'base64').toString('binary') |
||||||
|
: atobPolyfill; |
||||||
|
//
|
||||||
|
const _toUint8Array = _hasBuffer |
||||||
|
? (a) => _U8Afrom(Buffer.from(a, 'base64')) |
||||||
|
: (a) => _U8Afrom(_atob(a), c => c.charCodeAt(0)); |
||||||
|
/** |
||||||
|
* converts a Base64 string to a Uint8Array. |
||||||
|
*/ |
||||||
|
const toUint8Array = (a) => _toUint8Array(_unURI(a)); |
||||||
|
//
|
||||||
|
const _decode = _hasBuffer |
||||||
|
? (a) => Buffer.from(a, 'base64').toString('utf8') |
||||||
|
: _TD |
||||||
|
? (a) => _TD.decode(_toUint8Array(a)) |
||||||
|
: (a) => btou(_atob(a)); |
||||||
|
const _unURI = (a) => _tidyB64(a.replace(/[-_]/g, (m0) => m0 == '-' ? '+' : '/')); |
||||||
|
/** |
||||||
|
* converts a Base64 string to a UTF-8 string. |
||||||
|
* @param {String} src Base64 string. Both normal and URL-safe are supported |
||||||
|
* @returns {string} UTF-8 string |
||||||
|
*/ |
||||||
|
const decode = (src) => _decode(_unURI(src)); |
||||||
|
/** |
||||||
|
* check if a value is a valid Base64 string |
||||||
|
* @param {String} src a value to check |
||||||
|
*/ |
||||||
|
const isValid = (src) => { |
||||||
|
if (typeof src !== 'string') |
||||||
|
return false; |
||||||
|
const s = src.replace(/\s+/g, '').replace(/=+$/, ''); |
||||||
|
return !/[^\s0-9a-zA-Z\+/]/.test(s) || !/[^\s0-9a-zA-Z\-_]/.test(s); |
||||||
|
}; |
||||||
|
//
|
||||||
|
const _noEnum = (v) => { |
||||||
|
return { |
||||||
|
value: v, enumerable: false, writable: true, configurable: true |
||||||
|
}; |
||||||
|
}; |
||||||
|
/** |
||||||
|
* extend String.prototype with relevant methods |
||||||
|
*/ |
||||||
|
const extendString = function () { |
||||||
|
const _add = (name, body) => Object.defineProperty(String.prototype, name, _noEnum(body)); |
||||||
|
_add('fromBase64', function () { return decode(this); }); |
||||||
|
_add('toBase64', function (urlsafe) { return encode(this, urlsafe); }); |
||||||
|
_add('toBase64URI', function () { return encode(this, true); }); |
||||||
|
_add('toBase64URL', function () { return encode(this, true); }); |
||||||
|
_add('toUint8Array', function () { return toUint8Array(this); }); |
||||||
|
}; |
||||||
|
/** |
||||||
|
* extend Uint8Array.prototype with relevant methods |
||||||
|
*/ |
||||||
|
const extendUint8Array = function () { |
||||||
|
const _add = (name, body) => Object.defineProperty(Uint8Array.prototype, name, _noEnum(body)); |
||||||
|
_add('toBase64', function (urlsafe) { return fromUint8Array(this, urlsafe); }); |
||||||
|
_add('toBase64URI', function () { return fromUint8Array(this, true); }); |
||||||
|
_add('toBase64URL', function () { return fromUint8Array(this, true); }); |
||||||
|
}; |
||||||
|
/** |
||||||
|
* extend Builtin prototypes with relevant methods |
||||||
|
*/ |
||||||
|
const extendBuiltins = () => { |
||||||
|
extendString(); |
||||||
|
extendUint8Array(); |
||||||
|
}; |
||||||
|
const gBase64 = { |
||||||
|
version: version, |
||||||
|
VERSION: VERSION, |
||||||
|
atob: _atob, |
||||||
|
atobPolyfill: atobPolyfill, |
||||||
|
btoa: _btoa, |
||||||
|
btoaPolyfill: btoaPolyfill, |
||||||
|
fromBase64: decode, |
||||||
|
toBase64: encode, |
||||||
|
encode: encode, |
||||||
|
encodeURI: encodeURI, |
||||||
|
encodeURL: encodeURI, |
||||||
|
utob: utob, |
||||||
|
btou: btou, |
||||||
|
decode: decode, |
||||||
|
isValid: isValid, |
||||||
|
fromUint8Array: fromUint8Array, |
||||||
|
toUint8Array: toUint8Array, |
||||||
|
extendString: extendString, |
||||||
|
extendUint8Array: extendUint8Array, |
||||||
|
extendBuiltins: extendBuiltins, |
||||||
|
}; |
||||||
|
|
||||||
|
//
|
||||||
|
// export Base64 to the namespace
|
||||||
|
//
|
||||||
|
// ES5 is yet to have Object.assign() that may make transpilers unhappy.
|
||||||
|
// gBase64.Base64 = Object.assign({}, gBase64);
|
||||||
|
gBase64.Base64 = {}; |
||||||
|
Object.keys(gBase64).forEach(k => gBase64.Base64[k] = gBase64[k]); |
||||||
|
return gBase64; |
||||||
|
})); |
@ -0,0 +1,170 @@ |
|||||||
|
/* |
||||||
|
使用该类应开启以下权限(读取和写入) |
||||||
|
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> |
||||||
|
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> |
||||||
|
*/ |
||||||
|
/* |
||||||
|
文件编码 |
||||||
|
ansi(ascii):英文标准码0-127,包括控制符和其他英文字符编码,这在后来一直统一没再变过,当然它只需要一个字节保存 |
||||||
|
gb2312和gbk:这两个是ancii码加上汉字的扩展,汉字多达10万,在ancii编码基础上再加一个字节表示汉字,共可表示字符65535个,包括了繁体字。因此一个中文字符包含两个字节。eclipse中默认编码方式为gbk。在Windows中文系统中ANSI是默认的编码方式。对于英文文件是ASCII编码,对于简体中文文件是GB2312编码。 |
||||||
|
Unicode编码:顾名思义,这是国际统一标准编码,在这之前各国标准编码不统一,微软等公司需要为各国的计算机系统定制符合不同编码标准的系统,显然,成本太高,并且互联网的出现让不同编码标准的计算机交互变得困难,如:两国的邮件系统,会因为使用不同的编码标准而导致接受方收到的邮件是乱码。 |
||||||
|
utf-8和utf-16编码:UTF的意思是(UCS Transfer Format),显然是随着互联网的出现,需要解决Unicode在网络上的传输问题。顾名思义,UTF8就是每次8个位传输数据,而UTF16就是每次16个位,只不过为了传输时的可靠性,从UNICODE到UTF时并不是直接的对应,而是要过一些算法和规则来转换。UTF-8就是在互联网上使用最广的一种unicode的实现方式。 |
||||||
|
*/ |
||||||
|
|
||||||
|
/** |
||||||
|
* 获取手机内置存储的根路径 |
||||||
|
* @return {String} |
||||||
|
*/ |
||||||
|
const root = function() { |
||||||
|
const environment = plus.android.importClass("android.os.Environment"); |
||||||
|
return environment.getExternalStorageDirectory(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 获取指定文件夹下的所有文件和文件夹列表 |
||||||
|
* @param {String} path 文件夹路径 |
||||||
|
* @return {Array<String>} 文件和文件夹列表 |
||||||
|
*/ |
||||||
|
const filelist = function(dir = '') { |
||||||
|
const File = plus.android.importClass("java.io.File"); |
||||||
|
let list = []; |
||||||
|
let file = new File(dir); |
||||||
|
let tempList = file.listFiles(); |
||||||
|
for (let i = 0; i < tempList.length; i++) { |
||||||
|
let fileName = tempList[i].getName(); |
||||||
|
list.push(fileName); |
||||||
|
} |
||||||
|
return list; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 创建文件 |
||||||
|
* @return {boolean} flase=失败(已存在、操作失败),true=成功 |
||||||
|
*/ |
||||||
|
const createNewFile = function(path = '') { |
||||||
|
const File = plus.android.importClass('java.io.File'); |
||||||
|
let file = new File(path); |
||||||
|
if (!file.exists()) { |
||||||
|
return file.createNewFile(); |
||||||
|
} |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 创建文件夹 |
||||||
|
* @return {boolean} flase=失败(已存在、操作失败),true=成功 |
||||||
|
*/ |
||||||
|
const mkdirs = function(path = ''){ |
||||||
|
const File = plus.android.importClass('java.io.File'); |
||||||
|
let file = new File(path); |
||||||
|
if (!file.exists()) { |
||||||
|
return file.mkdirs(); |
||||||
|
} |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 读取文件 |
||||||
|
* @param {String} path 文件路径 |
||||||
|
* @param {String} charset 编码 |
||||||
|
* @return {Array<String>} 内容列表(按行读取),文件不存在或异常则返回false |
||||||
|
*/ |
||||||
|
const readTxt = function(path = '', charset = 'utf-8') { |
||||||
|
const File = plus.android.importClass('java.io.File'); |
||||||
|
const InputStreamReader = plus.android.importClass('java.io.InputStreamReader'); |
||||||
|
const BufferedReader = plus.android.importClass('java.io.BufferedReader'); |
||||||
|
const FileInputStream = plus.android.importClass('java.io.FileInputStream'); |
||||||
|
let file = new File(path); |
||||||
|
let inputStreamReader = null; |
||||||
|
let bufferedReader = null; |
||||||
|
let list = []; |
||||||
|
try { |
||||||
|
if (!file.exists()) { |
||||||
|
return false; |
||||||
|
} |
||||||
|
inputStreamReader = new InputStreamReader(new FileInputStream(file), charset); |
||||||
|
bufferedReader = new BufferedReader(inputStreamReader); |
||||||
|
let line = ''; |
||||||
|
while (null != (line = bufferedReader.readLine())) { |
||||||
|
list.push(line); |
||||||
|
} |
||||||
|
bufferedReader.close(); |
||||||
|
inputStreamReader.close(); |
||||||
|
} catch (e) { |
||||||
|
if (null != bufferedReader) { |
||||||
|
bufferedReader.close(); |
||||||
|
} |
||||||
|
if (null != inputStreamReader) { |
||||||
|
inputStreamReader.close(); |
||||||
|
} |
||||||
|
return false; |
||||||
|
} |
||||||
|
return list; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 写入文件内容 |
||||||
|
* @param {String} path 文件路径 |
||||||
|
* @param {String} content 内容 |
||||||
|
* @param {boolean} append 内容写入类型,false=不追加(覆盖原有内容),true=追加(从内容尾部写入) |
||||||
|
* @param {String} charset 编码 |
||||||
|
* @return {boolean} true=成功,false=失败 |
||||||
|
*/ |
||||||
|
const writeTxt = function(path = '', content = '', append = false, charset = 'utf-8') { |
||||||
|
const File = plus.android.importClass('java.io.File'); |
||||||
|
const FileOutputStream = plus.android.importClass('java.io.FileOutputStream'); |
||||||
|
const OutputStreamWriter = plus.android.importClass('java.io.OutputStreamWriter'); |
||||||
|
|
||||||
|
let outputStreamWriter; |
||||||
|
let file = new File(path); |
||||||
|
try { |
||||||
|
//不存在则创建新的文件
|
||||||
|
if (!file.exists()) { |
||||||
|
file.createNewFile(); |
||||||
|
} |
||||||
|
outputStreamWriter = new OutputStreamWriter(new FileOutputStream(path, append), charset); |
||||||
|
outputStreamWriter.write(content); |
||||||
|
outputStreamWriter.close(); |
||||||
|
} catch (e) { |
||||||
|
if (null != outputStreamWriter) { |
||||||
|
outputStreamWriter.close(); |
||||||
|
} |
||||||
|
return false; |
||||||
|
} |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 判断文件是否存在 |
||||||
|
* @param path 文件路径 |
||||||
|
* @return true=存在 false=不存在 |
||||||
|
*/ |
||||||
|
const isFileExist = function(path = ''){ |
||||||
|
const File = plus.android.importClass('java.io.File'); |
||||||
|
return new File(path).exists() |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 删除文件 |
||||||
|
* @param {String} path |
||||||
|
*/ |
||||||
|
const deleteFile = function(path = ''){ |
||||||
|
const File = plus.android.importClass('java.io.File'); |
||||||
|
let file = new File(path); |
||||||
|
if (file.exists()) { |
||||||
|
return file.delete(); |
||||||
|
} |
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export default { |
||||||
|
root, |
||||||
|
filelist, |
||||||
|
createNewFile, |
||||||
|
mkdirs, |
||||||
|
readTxt, |
||||||
|
writeTxt, |
||||||
|
isFileExist, |
||||||
|
deleteFile |
||||||
|
} |
@ -0,0 +1,205 @@ |
|||||||
|
import |
||||||
|
Base64 |
||||||
|
from './base64.js' |
||||||
|
|
||||||
|
import handleFile from '@/utils/handleFile.js'; |
||||||
|
|
||||||
|
// #ifdef APP
|
||||||
|
|
||||||
|
// 保存文字
|
||||||
|
let logTextData = ''; |
||||||
|
|
||||||
|
const { |
||||||
|
root |
||||||
|
} = handleFile |
||||||
|
|
||||||
|
const file = `${root()}/scanLogs` |
||||||
|
// #endif
|
||||||
|
|
||||||
|
/** |
||||||
|
* js 日期格式化 |
||||||
|
* 传时间戳参数进行时间戳转换,不传时返回当前时间(返回年月日时分秒格式) |
||||||
|
* @param {Object} timeStamp 时间戳参数 非必传 |
||||||
|
* @param {Object} format 格式化样式 非必传 |
||||||
|
*/ |
||||||
|
function getTimeStampDatetime(format = 'yyyy-MM-dd HH:mm:ss', timeStamp) { |
||||||
|
let nowDatetime = new Date() |
||||||
|
timeStamp ? nowDatetime = new Date(timeStamp) : nowDatetime = new Date() |
||||||
|
return nowDatetime.Format(format) |
||||||
|
} |
||||||
|
Date.prototype.Format = function(fmt) { |
||||||
|
var o = { |
||||||
|
"M+": this.getMonth() + 1, //月份
|
||||||
|
"d+": this.getDate(), //日
|
||||||
|
"H+": this.getHours(), //小时
|
||||||
|
"m+": this.getMinutes(), //分
|
||||||
|
"s+": this.getSeconds(), //秒
|
||||||
|
"q+": Math.floor((this.getMonth() + 3) / 3), //季度
|
||||||
|
"S": this.getMilliseconds() //毫秒
|
||||||
|
}; |
||||||
|
if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length)); |
||||||
|
for (var k in o) |
||||||
|
if (new RegExp("(" + k + ")").test(fmt)) fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (( |
||||||
|
"00" + o[ |
||||||
|
k]).substr(("" + o[k]).length))); |
||||||
|
return fmt; |
||||||
|
} |
||||||
|
|
||||||
|
// 获取日志文件名称
|
||||||
|
function getLogFileName() { |
||||||
|
// 今日日期
|
||||||
|
let nowDate = getTimeStampDatetime("yyyy-MM-dd") |
||||||
|
// 当前时间精确到秒
|
||||||
|
let nowTime = getTimeStampDatetime("HH:mm:ss") |
||||||
|
// 文件名称
|
||||||
|
let fileName = 'app-log.txt' |
||||||
|
// fileName = nowDate + 'log.txt'
|
||||||
|
fileName = nowDate + '.txt' |
||||||
|
// 内容换行符(自定义)
|
||||||
|
let newLine = "\r\n" + "======================================" + "\r\n" + nowTime + " =>>>" + "\r\n" |
||||||
|
|
||||||
|
return { |
||||||
|
nowDate: nowDate, |
||||||
|
newLine: newLine, |
||||||
|
fileName: fileName |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* 日志内容写入 |
||||||
|
* 文件写入手机地址,华为Android/data/应用包名/documents/app-log/2021-01-17log.text |
||||||
|
* @param {Object} params 写入内容 |
||||||
|
* @param {Object} encryption 是否加密 |
||||||
|
*/ |
||||||
|
// 防抖
|
||||||
|
let writelogtime = null; |
||||||
|
|
||||||
|
function writeLog(params, encryption) { |
||||||
|
let text = JSON.stringify(params) |
||||||
|
|
||||||
|
if (encryption) { |
||||||
|
logTextData += (getLogFileName().newLine + " " + Base64.encode(text)); |
||||||
|
} else { |
||||||
|
logTextData += (getLogFileName().newLine + " " + text); |
||||||
|
} |
||||||
|
if (writelogtime) { |
||||||
|
clearTimeout(writelogtime) |
||||||
|
} |
||||||
|
//不然就创建新的定时器 3秒没有新日志再执行写入操作
|
||||||
|
writelogtime = setTimeout(function() { |
||||||
|
writeLogTxt() |
||||||
|
}, 3000) |
||||||
|
} |
||||||
|
|
||||||
|
function writeLogTxt() { |
||||||
|
plus.io.requestFileSystem(plus.io.PUBLIC_DOCUMENTS, function(fs) { |
||||||
|
// 可通过fs操作PUBLIC_DOCUMENTS文件系统
|
||||||
|
// 创建日志文件夹
|
||||||
|
fs.root.getDirectory("app-log", { |
||||||
|
create: true, |
||||||
|
exclusive: false |
||||||
|
}, function(dir) { |
||||||
|
dir.getDirectory(getLogFileName().nowDate, { |
||||||
|
create: true, |
||||||
|
exclusive: false |
||||||
|
}, function(dir) { |
||||||
|
// 创建或写入文件
|
||||||
|
console.log("Directory Entry Name: " + dir.fullPath + getLogFileName().fileName); |
||||||
|
dir.getFile(getLogFileName().fileName, { |
||||||
|
create: true |
||||||
|
}, function(fileEntry) { |
||||||
|
// 找到文件准备写入操作
|
||||||
|
fileEntry.file(function(file) { |
||||||
|
// create a FileWriter to write to the file
|
||||||
|
fileEntry.createWriter(function(writer) { |
||||||
|
// Write data to file.
|
||||||
|
writer.seek(file.size - 1) |
||||||
|
// 换行插入日志文件
|
||||||
|
writer.write(logTextData); |
||||||
|
console.log('111 :>> ', 111); |
||||||
|
logTextData = ''; |
||||||
|
}, function(e) { |
||||||
|
console.error("日志写入错误", error) |
||||||
|
}); |
||||||
|
}); |
||||||
|
}); |
||||||
|
}, function(err) { |
||||||
|
console.error("文件夹创建失败", err) |
||||||
|
}); |
||||||
|
}, function(err) { |
||||||
|
console.error("文件夹创建失败", err) |
||||||
|
}); |
||||||
|
|
||||||
|
}, function(error) { |
||||||
|
console.error("文件系统进入错误", error) |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
function delLogFile() { |
||||||
|
console.log("日志删除") |
||||||
|
plus.io.resolveLocalFileSystemURL( |
||||||
|
file, //指定的目录
|
||||||
|
function(entry) { |
||||||
|
var directoryReader = entry.createReader(); //获取读取目录对象
|
||||||
|
directoryReader.readEntries( |
||||||
|
function(entries) { //历遍子目录即可
|
||||||
|
for (var i = 0; i < entries.length; i++) { |
||||||
|
console.log("文件信息:" + entries[i].name); |
||||||
|
let twoDate = Date.parse(getLogFileName().nowDate) - (86400000 * 2); |
||||||
|
console.log("保留三天内日志", twoDate) |
||||||
|
if (Date.parse(entries[i].name) < twoDate) { |
||||||
|
entries[i].removeRecursively(function(entry) { |
||||||
|
writeLog("日志删除成功" + entries[i].name, false); |
||||||
|
}, function(e) { |
||||||
|
writeLog("日志删除成功" + e.message, false); |
||||||
|
}); |
||||||
|
} |
||||||
|
} |
||||||
|
}, |
||||||
|
function(err) { |
||||||
|
console.log("没有异常日志"); |
||||||
|
}); |
||||||
|
}, |
||||||
|
function(err) { |
||||||
|
console.log("没有异常日志"); |
||||||
|
}); |
||||||
|
} |
||||||
|
// 查询本地日志文件(直接当前页面调用,该接口未使用)
|
||||||
|
function selectLogUrl() { |
||||||
|
plus.io.resolveLocalFileSystemURL( |
||||||
|
file, //指定的目录
|
||||||
|
function(entry) { |
||||||
|
var directoryReader = entry.createReader(); //获取读取目录对象
|
||||||
|
directoryReader.readEntries( |
||||||
|
function(entries) { //历遍子目录即可
|
||||||
|
let isLog = false; |
||||||
|
for (var i = 0; i < entries.length; i++) { |
||||||
|
console.log("文件信息:" + entries[i].name); |
||||||
|
if (entries[i].name == getLogFileName().nowDate) { |
||||||
|
console.log("获取到当天文件为", entries[i].name) |
||||||
|
var url = file + '/' + entries[i].name + '/' + getLogFileName().nowDate + '.txt'; |
||||||
|
var path = plus.io.convertLocalFileSystemURL(url); |
||||||
|
var newurl = plus.io.convertAbsoluteFileSystem(path); |
||||||
|
isLog = true; |
||||||
|
return newurl; |
||||||
|
} |
||||||
|
} |
||||||
|
if (isLog) { |
||||||
|
return ''; |
||||||
|
} |
||||||
|
}, |
||||||
|
function(err) { |
||||||
|
console.log("访问目录失败"); |
||||||
|
}); |
||||||
|
}, |
||||||
|
function(err) { |
||||||
|
console.log("访问目录失败"); |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
export default { |
||||||
|
writeLog, |
||||||
|
delLogFile, |
||||||
|
getLogFileName, |
||||||
|
selectLogUrl |
||||||
|
} |
Loading…
Reference in new issue