11.21
|
@ -572,7 +572,7 @@
|
||||||
|
|
||||||
.borContent {
|
.borContent {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 120px;
|
height: 140px;
|
||||||
overflow: scroll;
|
overflow: scroll;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
|
|
@ -0,0 +1,225 @@
|
||||||
|
<template>
|
||||||
|
<view class="progress-bar">
|
||||||
|
<view class="progress-bar-title" :style="titleStyle">{{ title }}</view>
|
||||||
|
|
||||||
|
<view v-if="!secondValue" class="bar">
|
||||||
|
<view
|
||||||
|
class="bar-single"
|
||||||
|
:style="{
|
||||||
|
backgroundColor: backColor,
|
||||||
|
borderRadius: barRadius,
|
||||||
|
height: barHeight,
|
||||||
|
lineHeight: barHeight
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
class="bar-content"
|
||||||
|
:style="{ width: firstWidth, minWidth: minWidth }"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
:class="{ 'bar-content-inner': animationReal }"
|
||||||
|
:style="{ background: contentColor }"
|
||||||
|
>
|
||||||
|
<text
|
||||||
|
v-if="textPosition === 'inside'"
|
||||||
|
class="bar-text"
|
||||||
|
:style="{ color: textColor, fontSize: textSize }"
|
||||||
|
>
|
||||||
|
{{ singleText }}
|
||||||
|
</text>
|
||||||
|
<text v-else class="placeholder">1</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view
|
||||||
|
v-if="textPosition === 'outside'"
|
||||||
|
class="bar-text"
|
||||||
|
:style="{ color: textColor, fontSize: textSize }"
|
||||||
|
>
|
||||||
|
{{ singleText }}
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<view v-else class="bar">
|
||||||
|
<view
|
||||||
|
class="bar-single bar-double"
|
||||||
|
:style="{
|
||||||
|
backgroundColor: backColor,
|
||||||
|
borderRadius: barRadius,
|
||||||
|
height: barHeight,
|
||||||
|
lineHeight: barHeight
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
class="bar-content"
|
||||||
|
:style="{ width: firstWidth, minWidth: minWidth }"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
:class="{ 'bar-content-inner': animationReal }"
|
||||||
|
:style="{ background: contentColor }"
|
||||||
|
>
|
||||||
|
<text
|
||||||
|
v-if="textPosition !== 'none'"
|
||||||
|
class="bar-text"
|
||||||
|
:style="{ color: textColor, fontSize: textSize }"
|
||||||
|
>
|
||||||
|
{{ singleText }}
|
||||||
|
</text>
|
||||||
|
<text v-else class="placeholder">1</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="bar-interval placeholder">1</view>
|
||||||
|
<view
|
||||||
|
class="bar-content"
|
||||||
|
:style="{ width: secondWidth, minWidth: minWidth }"
|
||||||
|
>
|
||||||
|
<view
|
||||||
|
:class="{ 'bar-content-inner': animationReal }"
|
||||||
|
:style="{ background: contentColor2 }"
|
||||||
|
>
|
||||||
|
<text
|
||||||
|
v-if="textPosition !== 'none'"
|
||||||
|
class="bar-text"
|
||||||
|
:style="{ color: textColor, fontSize: textSize }"
|
||||||
|
>
|
||||||
|
{{ doubleText }}
|
||||||
|
</text>
|
||||||
|
<text v-else class="placeholder">1</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import props from './props.js';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
mixins: [props],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
singleText: '',
|
||||||
|
doubleText: '',
|
||||||
|
firstWidth: '',
|
||||||
|
secondWidth: '',
|
||||||
|
animationReal: false
|
||||||
|
};
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
firstValue: {
|
||||||
|
handler(newVal) {
|
||||||
|
// 进度条长度改变后,重新加载动画
|
||||||
|
if (this.animationReal) {
|
||||||
|
this.animationReal = false;
|
||||||
|
this.firstWidth = this.manageWidth(0, 'singleText');
|
||||||
|
setTimeout(() => {
|
||||||
|
this.animationReal = true;
|
||||||
|
this.firstWidth = this.manageWidth(
|
||||||
|
newVal,
|
||||||
|
'singleText'
|
||||||
|
);
|
||||||
|
}, 100);
|
||||||
|
} else {
|
||||||
|
this.firstWidth = this.manageWidth(newVal, 'singleText');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
immediate: true
|
||||||
|
},
|
||||||
|
secondValue: {
|
||||||
|
handler(newVal) {
|
||||||
|
if (this.animationReal) {
|
||||||
|
this.animationReal = false;
|
||||||
|
this.secondWidth = this.manageWidth(0, 'doubleText');
|
||||||
|
setTimeout(() => {
|
||||||
|
this.animationReal = true;
|
||||||
|
this.secondWidth = this.manageWidth(
|
||||||
|
newVal,
|
||||||
|
'doubleText'
|
||||||
|
);
|
||||||
|
}, 100);
|
||||||
|
} else {
|
||||||
|
this.secondWidth = this.manageWidth(newVal, 'doubleText');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
immediate: true
|
||||||
|
},
|
||||||
|
animation: {
|
||||||
|
handler(newVal) {
|
||||||
|
this.animationReal = newVal;
|
||||||
|
},
|
||||||
|
immediate: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
manageWidth(target, flag) {
|
||||||
|
let value =
|
||||||
|
((target / this.total) * 100).toFixed(this.precision) + '%';
|
||||||
|
this.percentum ? (this[flag] = value) : (this[flag] = target);
|
||||||
|
// 减去间隔所占宽度
|
||||||
|
this.secondValue ? (value = `calc(${value} - 1px)`) : '';
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.progress-bar {
|
||||||
|
&-title {
|
||||||
|
font-size: 4vw;
|
||||||
|
font-weight: bold;
|
||||||
|
margin-bottom: 1vw;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bar {
|
||||||
|
/* #ifdef H5 */
|
||||||
|
cursor: pointer;
|
||||||
|
/* #endif */
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
|
||||||
|
&-single {
|
||||||
|
overflow: hidden;
|
||||||
|
width: 100vw;
|
||||||
|
}
|
||||||
|
|
||||||
|
&-double {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
&-content {
|
||||||
|
&-inner {
|
||||||
|
animation: addWidth 1.5s 0s;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&-interval {
|
||||||
|
background-color: #ffffff;
|
||||||
|
width: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&-text {
|
||||||
|
margin: 0 5px;
|
||||||
|
font-size: 4vw;
|
||||||
|
}
|
||||||
|
|
||||||
|
.placeholder {
|
||||||
|
color: rgba(0, 0, 0, 0);
|
||||||
|
font-size: 4vw;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes addWidth {
|
||||||
|
from {
|
||||||
|
width: 0;
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,84 @@
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
// 标题
|
||||||
|
title: {
|
||||||
|
type: String,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
// 标题样式
|
||||||
|
titleStyle: {
|
||||||
|
type: String,
|
||||||
|
},
|
||||||
|
// 总数,计算的分母
|
||||||
|
total: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
// 第一个值
|
||||||
|
firstValue: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
// 第二个值
|
||||||
|
secondValue: {
|
||||||
|
type: Number
|
||||||
|
},
|
||||||
|
// 进度条背景颜色
|
||||||
|
backColor: {
|
||||||
|
type: String,
|
||||||
|
default: '#f2f2f2'
|
||||||
|
},
|
||||||
|
// 进度条高度
|
||||||
|
barHeight: {
|
||||||
|
type: String
|
||||||
|
},
|
||||||
|
// 进度条圆角
|
||||||
|
barRadius: {
|
||||||
|
type: String
|
||||||
|
},
|
||||||
|
// 进度条颜色
|
||||||
|
contentColor: {
|
||||||
|
type: String,
|
||||||
|
default: '#4cd964'
|
||||||
|
},
|
||||||
|
// 进度条2颜色
|
||||||
|
contentColor2: {
|
||||||
|
type: String,
|
||||||
|
default: '#f0ad4e'
|
||||||
|
},
|
||||||
|
// 进度条最小长度
|
||||||
|
minWidth: {
|
||||||
|
type: String
|
||||||
|
},
|
||||||
|
// 文字大小
|
||||||
|
textSize: {
|
||||||
|
type: String,
|
||||||
|
default: '16px'
|
||||||
|
},
|
||||||
|
// 文字颜色
|
||||||
|
textColor: {
|
||||||
|
type: String,
|
||||||
|
default: '#333333'
|
||||||
|
},
|
||||||
|
// 文字内显、外显、不显,可选值:inside/outside/none
|
||||||
|
textPosition: {
|
||||||
|
type: String,
|
||||||
|
default: 'inside'
|
||||||
|
},
|
||||||
|
// 展示百分比或值
|
||||||
|
percentum: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false
|
||||||
|
},
|
||||||
|
// 百分比精确到小数点后几位
|
||||||
|
precision: {
|
||||||
|
type: [String, Number],
|
||||||
|
default: 2
|
||||||
|
},
|
||||||
|
// 是否选择动画加载
|
||||||
|
animation: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -151,6 +151,12 @@
|
||||||
// 选择船输入框
|
// 选择船输入框
|
||||||
shipInput(e) {
|
shipInput(e) {
|
||||||
this.shipValue = e
|
this.shipValue = e
|
||||||
|
if (e == "") {
|
||||||
|
this.vvyId = ""
|
||||||
|
this.vvyShip = ""
|
||||||
|
this.shipId = ""
|
||||||
|
this.shipName = ""
|
||||||
|
}
|
||||||
this.getShip()
|
this.getShip()
|
||||||
},
|
},
|
||||||
// 获取船舶
|
// 获取船舶
|
||||||
|
@ -254,7 +260,7 @@
|
||||||
line-height: 35px;
|
line-height: 35px;
|
||||||
padding-left: 10px;
|
padding-left: 10px;
|
||||||
margin-right: 15px;
|
margin-right: 15px;
|
||||||
margin-top: 8px;
|
margin-top: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn {
|
.btn {
|
||||||
|
@ -265,7 +271,7 @@
|
||||||
color: #fff;
|
color: #fff;
|
||||||
background-color: #0067CF;
|
background-color: #0067CF;
|
||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
margin-top: 8px;
|
margin-top: 10px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -75,7 +75,10 @@
|
||||||
<template v-if="itemList.length > 0">
|
<template v-if="itemList.length > 0">
|
||||||
<view class="itemBox">
|
<view class="itemBox">
|
||||||
<view class="itemList">
|
<view class="itemList">
|
||||||
<view class="exp" v-for="(item,index) in itemList" :key="item.index">
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="itemList" :column="2"
|
||||||
|
:columnSpace="1.5" :seat="2">
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="exp">
|
||||||
<view class="item">
|
<view class="item">
|
||||||
<view class="row">
|
<view class="row">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
|
@ -95,7 +98,8 @@
|
||||||
</view>
|
</view>
|
||||||
<text>{{itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)}}%</text>
|
<text>{{itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)}}%</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus0" v-if="item.totalPlanStatus == 0">
|
<view class="planStatus planStatus0"
|
||||||
|
v-if="item.totalPlanStatus == 0">
|
||||||
<text class="text">{{item.totalPlanStatusDesc}}</text>
|
<text class="text">{{item.totalPlanStatusDesc}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus3"
|
<view class="planStatus planStatus3"
|
||||||
|
@ -136,7 +140,8 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-if="itemActive != index" @click="isActive(index)">
|
<view class="expand" v-if="itemActive != item.index"
|
||||||
|
@click="isActive(item.index)">
|
||||||
展开
|
展开
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-else @click="itemActive = '-1'">
|
<view class="expand" v-else @click="itemActive = '-1'">
|
||||||
|
@ -144,7 +149,7 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="details">
|
<view class="details">
|
||||||
<view class="itemDetails" v-show="itemActive == index">
|
<view class="itemDetails" v-show="itemActive == item.index">
|
||||||
<template v-for="(item2,index2) in item.infoList">
|
<template v-for="(item2,index2) in item.infoList">
|
||||||
<view class="detailsBox" :key="index2">
|
<view class="detailsBox" :key="index2">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
|
@ -174,14 +179,12 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
</template>
|
||||||
|
</custom-waterfalls-flow>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
||||||
<!-- <view class="pageBox" v-if="itemList.length > 0">
|
|
||||||
<uni-pagination :show-icon="true" :total="total" :pageSize="pageSize" :current="current"
|
|
||||||
@change="changePage" />
|
|
||||||
</view> -->
|
|
||||||
</view>
|
</view>
|
||||||
</custom-tab-pane>
|
</custom-tab-pane>
|
||||||
<custom-tab-pane label="分指令" name="c1_2">
|
<custom-tab-pane label="分指令" name="c1_2">
|
||||||
|
@ -247,14 +250,18 @@
|
||||||
<checkbox-group @change="checkChange">
|
<checkbox-group @change="checkChange">
|
||||||
<view class="itemBox">
|
<view class="itemBox">
|
||||||
<view class="itemList">
|
<view class="itemList">
|
||||||
<view class="exp" v-for="(item,index) in itemList" :key="item.ndex">
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="itemList" :column="2"
|
||||||
|
:columnSpace="1.5" :seat="2">
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="exp">
|
||||||
<view class="item">
|
<view class="item">
|
||||||
<view class="row">
|
<view class="row">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
<view class="rowHead">
|
<view class="rowHead">
|
||||||
<checkbox :value="item.lwpId" :checked="false" />
|
<checkbox :value="item.lwpId" :checked="false" />
|
||||||
</view>
|
</view>
|
||||||
<image class="titleImg" src="../../static/images/zlIcon.png"
|
<image class="titleImg"
|
||||||
|
src="../../static/images/zlIcon.png"
|
||||||
mode="widthFix">
|
mode="widthFix">
|
||||||
</image>
|
</image>
|
||||||
<view class="text">
|
<view class="text">
|
||||||
|
@ -313,11 +320,12 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="xfBtn">
|
<view class="xfBtn">
|
||||||
<view class="btn" @click="distribute('center','solo',item)">指令下发
|
<view class="btn" @click="distribute('center','solo',item)">
|
||||||
|
指令下发
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-if="itemActive != index"
|
<view class="expand" v-if="itemActive != item.index"
|
||||||
@click="isActive(index)">
|
@click="isActive(item.index)">
|
||||||
展开
|
展开
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-else @click="itemActive = '-1'">
|
<view class="expand" v-else @click="itemActive = '-1'">
|
||||||
|
@ -325,12 +333,13 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="details">
|
<view class="details">
|
||||||
<view class="itemDetails" v-show="itemActive == index">
|
<view class="itemDetails" v-show="itemActive == item.index">
|
||||||
<template v-for="(item2,index2) in item.infoList">
|
<template v-for="(item2,index2) in item.infoList">
|
||||||
<view class="detailsBox" :key="index2">
|
<view class="detailsBox" :key="index2">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
<text>{{item2.mnfBl}}</text>
|
<text>{{item2.mnfBl}}</text>
|
||||||
<button @click="toDetails(item,item2)">详情</button>
|
<button
|
||||||
|
@click="toDetails(item,item2)">详情</button>
|
||||||
</view>
|
</view>
|
||||||
<view class="info">
|
<view class="info">
|
||||||
<view class="cell">
|
<view class="cell">
|
||||||
|
@ -355,15 +364,13 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
</template>
|
||||||
|
</custom-waterfalls-flow>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</checkbox-group>
|
</checkbox-group>
|
||||||
</template>
|
</template>
|
||||||
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
||||||
<!-- <view class="pageBox" v-if="itemList.length > 0">
|
|
||||||
<uni-pagination :show-icon="true" :total="total" :pageSize="pageSize" :current="current"
|
|
||||||
@change="changePage" />
|
|
||||||
</view> -->
|
|
||||||
</view>
|
</view>
|
||||||
</custom-tab-pane>
|
</custom-tab-pane>
|
||||||
<custom-tab-pane label="分指令-班组长" name="c1_4">
|
<custom-tab-pane label="分指令-班组长" name="c1_4">
|
||||||
|
@ -386,7 +393,10 @@
|
||||||
</view>
|
</view>
|
||||||
<template v-if="itemList.length > 0">
|
<template v-if="itemList.length > 0">
|
||||||
<view class="itemList">
|
<view class="itemList">
|
||||||
<view class="exp" v-for="(item,index) in itemList" :key="item.ndex">
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="itemList" :column="2"
|
||||||
|
:columnSpace="1.5" :seat="2">
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="exp">
|
||||||
<view class="item">
|
<view class="item">
|
||||||
<view class="row">
|
<view class="row">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
|
@ -400,19 +410,22 @@
|
||||||
<view class="schedule">
|
<view class="schedule">
|
||||||
<text class="text">卸船进度</text>
|
<text class="text">卸船进度</text>
|
||||||
<view class="progressBox">
|
<view class="progressBox">
|
||||||
<van-progress color="#0067CF" stroke-width="6px" :show-pivot="false"
|
<van-progress color="#0067CF" stroke-width="6px"
|
||||||
track-color="#DEE9F5"
|
:show-pivot="false" track-color="#DEE9F5"
|
||||||
:percentage="itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)" />
|
:percentage="itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)" />
|
||||||
</view>
|
</view>
|
||||||
<text>{{itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)}}%</text>
|
<text>{{itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)}}%</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus0" v-if="item.branchPlanStatus == 0">
|
<view class="planStatus planStatus0"
|
||||||
|
v-if="item.branchPlanStatus == 0">
|
||||||
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus3" v-else-if="item.branchPlanStatus == 2">
|
<view class="planStatus planStatus3"
|
||||||
|
v-else-if="item.branchPlanStatus == 2">
|
||||||
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus5" v-else-if="item.branchPlanStatus == 4">
|
<view class="planStatus planStatus5"
|
||||||
|
v-else-if="item.branchPlanStatus == 4">
|
||||||
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus1" v-else>
|
<view class="planStatus planStatus1" v-else>
|
||||||
|
@ -447,7 +460,8 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-if="itemActive != index" @click="isActive(index)">
|
<view class="expand" v-if="itemActive != item.index"
|
||||||
|
@click="isActive(item.index)">
|
||||||
展开
|
展开
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-else @click="itemActive = '-1'">
|
<view class="expand" v-else @click="itemActive = '-1'">
|
||||||
|
@ -455,7 +469,7 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="details">
|
<view class="details">
|
||||||
<view class="itemDetails" v-show="itemActive == index">
|
<view class="itemDetails" v-show="itemActive == item.index">
|
||||||
<template v-for="(item2,index2) in item.infoList">
|
<template v-for="(item2,index2) in item.infoList">
|
||||||
<view class="detailsBox" :key="index2">
|
<view class="detailsBox" :key="index2">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
|
@ -485,13 +499,11 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
</template>
|
||||||
|
</custom-waterfalls-flow>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
||||||
<!-- <view class="pageBox" v-if="itemList.length > 0">
|
|
||||||
<uni-pagination :show-icon="true" :total="total" :pageSize="pageSize" :current="current"
|
|
||||||
@change="changePage" />
|
|
||||||
</view> -->
|
|
||||||
</view>
|
</view>
|
||||||
</custom-tab-pane>
|
</custom-tab-pane>
|
||||||
<custom-tab-pane label="场位图" name="c1_3">
|
<custom-tab-pane label="场位图" name="c1_3">
|
||||||
|
@ -767,8 +779,7 @@
|
||||||
placeTabs: 0,
|
placeTabs: 0,
|
||||||
|
|
||||||
// 分页
|
// 分页
|
||||||
total: 0,
|
pageSize: 10,
|
||||||
pageSize: 4,
|
|
||||||
current: 1,
|
current: 1,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -813,6 +824,7 @@
|
||||||
this.bvmName = ''
|
this.bvmName = ''
|
||||||
this.itemList = []
|
this.itemList = []
|
||||||
this.tabsValue = e.value
|
this.tabsValue = e.value
|
||||||
|
this.current = 1
|
||||||
this.zTjActive = -1
|
this.zTjActive = -1
|
||||||
if (e.value == 1 || e.value == 2) {
|
if (e.value == 1 || e.value == 2) {
|
||||||
this.loadOtherOrder()
|
this.loadOtherOrder()
|
||||||
|
@ -897,7 +909,7 @@
|
||||||
this.brdId = num
|
this.brdId = num
|
||||||
}
|
}
|
||||||
uni.request({
|
uni.request({
|
||||||
url: `${this.$local}/api/unload/command/page?vvyId=${this.shipInfo.vvyId}&brdId=${this.brdId}&mnfBl=${this.mnfBl}&bvmId=${this.bvmId}&tradeType=${this.tradeType}`,
|
url: `${this.$local}/api/unload/command/page?vvyId=${this.shipInfo.vvyId}&brdId=${this.brdId}&mnfBl=${this.mnfBl}&bvmId=${this.bvmId}&tradeType=${this.tradeType}&size=${this.pageSize}¤t=${this.current}`,
|
||||||
header: {
|
header: {
|
||||||
'Content-Type': 'application/json', //自定义请求头信息
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
'Authorization': `Bearer ${this.loginObj.access_token}`
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
|
@ -905,10 +917,15 @@
|
||||||
method: 'GET', //请求方式,必须为大写
|
method: 'GET', //请求方式,必须为大写
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
console.log(res)
|
console.log(res)
|
||||||
this.itemList = res.data.data.records
|
this.itemList.push(...res.data.data.records)
|
||||||
this.total = res.data.data.total
|
if (res.data.data.records.length == 10) {
|
||||||
|
this.current++
|
||||||
|
this.loadSumOrder()
|
||||||
|
}
|
||||||
this.itemList.forEach((v, index) => {
|
this.itemList.forEach((v, index) => {
|
||||||
// if (v.branchPlanStatus == 0) {
|
// if (v.branchPlanStatus == 0) {
|
||||||
|
v.image = '../../static/images/theme/car1.png'
|
||||||
|
v.index = index
|
||||||
this.zzlLwpIdList.push(v.lwpId)
|
this.zzlLwpIdList.push(v.lwpId)
|
||||||
// }
|
// }
|
||||||
this.getBottomInfo(v.lwpId, index)
|
this.getBottomInfo(v.lwpId, index)
|
||||||
|
@ -934,9 +951,14 @@
|
||||||
method: 'GET', //请求方式,必须为大写
|
method: 'GET', //请求方式,必须为大写
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
console.log(res)
|
console.log(res)
|
||||||
this.itemList = res.data.data.records
|
this.itemList.push(...res.data.data.records)
|
||||||
this.total = res.data.data.total
|
if (res.data.data.records.length == 10) {
|
||||||
|
this.current++
|
||||||
|
this.loadOtherOrder()
|
||||||
|
}
|
||||||
this.itemList.forEach((v, index) => {
|
this.itemList.forEach((v, index) => {
|
||||||
|
v.image = '../../static/images/theme/car1.png'
|
||||||
|
v.index = index
|
||||||
this.getBottomInfo(v.lwpId, index)
|
this.getBottomInfo(v.lwpId, index)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -1321,9 +1343,14 @@
|
||||||
color: #108ee9;
|
color: #108ee9;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
page {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
.content {
|
.content {
|
||||||
background-color: #F6F7F9;
|
background-color: #F6F7F9;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
|
||||||
/deep/.tab .tab-bar {
|
/deep/.tab .tab-bar {
|
||||||
height: 66px;
|
height: 66px;
|
||||||
|
@ -1338,6 +1365,7 @@
|
||||||
color: #23262E;
|
color: #23262E;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
font-family: PingFangSC-Semibold;
|
font-family: PingFangSC-Semibold;
|
||||||
|
padding: 0 24px !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
/deep/.tab .tab-bar-item.active {
|
/deep/.tab .tab-bar-item.active {
|
||||||
|
@ -1356,6 +1384,27 @@
|
||||||
|
|
||||||
.main {
|
.main {
|
||||||
height: calc(100vh - 134px);
|
height: calc(100vh - 134px);
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.askBox {
|
||||||
|
width: 97%;
|
||||||
|
padding: 20px;
|
||||||
|
margin: 20px;
|
||||||
|
background: #fff;
|
||||||
|
height: 300px;
|
||||||
|
overflow: scroll;
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-size: 20px;
|
||||||
|
font-weight: bold;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.askContent {
|
||||||
|
font-size: 16px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.pageBox {
|
.pageBox {
|
||||||
|
@ -1599,27 +1648,35 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.itemBox {
|
.itemBox {
|
||||||
height: 465px;
|
height: 535px;
|
||||||
overflow: scroll;
|
overflow: scroll;
|
||||||
|
padding-bottom: 60px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.itemList {
|
.itemList {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
|
align-items: flex-start;
|
||||||
padding: 0 12px;
|
padding: 0 12px;
|
||||||
max-height: 70vh;
|
|
||||||
overflow: scroll;
|
|
||||||
|
|
||||||
.exp {
|
.exp {
|
||||||
// width: 100%;
|
width: 100%;
|
||||||
width: 49.5%;
|
|
||||||
margin-bottom: 15px;
|
margin-bottom: 15px;
|
||||||
background: #FFFFFF;
|
background: #FFFFFF;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
padding: 12px;
|
padding: 12px;
|
||||||
padding-right: 0;
|
padding-right: 0;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
/deep/.waterfalls-flow {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/deep/.waterfalls-flow-column {
|
||||||
|
margin-left: 2%;
|
||||||
|
}
|
||||||
|
|
||||||
.rowHead {
|
.rowHead {
|
||||||
/deep/uni-checkbox .uni-checkbox-input {
|
/deep/uni-checkbox .uni-checkbox-input {
|
||||||
|
@ -1630,6 +1687,7 @@
|
||||||
|
|
||||||
.row {
|
.row {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
padding: 5px 0px;
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
@ -1831,7 +1889,6 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
.pzPot {
|
.pzPot {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
@ -1892,6 +1949,7 @@
|
||||||
|
|
||||||
.imgTable {
|
.imgTable {
|
||||||
margin-top: 20px;
|
margin-top: 20px;
|
||||||
|
padding: 0 20px;
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
|
@ -1905,36 +1963,43 @@
|
||||||
|
|
||||||
.imgLi {
|
.imgLi {
|
||||||
display: flex;
|
display: flex;
|
||||||
position: relative;
|
|
||||||
|
|
||||||
.shipImg {
|
|
||||||
width: 100%;
|
|
||||||
height: 171px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.maskBox {
|
.maskBox {
|
||||||
width: calc(100% - 200px);
|
width: 1060px;
|
||||||
height: 150px;
|
height: 191px;
|
||||||
background-color: transparent;
|
margin: 0 auto;
|
||||||
display: flex;
|
background: url('@/static/images/ship-split.svg');
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-size: contain;
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
.line {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 11px;
|
width: 850px;
|
||||||
|
height: 85%;
|
||||||
|
top: 15px;
|
||||||
left: 60px;
|
left: 60px;
|
||||||
|
display: flex;
|
||||||
|
overflow-y: auto;
|
||||||
|
|
||||||
.testLine {
|
.testLine {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
height: 100%;
|
|
||||||
border-left: 1px solid #999;
|
border-left: 1px solid #999;
|
||||||
}
|
}
|
||||||
|
|
||||||
.testLine:first-child {
|
.testLine:first-child {
|
||||||
border-left: none;
|
border: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.testInfo {
|
.mycanvas {
|
||||||
height: 30px;
|
position: absolute;
|
||||||
background-color: #0067CF;
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 850px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.heightBox {
|
.heightBox {
|
||||||
|
@ -1945,36 +2010,34 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.imgSize {
|
.imgSize {
|
||||||
width: 28px;
|
width: 60px;
|
||||||
height: 28px;
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
||||||
.leftTopImg {
|
.leftTopImg {
|
||||||
transform: rotate(90deg);
|
transform: rotate(90deg);
|
||||||
top: 10px;
|
top: 5px;
|
||||||
left: 8px;
|
left: -5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.imgSize2 {
|
.imgSize2 {
|
||||||
width: 100px;
|
width: 130px;
|
||||||
height: 30px;
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
||||||
.leftBotImg {
|
.leftBotImg {
|
||||||
bottom: 10px;
|
bottom: 2px;
|
||||||
left: 8px;
|
left: -1px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.topImg {
|
.topImg {
|
||||||
top: -5px;
|
top: 0;
|
||||||
left: 40%;
|
left: 40%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.botImg {
|
.botImg {
|
||||||
transform: rotate(180deg);
|
transform: rotate(180deg);
|
||||||
bottom: -6px;
|
bottom: 0;
|
||||||
left: 40%;
|
left: 40%;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -151,6 +151,12 @@
|
||||||
// 选择船输入框
|
// 选择船输入框
|
||||||
shipInput(e) {
|
shipInput(e) {
|
||||||
this.shipValue = e
|
this.shipValue = e
|
||||||
|
if (e == "") {
|
||||||
|
this.vvyId = ""
|
||||||
|
this.vvyShip = ""
|
||||||
|
this.shipId = ""
|
||||||
|
this.shipName = ""
|
||||||
|
}
|
||||||
this.getShip()
|
this.getShip()
|
||||||
},
|
},
|
||||||
// 获取船舶
|
// 获取船舶
|
||||||
|
@ -206,10 +212,6 @@
|
||||||
this.lotusLoadingData.isShow = false
|
this.lotusLoadingData.isShow = false
|
||||||
this.total = res.data.data.total
|
this.total = res.data.data.total
|
||||||
this.itemList.push(...res.data.data.records)
|
this.itemList.push(...res.data.data.records)
|
||||||
this.vvyId = ""
|
|
||||||
this.vvyShip = ""
|
|
||||||
this.shipId = ""
|
|
||||||
this.shipName = ""
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
@ -265,7 +267,7 @@
|
||||||
line-height: 35px;
|
line-height: 35px;
|
||||||
padding-left: 10px;
|
padding-left: 10px;
|
||||||
margin-right: 15px;
|
margin-right: 15px;
|
||||||
margin-top: 8px;
|
margin-top: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn {
|
.btn {
|
||||||
|
@ -276,7 +278,7 @@
|
||||||
color: #fff;
|
color: #fff;
|
||||||
background-color: #0067CF;
|
background-color: #0067CF;
|
||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
margin-top: 8px;
|
margin-top: 10px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -78,7 +78,10 @@
|
||||||
<template v-if="itemList.length > 0">
|
<template v-if="itemList.length > 0">
|
||||||
<view class="itemBox">
|
<view class="itemBox">
|
||||||
<view class="itemList">
|
<view class="itemList">
|
||||||
<view class="exp" v-for="(item,index) in itemList" :key="item.index">
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="itemList" :column="2"
|
||||||
|
:columnSpace="1.5" :seat="2">
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="exp">
|
||||||
<view class="item">
|
<view class="item">
|
||||||
<view class="row">
|
<view class="row">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
|
@ -98,7 +101,8 @@
|
||||||
</view>
|
</view>
|
||||||
<text>{{itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)}}%</text>
|
<text>{{itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)}}%</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus0" v-if="item.totalPlanStatus == 0">
|
<view class="planStatus planStatus0"
|
||||||
|
v-if="item.totalPlanStatus == 0">
|
||||||
<text class="text">{{item.totalPlanStatusDesc}}</text>
|
<text class="text">{{item.totalPlanStatusDesc}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus3"
|
<view class="planStatus planStatus3"
|
||||||
|
@ -139,7 +143,8 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-if="itemActive != index" @click="isActive(index)">
|
<view class="expand" v-if="itemActive != item.index"
|
||||||
|
@click="isActive(item.index)">
|
||||||
展开
|
展开
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-else @click="itemActive = '-1'">
|
<view class="expand" v-else @click="itemActive = '-1'">
|
||||||
|
@ -147,7 +152,7 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="details">
|
<view class="details">
|
||||||
<view class="itemDetails" v-show="itemActive == index">
|
<view class="itemDetails" v-show="itemActive == item.index">
|
||||||
<template v-for="(item2,index2) in item.infoList">
|
<template v-for="(item2,index2) in item.infoList">
|
||||||
<view class="detailsBox" :key="index2">
|
<view class="detailsBox" :key="index2">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
|
@ -177,15 +182,12 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
</template>
|
||||||
|
</custom-waterfalls-flow>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
||||||
<!-- <view class="pageBox" v-if="itemList.length > 0">
|
|
||||||
<uni-pagination :show-icon="true" :total="total" :pageSize="pageSize" :current="current"
|
|
||||||
@change="changePage" />
|
|
||||||
</view> -->
|
|
||||||
</view>
|
</view>
|
||||||
</custom-tab-pane>
|
</custom-tab-pane>
|
||||||
<custom-tab-pane label="分指令" name="c1_2">
|
<custom-tab-pane label="分指令" name="c1_2">
|
||||||
|
@ -254,14 +256,18 @@
|
||||||
<checkbox-group @change="checkChange">
|
<checkbox-group @change="checkChange">
|
||||||
<view class="itemBox">
|
<view class="itemBox">
|
||||||
<view class="itemList">
|
<view class="itemList">
|
||||||
<view class="exp" v-for="(item,index) in itemList" :key="index">
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="itemList" :column="2"
|
||||||
|
:columnSpace="1.5" :seat="2">
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="exp">
|
||||||
<view class="item">
|
<view class="item">
|
||||||
<view class="row">
|
<view class="row">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
<view class="rowHead">
|
<view class="rowHead">
|
||||||
<checkbox :value="item.lwpId" :checked="false" />
|
<checkbox :value="item.lwpId" :checked="false" />
|
||||||
</view>
|
</view>
|
||||||
<image class="titleImg" src="../../static/images/zlIcon.png"
|
<image class="titleImg"
|
||||||
|
src="../../static/images/zlIcon.png"
|
||||||
mode="widthFix">
|
mode="widthFix">
|
||||||
</image>
|
</image>
|
||||||
<view class="text">
|
<view class="text">
|
||||||
|
@ -320,11 +326,12 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="xfBtn">
|
<view class="xfBtn">
|
||||||
<view class="btn" @click="distribute('center','solo',item)">指令下发
|
<view class="btn" @click="distribute('center','solo',item)">
|
||||||
|
指令下发
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-if="itemActive != index"
|
<view class="expand" v-if="itemActive != item.index"
|
||||||
@click="isActive(index)">
|
@click="isActive(item.index)">
|
||||||
展开
|
展开
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-else @click="itemActive = '-1'">
|
<view class="expand" v-else @click="itemActive = '-1'">
|
||||||
|
@ -332,12 +339,13 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="details">
|
<view class="details">
|
||||||
<view class="itemDetails" v-show="itemActive == index">
|
<view class="itemDetails" v-show="itemActive == item.index">
|
||||||
<template v-for="(item2,index2) in item.infoList">
|
<template v-for="(item2,index2) in item.infoList">
|
||||||
<view class="detailsBox" :key="index2">
|
<view class="detailsBox" :key="index2">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
<text>{{item2.mnfBl}}</text>
|
<text>{{item2.mnfBl}}</text>
|
||||||
<button @click="toDetails(item2.spsId)">详情</button>
|
<button
|
||||||
|
@click="toDetails(item2.spsId)">详情</button>
|
||||||
</view>
|
</view>
|
||||||
<view class="info">
|
<view class="info">
|
||||||
<view class="cell">
|
<view class="cell">
|
||||||
|
@ -362,15 +370,13 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
</template>
|
||||||
|
</custom-waterfalls-flow>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</checkbox-group>
|
</checkbox-group>
|
||||||
</template>
|
</template>
|
||||||
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
||||||
<!-- <view class="pageBox" v-if="itemList.length > 0">
|
|
||||||
<uni-pagination :show-icon="true" :total="total" :pageSize="pageSize" :current="current"
|
|
||||||
@change="changePage" />
|
|
||||||
</view> -->
|
|
||||||
</view>
|
</view>
|
||||||
</custom-tab-pane>
|
</custom-tab-pane>
|
||||||
<custom-tab-pane label="分指令-班组长" name="c1_6">
|
<custom-tab-pane label="分指令-班组长" name="c1_6">
|
||||||
|
@ -396,7 +402,10 @@
|
||||||
</view>
|
</view>
|
||||||
<template v-if="itemList.length > 0">
|
<template v-if="itemList.length > 0">
|
||||||
<view class="itemList">
|
<view class="itemList">
|
||||||
<view class="exp" v-for="(item,index) in itemList" :key="item.ndex">
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="itemList" :column="2"
|
||||||
|
:columnSpace="1.5" :seat="2">
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="exp">
|
||||||
<view class="item">
|
<view class="item">
|
||||||
<view class="row">
|
<view class="row">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
|
@ -410,19 +419,22 @@
|
||||||
<view class="schedule">
|
<view class="schedule">
|
||||||
<text class="text">装船进度</text>
|
<text class="text">装船进度</text>
|
||||||
<view class="progressBox">
|
<view class="progressBox">
|
||||||
<van-progress color="#0067CF" stroke-width="6px" :show-pivot="false"
|
<van-progress color="#0067CF" stroke-width="6px"
|
||||||
track-color="#DEE9F5"
|
:show-pivot="false" track-color="#DEE9F5"
|
||||||
:percentage="itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)" />
|
:percentage="itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)" />
|
||||||
</view>
|
</view>
|
||||||
<text>{{itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)}}%</text>
|
<text>{{itemSum(item.loadingProgress.workProgress, item.loadingProgress.totalProgress)}}%</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus0" v-if="item.branchPlanStatus == 0">
|
<view class="planStatus planStatus0"
|
||||||
|
v-if="item.branchPlanStatus == 0">
|
||||||
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus3" v-else-if="item.branchPlanStatus == 2">
|
<view class="planStatus planStatus3"
|
||||||
|
v-else-if="item.branchPlanStatus == 2">
|
||||||
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus5" v-else-if="item.branchPlanStatus == 4">
|
<view class="planStatus planStatus5"
|
||||||
|
v-else-if="item.branchPlanStatus == 4">
|
||||||
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
<text class="text">{{item.branchPlanStatusDesc}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="planStatus planStatus1" v-else>
|
<view class="planStatus planStatus1" v-else>
|
||||||
|
@ -457,7 +469,8 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-if="itemActive != index" @click="isActive(index)">
|
<view class="expand" v-if="itemActive != item.index"
|
||||||
|
@click="isActive(item.index)">
|
||||||
展开
|
展开
|
||||||
</view>
|
</view>
|
||||||
<view class="expand" v-else @click="itemActive = '-1'">
|
<view class="expand" v-else @click="itemActive = '-1'">
|
||||||
|
@ -465,7 +478,7 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="details">
|
<view class="details">
|
||||||
<view class="itemDetails" v-show="itemActive == index">
|
<view class="itemDetails" v-show="itemActive == item.index">
|
||||||
<template v-for="(item2,index2) in item.infoList">
|
<template v-for="(item2,index2) in item.infoList">
|
||||||
<view class="detailsBox" :key="index2">
|
<view class="detailsBox" :key="index2">
|
||||||
<view class="title">
|
<view class="title">
|
||||||
|
@ -495,13 +508,11 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
</template>
|
||||||
|
</custom-waterfalls-flow>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
||||||
<!-- <view class="pageBox" v-if="itemList.length > 0">
|
|
||||||
<uni-pagination :show-icon="true" :total="total" :pageSize="pageSize" :current="current"
|
|
||||||
@change="changePage" />
|
|
||||||
</view> -->
|
|
||||||
</view>
|
</view>
|
||||||
</custom-tab-pane>
|
</custom-tab-pane>
|
||||||
<custom-tab-pane label="配载图" name="c1_3">
|
<custom-tab-pane label="配载图" name="c1_3">
|
||||||
|
@ -536,33 +547,39 @@
|
||||||
<text class="text">是否有跳板: {{item.whetherJumpName}}</text>
|
<text class="text">是否有跳板: {{item.whetherJumpName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="imgLi">
|
<view class="imgLi">
|
||||||
<image class="shipImg" src="@/static/images/ship-split.svg"></image>
|
<!-- <image class="shipImg" src="@/static/images/ship-split.svg" mode="widthFix"></image> -->
|
||||||
<view class="maskBox">
|
<view class="maskBox">
|
||||||
<view class="testLine" v-for="item2 in item.shipSpace" :key="item2"></view>
|
<view class="line">
|
||||||
<view class="testInfo">
|
<!-- {{item.maxHeight}} -->
|
||||||
|
<view class="testLine" v-for="item2 in item.shipSpace" :key="item2"
|
||||||
|
:style="{height:item.maxHeight+'px'}">
|
||||||
|
</view>
|
||||||
|
<canvas class="mycanvas" :canvas-id="'mycanvas' + index" @:data-id="index"
|
||||||
|
:style="{height:item.maxHeight+'px'}"></canvas>
|
||||||
|
</view>
|
||||||
|
<!-- -->
|
||||||
|
<view class="imgSize leftTopImg"
|
||||||
|
v-if="item.whetherJump == 2 || item.whetherJump == 4 || item.whetherJump == 6 || item.whetherJump == 7 || item.whetherJump == 8 || item.whetherJump == 9">
|
||||||
|
<image src="@/static/images/jump-tail.svg" mode="widthFix"></image>
|
||||||
|
</view>
|
||||||
|
<view class="imgSize leftBotImg"
|
||||||
|
v-if="item.whetherJump == 4 || item.whetherJump == 7 || item.whetherJump == 9">
|
||||||
|
<image src="@/static/images/jump-tail.svg" mode="widthFix"></image>
|
||||||
|
</view>
|
||||||
|
<view class="imgSize2 topImg"
|
||||||
|
v-if="item.whetherJump == 3 || item.whetherJump == 5 || item.whetherJump == 6 || item.whetherJump == 7 || item.whetherJump == 8 || item.whetherJump == 9">
|
||||||
|
<image src="@/static/images/jump-mid.svg" mode="widthFix"></image>
|
||||||
|
</view>
|
||||||
|
<view class="imgSize2 botImg"
|
||||||
|
v-if="item.whetherJump == 5 || item.whetherJump == 8 || item.whetherJump == 9">
|
||||||
|
<image src="@/static/images/jump-mid.svg" mode="widthFix"></image>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="heightBox">
|
<view class="heightBox">
|
||||||
{{item.tierHeight}}m
|
{{item.tierHeight}}m
|
||||||
</view>
|
</view>
|
||||||
<view class="imgSize leftTopImg"
|
|
||||||
v-if="item.whetherJump == 2 || item.whetherJump == 4 || item.whetherJump == 6 || item.whetherJump == 7 || item.whetherJump == 8 || item.whetherJump == 9">
|
|
||||||
<image src="@/static/images/jump-tail.svg" mode=""></image>
|
|
||||||
</view>
|
|
||||||
<view class="imgSize leftBotImg"
|
|
||||||
v-if="item.whetherJump == 4 || item.whetherJump == 7 || item.whetherJump == 9">
|
|
||||||
<image src="@/static/images/jump-tail.svg" mode=""></image>
|
|
||||||
</view>
|
|
||||||
<view class="imgSize2 topImg"
|
|
||||||
v-if="item.whetherJump == 3 || item.whetherJump == 5 || item.whetherJump == 6 || item.whetherJump == 7 || item.whetherJump == 8 || item.whetherJump == 9">
|
|
||||||
<image src="@/static/images/jump-mid.svg" mode=""></image>
|
|
||||||
</view>
|
|
||||||
<view class="imgSize2 botImg"
|
|
||||||
v-if="item.whetherJump == 5 || item.whetherJump == 8 || item.whetherJump == 9">
|
|
||||||
<image src="@/static/images/jump-mid.svg" mode=""></image>
|
|
||||||
</view>
|
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
<view class="tableHead">
|
<view class="tableHead">
|
||||||
<text>货物明细</text>
|
<text>货物明细</text>
|
||||||
<uni-icons class="jt" type="bottom" size="24"></uni-icons>
|
<uni-icons class="jt" type="bottom" size="24"></uni-icons>
|
||||||
|
@ -584,6 +601,7 @@
|
||||||
<uni-th align="center" width="20">位置</uni-th>
|
<uni-th align="center" width="20">位置</uni-th>
|
||||||
</uni-tr>
|
</uni-tr>
|
||||||
<!-- 表格数据行 -->
|
<!-- 表格数据行 -->
|
||||||
|
<template v-if="goodsInfo.length && goodsInfo.length > 0">
|
||||||
<uni-tr v-for="(item2,index2) in goodsInfo[index].stowageList" :key="index2">
|
<uni-tr v-for="(item2,index2) in goodsInfo[index].stowageList" :key="index2">
|
||||||
<uni-td align="center">{{item2.stowageNo}}</uni-td>
|
<uni-td align="center">{{item2.stowageNo}}</uni-td>
|
||||||
<uni-td align="center">{{item2.mnfBl}}</uni-td>
|
<uni-td align="center">{{item2.mnfBl}}</uni-td>
|
||||||
|
@ -598,6 +616,7 @@
|
||||||
<uni-td align="center">{{item2.carHeight}}</uni-td>
|
<uni-td align="center">{{item2.carHeight}}</uni-td>
|
||||||
<uni-td align="center">{{item2.cabinNoList}}</uni-td>
|
<uni-td align="center">{{item2.cabinNoList}}</uni-td>
|
||||||
</uni-tr>
|
</uni-tr>
|
||||||
|
</template>
|
||||||
</uni-table>
|
</uni-table>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
|
@ -892,9 +911,8 @@
|
||||||
placeTabs: 0,
|
placeTabs: 0,
|
||||||
|
|
||||||
// 分页
|
// 分页
|
||||||
// total: 0,
|
pageSize: 10,
|
||||||
// pageSize: 4,
|
current: 1,
|
||||||
// current: 1,
|
|
||||||
|
|
||||||
// 配载图港口
|
// 配载图港口
|
||||||
pzPotList: [],
|
pzPotList: [],
|
||||||
|
@ -920,7 +938,6 @@
|
||||||
},
|
},
|
||||||
onLoad(options) {
|
onLoad(options) {
|
||||||
this.shipInfo = JSON.parse(decodeURIComponent(options.params)).shipInfo
|
this.shipInfo = JSON.parse(decodeURIComponent(options.params)).shipInfo
|
||||||
console.log(this.shipInfo)
|
|
||||||
this.title = `${this.shipInfo.spmIdDesc} / ${this.shipInfo.vvyName}`
|
this.title = `${this.shipInfo.spmIdDesc} / ${this.shipInfo.vvyName}`
|
||||||
this.loginObj = uni.getStorageSync('loginObj')
|
this.loginObj = uni.getStorageSync('loginObj')
|
||||||
if (this.tabsValue == 0) {
|
if (this.tabsValue == 0) {
|
||||||
|
@ -938,6 +955,9 @@
|
||||||
HeadInfo,
|
HeadInfo,
|
||||||
place,
|
place,
|
||||||
},
|
},
|
||||||
|
onReachBottom() {
|
||||||
|
console.log(this.itemList)
|
||||||
|
},
|
||||||
methods: {
|
methods: {
|
||||||
// 点击tabs
|
// 点击tabs
|
||||||
changeTabs(e) {
|
changeTabs(e) {
|
||||||
|
@ -950,6 +970,7 @@
|
||||||
this.tabsValue = e.value
|
this.tabsValue = e.value
|
||||||
this.zTjActive = -1
|
this.zTjActive = -1
|
||||||
this.shipDeck = ""
|
this.shipDeck = ""
|
||||||
|
this.current = 1
|
||||||
if (e.value == 1 || e.value == 2) {
|
if (e.value == 1 || e.value == 2) {
|
||||||
this.loadOtherOrder()
|
this.loadOtherOrder()
|
||||||
} else if (e.value == 0) {
|
} else if (e.value == 0) {
|
||||||
|
@ -1027,24 +1048,27 @@
|
||||||
let fsum = v.branchSentCount + v.branchNotSentCount
|
let fsum = v.branchSentCount + v.branchNotSentCount
|
||||||
this.$set(v, "fsum", fsum)
|
this.$set(v, "fsum", fsum)
|
||||||
})
|
})
|
||||||
console.log(this.zTjList)
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
// 获取总指令列表
|
// 获取总指令列表
|
||||||
loadSumOrder() {
|
loadSumOrder() {
|
||||||
uni.request({
|
uni.request({
|
||||||
url: `${this.$local}/api/shipInstructions/page?vvyId=${this.shipInfo.vvyId}&brdId=${this.brdId}&mnfBl=${this.mnfBl}&potId=${this.potId}&bvmId=${this.bvmId}&shipDeck=${this.shipDeck}`,
|
url: `${this.$local}/api/shipInstructions/page?vvyId=${this.shipInfo.vvyId}&brdId=${this.brdId}&mnfBl=${this.mnfBl}&potId=${this.potId}&bvmId=${this.bvmId}&shipDeck=${this.shipDeck}&size=${this.pageSize}¤t=${this.current}`,
|
||||||
// &size=${this.pageSize}¤t=${this.current}
|
|
||||||
header: {
|
header: {
|
||||||
'Content-Type': 'application/json', //自定义请求头信息
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
'Authorization': `Bearer ${this.loginObj.access_token}`
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
},
|
},
|
||||||
method: 'GET', //请求方式,必须为大写
|
method: 'GET', //请求方式,必须为大写
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
this.itemList = res.data.data.records
|
this.itemList.push(...res.data.data.records)
|
||||||
this.total = res.data.data.total
|
if (res.data.data.records.length == 10) {
|
||||||
|
this.current++
|
||||||
|
this.loadSumOrder()
|
||||||
|
}
|
||||||
this.itemList.forEach((v, index) => {
|
this.itemList.forEach((v, index) => {
|
||||||
|
v.image = '../../static/images/theme/car1.png'
|
||||||
|
v.index = index
|
||||||
// if (v.branchPlanStatus == 0) {
|
// if (v.branchPlanStatus == 0) {
|
||||||
this.zzlLwpIdList.push(v.lwpId)
|
this.zzlLwpIdList.push(v.lwpId)
|
||||||
// }
|
// }
|
||||||
|
@ -1061,16 +1085,22 @@
|
||||||
this.sendValue = 1
|
this.sendValue = 1
|
||||||
}
|
}
|
||||||
uni.request({
|
uni.request({
|
||||||
url: `${this.$local}/api/shipInstructions/pageCommandForBranch?vvyId=${this.shipInfo.vvyId}&brdId=${this.brdId}&mnfBl=${this.mnfBl}&potId=${this.potId}&bvmId=${this.bvmId}&sendStatus=${this.sendValue}&shipDeck=${this.shipDeck}&teamFlag=${teamFlag}`, // &size=${this.pageSize}¤t=${this.current}
|
url: `${this.$local}/api/shipInstructions/pageCommandForBranch?vvyId=${this.shipInfo.vvyId}&brdId=${this.brdId}&mnfBl=${this.mnfBl}&potId=${this.potId}&bvmId=${this.bvmId}&sendStatus=${this.sendValue}&shipDeck=${this.shipDeck}&teamFlag=${teamFlag}&size=${this.pageSize}¤t=${this.current}`,
|
||||||
header: {
|
header: {
|
||||||
'Content-Type': 'application/json', //自定义请求头信息
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
'Authorization': `Bearer ${this.loginObj.access_token}`
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
},
|
},
|
||||||
method: 'GET', //请求方式,必须为大写
|
method: 'GET', //请求方式,必须为大写
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
this.itemList = res.data.data.records
|
console.log(res)
|
||||||
this.total = res.data.data.total
|
this.itemList.push(...res.data.data.records)
|
||||||
|
if (res.data.data.records.length == 10) {
|
||||||
|
this.current++
|
||||||
|
this.loadOtherOrder()
|
||||||
|
}
|
||||||
this.itemList.forEach((v, index) => {
|
this.itemList.forEach((v, index) => {
|
||||||
|
v.image = '../../static/images/theme/car1.png'
|
||||||
|
v.index = index
|
||||||
this.getBottomInfo(v.lwpId, index)
|
this.getBottomInfo(v.lwpId, index)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -1482,7 +1512,6 @@
|
||||||
// 配载图相关
|
// 配载图相关
|
||||||
// 获取配载图港口及对应颜色
|
// 获取配载图港口及对应颜色
|
||||||
getPotList() {
|
getPotList() {
|
||||||
this.shipInfo.vvyId = 'f4303b2c8b6d715f4007f961f2498b10'
|
|
||||||
uni.request({
|
uni.request({
|
||||||
url: `${this.$local}/api/stowage/portList?vvyId=${this.shipInfo.vvyId}`,
|
url: `${this.$local}/api/stowage/portList?vvyId=${this.shipInfo.vvyId}`,
|
||||||
header: {
|
header: {
|
||||||
|
@ -1496,7 +1525,7 @@
|
||||||
this.pzPotList.forEach(v => {
|
this.pzPotList.forEach(v => {
|
||||||
this.$set(v, 'background', `background:${v.potColor}`)
|
this.$set(v, 'background', `background:${v.potColor}`)
|
||||||
})
|
})
|
||||||
console.log(this.pzPotList)
|
// console.log(this.pzPotList, '配载图顶部数据, 颜色色块')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -1511,9 +1540,14 @@
|
||||||
},
|
},
|
||||||
method: 'GET', //请求方式,必须为大写
|
method: 'GET', //请求方式,必须为大写
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
console.log(res)
|
|
||||||
if (res.data.status == "200") {
|
if (res.data.status == "200") {
|
||||||
this.imgInfo = res.data.data
|
this.imgInfo = res.data.data
|
||||||
|
// 默认船舱画布高度最小为162 暂定10000 防止不画
|
||||||
|
this.imgInfo.cabinInfoList.forEach(item => {
|
||||||
|
//
|
||||||
|
item.maxHeight = 10000
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -1530,11 +1564,59 @@
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
if (res.data.status == "200") {
|
if (res.data.status == "200") {
|
||||||
this.goodsInfo = res.data.data
|
this.goodsInfo = res.data.data
|
||||||
console.log(this.goodsInfo)
|
this.goodsInfo.forEach((item, index) => {
|
||||||
|
item.stowageList.forEach(ele => {
|
||||||
|
let vertexPositionArr = ele.vertexPosition.split(',')
|
||||||
|
let y = vertexPositionArr[1] * 35
|
||||||
|
let bigNum = (vertexPositionArr[1] + ele.blockLength) * 35
|
||||||
|
if (bigNum < 162) {
|
||||||
|
bigNum = 162
|
||||||
|
}
|
||||||
|
this.$set(this.imgInfo.cabinInfoList, index, {
|
||||||
|
...this.imgInfo.cabinInfoList[index],
|
||||||
|
maxHeight: bigNum
|
||||||
|
})
|
||||||
|
})
|
||||||
|
this.initCanvas(item, index)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
initCanvas(item, index) {
|
||||||
|
let canvas = 'mycanvas' + index
|
||||||
|
const ctx = uni.createCanvasContext(canvas, this)
|
||||||
|
let sum = item.cabinNo
|
||||||
|
item.stowageList.forEach(ele => {
|
||||||
|
// 宽为850 每个仓位为20 850/20
|
||||||
|
let widthDw = (42 / sum).toFixed(2)
|
||||||
|
let vertexPositionArr = ele.vertexPosition.split(',')
|
||||||
|
let x = vertexPositionArr[0] * widthDw
|
||||||
|
let y = vertexPositionArr[1] * 35
|
||||||
|
// let bigNum = (vertexPositionArr[1] + ele.blockLength) * 35
|
||||||
|
// console.log(this.imgInfo.cabinInfoList[index]);
|
||||||
|
// if (this.imgInfo.cabinInfoList[index].maxHeight < bigNum) {
|
||||||
|
// this.$set(this.imgInfo.cabinInfoList, index, {
|
||||||
|
// ...this.imgInfo.cabinInfoList[index],
|
||||||
|
// maxHeight: bigNum
|
||||||
|
// })
|
||||||
|
// }
|
||||||
|
console.log(this.imgInfo.cabinInfoList[index].maxHeight);
|
||||||
|
ctx.setFillStyle(ele.potColor);
|
||||||
|
// 每层加一个
|
||||||
|
ctx.fillRect(x, y, widthDw * ele.blockWidth - 3, ele.blockLength * 35 - 3);
|
||||||
|
ctx.fillStyle = '#fff'
|
||||||
|
ctx.font = 'bold 18px Arial'
|
||||||
|
ctx.setFontSize(20);
|
||||||
|
let stowageNo = ''
|
||||||
|
if (ele.stowageNo) {
|
||||||
|
stowageNo = ',' + ele.stowageNo
|
||||||
|
}
|
||||||
|
ctx.fillText(ele.potName + ',' + ele.amount + stowageNo, x, y + 22, widthDw * ele
|
||||||
|
.blockWidth, ele.blockLength * 35)
|
||||||
|
})
|
||||||
|
ctx.draw();
|
||||||
|
},
|
||||||
// 场位图相关
|
// 场位图相关
|
||||||
// 点击车道
|
// 点击车道
|
||||||
toGoPlace() {
|
toGoPlace() {
|
||||||
|
@ -1553,6 +1635,10 @@
|
||||||
color: #108ee9;
|
color: #108ee9;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
page {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
.content2 {
|
.content2 {
|
||||||
background-color: #F6F7F9;
|
background-color: #F6F7F9;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
@ -1590,6 +1676,7 @@
|
||||||
|
|
||||||
.main {
|
.main {
|
||||||
height: calc(100vh - 134px);
|
height: calc(100vh - 134px);
|
||||||
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
.askBox {
|
.askBox {
|
||||||
|
@ -1853,25 +1940,35 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.itemBox {
|
.itemBox {
|
||||||
height: 465px;
|
height: 535px;
|
||||||
overflow: scroll;
|
overflow: scroll;
|
||||||
|
padding-bottom: 60px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.itemList {
|
.itemList {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
|
align-items: flex-start;
|
||||||
padding: 0 12px;
|
padding: 0 12px;
|
||||||
|
|
||||||
.exp {
|
.exp {
|
||||||
// width: 100%;
|
width: 100%;
|
||||||
width: 49.5%;
|
|
||||||
margin-bottom: 15px;
|
margin-bottom: 15px;
|
||||||
background: #FFFFFF;
|
background: #FFFFFF;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
padding: 12px;
|
padding: 12px;
|
||||||
padding-right: 0;
|
padding-right: 0;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
/deep/.waterfalls-flow {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/deep/.waterfalls-flow-column {
|
||||||
|
margin-left: 2%;
|
||||||
|
}
|
||||||
|
|
||||||
.rowHead {
|
.rowHead {
|
||||||
/deep/uni-checkbox .uni-checkbox-input {
|
/deep/uni-checkbox .uni-checkbox-input {
|
||||||
|
@ -1882,7 +1979,7 @@
|
||||||
|
|
||||||
.row {
|
.row {
|
||||||
display: flex;
|
display: flex;
|
||||||
padding: 5px 0;
|
padding: 5px 0px;
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
@ -2084,7 +2181,6 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
.pzPot {
|
.pzPot {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
@ -2145,6 +2241,7 @@
|
||||||
|
|
||||||
.imgTable {
|
.imgTable {
|
||||||
margin-top: 20px;
|
margin-top: 20px;
|
||||||
|
padding: 0 20px;
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
|
@ -2158,36 +2255,43 @@
|
||||||
|
|
||||||
.imgLi {
|
.imgLi {
|
||||||
display: flex;
|
display: flex;
|
||||||
position: relative;
|
|
||||||
|
|
||||||
.shipImg {
|
|
||||||
width: 100%;
|
|
||||||
height: 171px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.maskBox {
|
.maskBox {
|
||||||
width: calc(100% - 200px);
|
width: 1060px;
|
||||||
height: 150px;
|
height: 191px;
|
||||||
background-color: transparent;
|
margin: 0 auto;
|
||||||
display: flex;
|
background: url('@/static/images/ship-split.svg');
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-size: contain;
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
.line {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 11px;
|
width: 850px;
|
||||||
|
height: 85%;
|
||||||
|
top: 15px;
|
||||||
left: 60px;
|
left: 60px;
|
||||||
|
display: flex;
|
||||||
|
overflow-y: auto;
|
||||||
|
|
||||||
.testLine {
|
.testLine {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
height: 100%;
|
|
||||||
border-left: 1px solid #999;
|
border-left: 1px solid #999;
|
||||||
}
|
}
|
||||||
|
|
||||||
.testLine:first-child {
|
.testLine:first-child {
|
||||||
border-left: none;
|
border: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.testInfo {
|
.mycanvas {
|
||||||
height: 30px;
|
position: absolute;
|
||||||
background-color: #0067CF;
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 850px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.heightBox {
|
.heightBox {
|
||||||
|
@ -2198,36 +2302,34 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.imgSize {
|
.imgSize {
|
||||||
width: 28px;
|
width: 60px;
|
||||||
height: 28px;
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
||||||
.leftTopImg {
|
.leftTopImg {
|
||||||
transform: rotate(90deg);
|
transform: rotate(90deg);
|
||||||
top: 10px;
|
top: 5px;
|
||||||
left: 8px;
|
left: -5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.imgSize2 {
|
.imgSize2 {
|
||||||
width: 100px;
|
width: 130px;
|
||||||
height: 30px;
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
||||||
.leftBotImg {
|
.leftBotImg {
|
||||||
bottom: 10px;
|
bottom: 2px;
|
||||||
left: 8px;
|
left: -1px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.topImg {
|
.topImg {
|
||||||
top: -5px;
|
top: 0;
|
||||||
left: 40%;
|
left: 40%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.botImg {
|
.botImg {
|
||||||
transform: rotate(180deg);
|
transform: rotate(180deg);
|
||||||
bottom: -6px;
|
bottom: 0;
|
||||||
left: 40%;
|
left: 40%;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,8 +29,8 @@
|
||||||
<text>板车照片</text>
|
<text>板车照片</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="picture">
|
<view class="picture">
|
||||||
<template v-for="item in boardCarPhotos">
|
<template v-for="(item,index) in boardCarPhotos">
|
||||||
<image :key="item" :src="item"></image>
|
<image :key="item" :src="item" @click="clickImg(boardCarPhotos,index)"></image>
|
||||||
</template>
|
</template>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
@ -40,8 +40,8 @@
|
||||||
<text>板车车牌照</text>
|
<text>板车车牌照</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="picture">
|
<view class="picture">
|
||||||
<template v-for="item in boardCarList">
|
<template v-for="(item,index) in boardCarList">
|
||||||
<image :key="item" :src="item"></image>
|
<image :key="item" :src="item" @click="clickImg(boardCarList,index)"></image>
|
||||||
</template>
|
</template>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
@ -122,8 +122,8 @@
|
||||||
<text>质损照片</text>
|
<text>质损照片</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="picture">
|
<view class="picture">
|
||||||
<template v-for="item in qualityDamage">
|
<template v-for="(item,index) in qualityDamage">
|
||||||
<image :key="item" :src="item"></image>
|
<image :key="item" :src="item" @click="clickImg(qualityDamage,index)"></image>
|
||||||
</template>
|
</template>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
@ -133,8 +133,8 @@
|
||||||
<text>车架号图片</text>
|
<text>车架号图片</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="picture">
|
<view class="picture">
|
||||||
<template v-for="item in carFrameNumber">
|
<template v-for="(item,index) in carFrameNumber">
|
||||||
<image :key="item" :src="item"></image>
|
<image :key="item" :src="item" @click="clickImg(carFrameNumber,index)"></image>
|
||||||
</template>
|
</template>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
@ -147,9 +147,6 @@
|
||||||
<image :src="signImg"></image>
|
<image :src="signImg"></image>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<!-- <view class="btnList">
|
|
||||||
<van-button type="default" @click="cancel">取消</van-button>
|
|
||||||
</view> -->
|
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
|
@ -248,12 +245,13 @@
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
// 取消
|
// 点击图片放大
|
||||||
cancel() {
|
clickImg(urlList, index) {
|
||||||
uni.navigateTo({
|
uni.previewImage({
|
||||||
url: '/pages/quality/index'
|
current: index,
|
||||||
|
urls: urlList,
|
||||||
})
|
})
|
||||||
}
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
@ -367,17 +365,6 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.btnList {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
|
|
||||||
/deep/ .van-button {
|
|
||||||
margin: 30px 0;
|
|
||||||
width: 120px;
|
|
||||||
height: 50px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
|
@ -431,6 +431,7 @@
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
if (res.statusCode == 200) {
|
if (res.statusCode == 200) {
|
||||||
this.infoData = res.data.data
|
this.infoData = res.data.data
|
||||||
|
console.log(this.infoData)
|
||||||
// 获取板车照片
|
// 获取板车照片
|
||||||
this.bcPhoto = this.infoData.boardCarPhotos
|
this.bcPhoto = this.infoData.boardCarPhotos
|
||||||
this.infoData.boardCarPhotos.forEach(v => {
|
this.infoData.boardCarPhotos.forEach(v => {
|
||||||
|
@ -1307,6 +1308,7 @@
|
||||||
this.carPhoto2.push(e.tempFiles[0])
|
this.carPhoto2.push(e.tempFiles[0])
|
||||||
let data = JSON.parse(res.data).data
|
let data = JSON.parse(res.data).data
|
||||||
this.carPhoto.push(data)
|
this.carPhoto.push(data)
|
||||||
|
console.log(this.carPhoto)
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
@ -1345,6 +1347,12 @@
|
||||||
if (this.current2 != '其他') {
|
if (this.current2 != '其他') {
|
||||||
this.other = ""
|
this.other = ""
|
||||||
}
|
}
|
||||||
|
let editComplete = ""
|
||||||
|
if (this.type == 'add') {
|
||||||
|
editComplete = "1"
|
||||||
|
} else if (this.type == "edit") {
|
||||||
|
editComplete = "0"
|
||||||
|
}
|
||||||
let editReqDTO = {
|
let editReqDTO = {
|
||||||
"boardCarLicensePlates": this.bcLicense, // 板车车牌照
|
"boardCarLicensePlates": this.bcLicense, // 板车车牌照
|
||||||
"boardCarPhotos": this.bcPhoto, // 板车照片
|
"boardCarPhotos": this.bcPhoto, // 板车照片
|
||||||
|
@ -1354,7 +1362,7 @@
|
||||||
"dsOther": this.other2, // 损伤情况其他备注
|
"dsOther": this.other2, // 损伤情况其他备注
|
||||||
"dispList": this.checkData3, // 处置情况
|
"dispList": this.checkData3, // 处置情况
|
||||||
"dpsOther": this.other3, // 处置情况其他备注
|
"dpsOther": this.other3, // 处置情况其他备注
|
||||||
"editComplete": "1", // pad提交传 1 编辑完毕 扫描提交的数据不全不给后台展示
|
"editComplete": editComplete, // pad提交传 1 编辑完毕 扫描提交的数据不全不给后台展示
|
||||||
"godId": this.godId, // 货物id
|
"godId": this.godId, // 货物id
|
||||||
"goodsType": this.carId, // 车型id
|
"goodsType": this.carId, // 车型id
|
||||||
"linkFeedback": this.feed, // 客户反馈
|
"linkFeedback": this.feed, // 客户反馈
|
||||||
|
|
|
@ -376,7 +376,7 @@
|
||||||
background: #FAFAFA;
|
background: #FAFAFA;
|
||||||
border-top: 1px solid #EEEEEE;
|
border-top: 1px solid #EEEEEE;
|
||||||
border-bottom: 1px solid #EEEEEE;
|
border-bottom: 1px solid #EEEEEE;
|
||||||
padding: 6px 10px;
|
padding: 10px;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 66px;
|
top: 66px;
|
||||||
z-index: 995;
|
z-index: 995;
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
<canvas class="mycanvas" canvas-id="mycanvas" @touchstart="touchstart" @touchmove="touchmove"
|
<canvas class="mycanvas" canvas-id="mycanvas" @touchstart="touchstart" @touchmove="touchmove"
|
||||||
@touchend="touchend" disable-scroll="true"></canvas>
|
@touchend="touchend" disable-scroll="true"></canvas>
|
||||||
<view class="canvasBg">
|
<view class="canvasBg">
|
||||||
<image src="../../static/images/zs5.jpg" mode=""></image>
|
<image src="../../static/images/zs2.png" mode=""></image>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="sigh-btns">
|
<view class="sigh-btns">
|
||||||
|
@ -51,28 +51,31 @@
|
||||||
methods: {
|
methods: {
|
||||||
init() {
|
init() {
|
||||||
this.ctx = uni.createCanvasContext('mycanvas', this); //创建绘图对象
|
this.ctx = uni.createCanvasContext('mycanvas', this); //创建绘图对象
|
||||||
this.ctx.setFillStyle('#ff0000');
|
|
||||||
//设置画笔样式
|
//设置画笔样式
|
||||||
this.ctx.lineWidth = 4;
|
this.ctx.lineWidth = 4;
|
||||||
this.ctx.lineCap = 'round';
|
this.ctx.lineCap = 'round';
|
||||||
this.ctx.lineJoin = 'round';
|
this.ctx.lineJoin = 'round';
|
||||||
var that = this
|
var that = this
|
||||||
uni.getSystemInfo({
|
uni.getImageInfo({
|
||||||
success: function(res) {
|
src: '../../static/images/zs2.png',
|
||||||
that.ctx.drawImage("../../static/images/zs5.jpg", 0, 0, 1000, 222)
|
success(res1) {
|
||||||
that.width = res.windowWidth;
|
//填充白色背景图片
|
||||||
that.height = res.windowHeight;
|
that.ctx.drawImage('../../static/images/zs2.png', 0, 0, 624, 224);
|
||||||
|
let pattern = that.ctx.createPattern(img, 'no-repeat')
|
||||||
|
that.ctx.fillStyle = pattern;
|
||||||
},
|
},
|
||||||
|
fail() {
|
||||||
|
console.log("fail");
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
},
|
},
|
||||||
//触摸开始,获取到起点
|
//触摸开始,获取到起点
|
||||||
touchstart: function(e) {
|
touchstart: function(e) {
|
||||||
let startX = e.changedTouches[0].x;
|
let startX = e.changedTouches[0].x;
|
||||||
let startY = e.changedTouches[0].y;
|
let startY = e.changedTouches[0].y;
|
||||||
let startPoint = {
|
let startPoint = {
|
||||||
X: startX,
|
X: startX / 1.75,
|
||||||
Y: startY
|
Y: startY / 1.75
|
||||||
};
|
};
|
||||||
|
|
||||||
/* **************************************************
|
/* **************************************************
|
||||||
|
@ -89,8 +92,8 @@
|
||||||
let moveX = e.changedTouches[0].x;
|
let moveX = e.changedTouches[0].x;
|
||||||
let moveY = e.changedTouches[0].y;
|
let moveY = e.changedTouches[0].y;
|
||||||
let movePoint = {
|
let movePoint = {
|
||||||
X: moveX,
|
X: moveX / 1.75,
|
||||||
Y: moveY
|
Y: moveY / 1.75
|
||||||
};
|
};
|
||||||
this.points.push(movePoint); //存点
|
this.points.push(movePoint); //存点
|
||||||
let len = this.points.length;
|
let len = this.points.length;
|
||||||
|
@ -115,6 +118,8 @@
|
||||||
let point1 = this.points[0];
|
let point1 = this.points[0];
|
||||||
let point2 = this.points[1];
|
let point2 = this.points[1];
|
||||||
this.points.shift();
|
this.points.shift();
|
||||||
|
this.ctx.strokeStyle = 'red';
|
||||||
|
this.ctx.lineWidth = 2;
|
||||||
this.ctx.moveTo(point1.X, point1.Y);
|
this.ctx.moveTo(point1.X, point1.Y);
|
||||||
this.ctx.lineTo(point2.X, point2.Y);
|
this.ctx.lineTo(point2.X, point2.Y);
|
||||||
this.ctx.stroke();
|
this.ctx.stroke();
|
||||||
|
@ -130,7 +135,7 @@
|
||||||
//清空画布
|
//清空画布
|
||||||
handleReset: function() {
|
handleReset: function() {
|
||||||
var that = this
|
var that = this
|
||||||
that.ctx.clearRect(0, 0, 1000, 222);
|
that.ctx.clearRect(0, 0, 624, 224);
|
||||||
that.ctx.draw(true);
|
that.ctx.draw(true);
|
||||||
tempPoint = [];
|
tempPoint = [];
|
||||||
that.init()
|
that.init()
|
||||||
|
@ -151,30 +156,6 @@
|
||||||
uni.navigateBack({
|
uni.navigateBack({
|
||||||
delta: 1
|
delta: 1
|
||||||
});
|
});
|
||||||
// let timestamp = new Date().getTime();
|
|
||||||
// let sunumber = Math.floor(Math.random() * 999);
|
|
||||||
// var file = that.base64ToFile(that.url, timestamp + sunumber)
|
|
||||||
// uni.uploadFile({
|
|
||||||
// url: `${that.$local}/api/file/upload`, //上传图片api
|
|
||||||
// header: {
|
|
||||||
// 'Authorization': `Bearer ${that.loginObj.access_token}`
|
|
||||||
// },
|
|
||||||
// file: file,
|
|
||||||
// fileType: 'image',
|
|
||||||
// name: 'file',
|
|
||||||
// success: (res) => {
|
|
||||||
// console.log(res)
|
|
||||||
// console.log(JSON.parse(res.data))
|
|
||||||
// that.signImg = JSON.parse(res.data).data.filePath
|
|
||||||
// console.log(that.signImg)
|
|
||||||
// // uni.navigateBack({
|
|
||||||
// // delta: 1
|
|
||||||
// // });
|
|
||||||
// },
|
|
||||||
// fail: (err) => {
|
|
||||||
// console.log(err)
|
|
||||||
// }
|
|
||||||
// })
|
|
||||||
})
|
})
|
||||||
.catch(error => {
|
.catch(error => {
|
||||||
console.error(error)
|
console.error(error)
|
||||||
|
@ -216,6 +197,7 @@
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
margin-top: 48px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sign-view {
|
.sign-view {
|
||||||
|
@ -228,7 +210,7 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn {
|
.btn {
|
||||||
margin: 10px 10px 0;
|
margin: 100px 10px 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/deep/.van-button {
|
/deep/.van-button {
|
||||||
|
@ -240,17 +222,19 @@
|
||||||
.mycanvas {
|
.mycanvas {
|
||||||
margin: auto 0rpx;
|
margin: auto 0rpx;
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
margin-top: 80px;
|
margin-top: 130px;
|
||||||
width: 1000px;
|
width: 624px;
|
||||||
height: 222px;
|
height: 224px;
|
||||||
z-index: 999;
|
z-index: 999;
|
||||||
|
transform: scale(1.75)
|
||||||
}
|
}
|
||||||
|
|
||||||
.canvasBg {
|
.canvasBg {
|
||||||
width: 1000px;
|
width: 624px;
|
||||||
height: 222px;
|
height: 224px;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 80px;
|
top: 130px;
|
||||||
|
transform: scale(1.75)
|
||||||
}
|
}
|
||||||
|
|
||||||
.canvsborder {
|
.canvsborder {
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>作业日期:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>作业日期:</p>
|
||||||
|
@ -87,6 +87,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
optionData: [],
|
optionData: [],
|
||||||
|
@ -165,6 +166,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
|
@ -101,7 +101,7 @@
|
||||||
],
|
],
|
||||||
otherListh: [{
|
otherListh: [{
|
||||||
name: "安全巡检",
|
name: "安全巡检",
|
||||||
url: "",
|
url: "patrol",
|
||||||
imgUrl: "../../static/images/shipWork/aqxj.png",
|
imgUrl: "../../static/images/shipWork/aqxj.png",
|
||||||
tableName: "safetyInspectionRespList",
|
tableName: "safetyInspectionRespList",
|
||||||
},
|
},
|
||||||
|
|
|
@ -85,10 +85,10 @@
|
||||||
</template>
|
</template>
|
||||||
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
<o-empty v-else height="70vh" bg="#f5f6fa" />
|
||||||
</view>
|
</view>
|
||||||
<view class="pageBox" v-if="itemList.length > 0">
|
<!-- <view class="pageBox" v-if="itemList.length > 0">
|
||||||
<uni-pagination :show-icon="true" :total="total" :pageSize="pageSize" :current="current"
|
<uni-pagination :show-icon="true" :total="total" :pageSize="pageSize" :current="current"
|
||||||
@change="changePage" />
|
@change="changePage" />
|
||||||
</view>
|
</view> -->
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<LotusLoading :lotusLoadingData="lotusLoadingData"></LotusLoading>
|
<LotusLoading :lotusLoadingData="lotusLoadingData"></LotusLoading>
|
||||||
|
@ -133,7 +133,7 @@
|
||||||
vtpList: [],
|
vtpList: [],
|
||||||
// 分页
|
// 分页
|
||||||
total: 0,
|
total: 0,
|
||||||
pageSize: 6,
|
pageSize: 12,
|
||||||
current: 1,
|
current: 1,
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
|
@ -197,6 +197,11 @@
|
||||||
HeadInfo,
|
HeadInfo,
|
||||||
LotusLoading
|
LotusLoading
|
||||||
},
|
},
|
||||||
|
onReachBottom() {
|
||||||
|
this.current++
|
||||||
|
this.initData()
|
||||||
|
this.lotusLoadingData.isShow = true
|
||||||
|
},
|
||||||
onBackPress(options) {
|
onBackPress(options) {
|
||||||
// 触发返回就会调用此方法,这里实现的是禁用物理返回,顶部导航栏的自定义返回 uni.navigateBack 仍可使用
|
// 触发返回就会调用此方法,这里实现的是禁用物理返回,顶部导航栏的自定义返回 uni.navigateBack 仍可使用
|
||||||
if (options.from == 'backbutton') {
|
if (options.from == 'backbutton') {
|
||||||
|
@ -254,8 +259,9 @@
|
||||||
method: 'GET', //请求方式,必须为大写
|
method: 'GET', //请求方式,必须为大写
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
if (res.statusCode === 200) {
|
if (res.statusCode === 200) {
|
||||||
|
this.lotusLoadingData.isShow = false
|
||||||
this.total = res.data.data.total
|
this.total = res.data.data.total
|
||||||
this.itemList = res.data.data.records
|
this.itemList.push(...res.data.data.records)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -335,7 +341,6 @@
|
||||||
},
|
},
|
||||||
// 船舶基本信息
|
// 船舶基本信息
|
||||||
shipDataInfo(info, vtpId) {
|
shipDataInfo(info, vtpId) {
|
||||||
console.log(vtpId)
|
|
||||||
console.log(info)
|
console.log(info)
|
||||||
let date = new Date().getTime()
|
let date = new Date().getTime()
|
||||||
let webId = uuidv4()
|
let webId = uuidv4()
|
||||||
|
@ -398,6 +403,7 @@
|
||||||
vvyIds.push(item.inVvyId)
|
vvyIds.push(item.inVvyId)
|
||||||
vvyIds.push(item.outVvyId)
|
vvyIds.push(item.outVvyId)
|
||||||
}
|
}
|
||||||
|
console.log(vvyIds)
|
||||||
uni.request({
|
uni.request({
|
||||||
url: `${this.$local}/api/shipOperate/download?vvyIds=${vvyIds}`,
|
url: `${this.$local}/api/shipOperate/download?vvyIds=${vvyIds}`,
|
||||||
header: {
|
header: {
|
||||||
|
@ -994,11 +1000,18 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.content {
|
.content {
|
||||||
padding: 16px;
|
|
||||||
min-height: calc(100vh - 68px);
|
min-height: calc(100vh - 68px);
|
||||||
|
|
||||||
.form {
|
.form {
|
||||||
|
width: 100%;
|
||||||
|
background: #f5f6fa;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
height: ;
|
||||||
|
position: fixed;
|
||||||
|
top: 66px;
|
||||||
|
right: 0;
|
||||||
|
z-index: 995;
|
||||||
|
padding: 16px;
|
||||||
|
|
||||||
.select {
|
.select {
|
||||||
width: 200px;
|
width: 200px;
|
||||||
|
@ -1028,6 +1041,8 @@
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
|
padding: 16px;
|
||||||
|
margin-top: 30px;
|
||||||
|
|
||||||
/deep/.o-empty {
|
/deep/.o-empty {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>作业日期:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>作业日期:</p>
|
||||||
|
@ -128,6 +128,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
optionData: [],
|
optionData: [],
|
||||||
|
@ -211,6 +212,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>作业日期:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>作业日期:</p>
|
||||||
|
@ -72,6 +72,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
optionData: [],
|
optionData: [],
|
||||||
|
@ -148,6 +149,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>泊位:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>泊位:</p>
|
||||||
|
@ -106,6 +106,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
optionData: [],
|
optionData: [],
|
||||||
|
@ -214,6 +215,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
|
@ -12,11 +12,11 @@
|
||||||
<view class="liInfo">
|
<view class="liInfo">
|
||||||
<p>类型:{{item.type}}</text></p>
|
<p>类型:{{item.type}}</text></p>
|
||||||
<p>贸易类型:{{item.tradeTypeName}}</p>
|
<p>贸易类型:{{item.tradeTypeName}}</p>
|
||||||
<p>进出口:{{item.importExportFlagName}}</p>
|
<p>进出口:{{item.importExportName}}</p>
|
||||||
</view>
|
</view>
|
||||||
<view class="status didNot">
|
<!-- <view class="status didNot">
|
||||||
<p>未上传</p>
|
<p>未上传</p>
|
||||||
</view>
|
</view> -->
|
||||||
<!-- <view class="status success">
|
<!-- <view class="status success">
|
||||||
<p>已上传</p>
|
<p>已上传</p>
|
||||||
</view>
|
</view>
|
||||||
|
@ -38,40 +38,68 @@
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
title: "船只 - 安全巡检",
|
title: "船只 - 安全巡检",
|
||||||
|
vtpId: "",
|
||||||
shipInfo: {},
|
shipInfo: {},
|
||||||
infoList: []
|
infoList: [],
|
||||||
|
portObj: {},
|
||||||
|
loginObj: {},
|
||||||
|
vvyList: "",
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
// this.shipInfo = uni.getStorageSync('shipInfo')
|
let title = uni.getStorageSync('shipWorkTitle')
|
||||||
// this.infoList = uni.getStorageSync('addPatrolArr')
|
this.title = `${title} / 安全巡检`
|
||||||
this.executeSql1('safetyInspectionRespList')
|
this.portObj = uni.getStorageSync('portObj')
|
||||||
this.executeSql1('shipInfoTable')
|
this.loginObj = uni.getStorageSync('loginObj')
|
||||||
|
this.vtpId = uni.getStorageSync('vtpId')
|
||||||
|
this.executeSql1('voyageScheduleDataDetailRespDTOList')
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
// 查
|
// 查
|
||||||
executeSql1(tableName) {
|
executeSql1(tableName) {
|
||||||
let sql = `select * from ${tableName}`
|
let sql = `select * from ${tableName} WHERE vtpId = '${this.vtpId}'`
|
||||||
sqlite.executeSqlCeshi(sql).then((value) => {
|
sqlite.executeSqlCeshi(sql).then((value) => {
|
||||||
// 在resolve时执行的回调函数
|
// 在resolve时执行的回调函数
|
||||||
if (tableName == 'shipInfoTable') {
|
this.vvyList = value
|
||||||
this.shipInfo = value[0]
|
let vvyIds = []
|
||||||
this.title = `${this.shipInfo.vslCnname} - 安全巡检`
|
this.vvyList.forEach(v => {
|
||||||
} else {
|
vvyIds.push(v.vvyId)
|
||||||
this.infoList = value
|
})
|
||||||
}
|
this.initData(vvyIds)
|
||||||
}).catch((error) => {
|
}).catch((error) => {
|
||||||
// 在reject时执行的回调函数
|
// 在reject时执行的回调函数
|
||||||
console.error(error);
|
console.error(error);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
initData(vvyIds) {
|
||||||
|
// console.log(`pamId=${this.portObj.portId}&vvyIds=${vvyIds}`)
|
||||||
|
uni.request({
|
||||||
|
url: `${this.$local}/api/safetyInspection/page?pamId=${this.portObj.portId}&vvyIds=${vvyIds}`,
|
||||||
|
header: {
|
||||||
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
|
},
|
||||||
|
method: 'GET', //请求方式,必须为大写
|
||||||
|
success: (res) => {
|
||||||
|
this.infoList = res.data.data.records
|
||||||
|
},
|
||||||
|
fail: () => {
|
||||||
|
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
add(state, item, index) {
|
add(state, item, index) {
|
||||||
if (state != 'add') {
|
if (state != 'add') {
|
||||||
uni.setStorageSync('patrolRow', item);
|
uni.setStorageSync('patrolRow', item);
|
||||||
uni.setStorageSync('patrolRowIndex', index);
|
uni.setStorageSync('patrolRowIndex', index);
|
||||||
}
|
}
|
||||||
|
let id = ""
|
||||||
|
if (state != 'add') {
|
||||||
|
id = item.vsiId
|
||||||
|
}
|
||||||
const obj = {
|
const obj = {
|
||||||
state: state,
|
state: state,
|
||||||
|
id: id,
|
||||||
}
|
}
|
||||||
const params = encodeURIComponent(JSON.stringify(obj));
|
const params = encodeURIComponent(JSON.stringify(obj));
|
||||||
uni.navigateTo({
|
uni.navigateTo({
|
||||||
|
|
|
@ -11,32 +11,32 @@
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>航次:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>航次:</p>
|
||||||
<uni-data-select v-model="vvyId" :localdata="hcList" @change="hcChange"
|
<uni-data-select v-model="vvyId" :localdata="hcList" @change="hcChange"
|
||||||
v-if="obj.state != 'look'"></uni-data-select>
|
v-if="obj.state != 'look'"></uni-data-select>
|
||||||
<text v-else>{{vvyName}}</text>
|
<text v-else>{{patrolRow.vvyName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>进出口:</p>
|
<p>进出口:</p>
|
||||||
<text>{{shipInfo.importExportFlagName}}</text>
|
<text>{{importExportName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>类型:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>类型:</p>
|
||||||
<uni-easyinput v-if="obj.state != 'look'" v-model="type" placeholder="请输入"></uni-easyinput>
|
<uni-easyinput v-if="obj.state != 'look'" v-model="type" placeholder="请输入"></uni-easyinput>
|
||||||
<text v-else>{{type}}</text>
|
<text v-else>{{patrolRow.type}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li tpLi">
|
<view class="li tpLi">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>图片上传:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>图片上传:</p>
|
||||||
<template v-if="obj.state != 'look'">
|
<template v-if="obj.state != 'look'">
|
||||||
<view class="picture">
|
<view class="picture">
|
||||||
<uni-file-picker limit="9" v-model="urlList" fileMediatype="image" @select="select"
|
<uni-file-picker limit="9" v-model="urlList2" fileMediatype="image" @select="select"
|
||||||
@delete="delUrl" title="最多选择9张图片"></uni-file-picker>
|
@delete="delUrl" title="最多选择9张图片"></uni-file-picker>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
<view class="pictureLook" v-else>
|
<view class="pictureLook" v-else>
|
||||||
<view v-for="(item,index) in urlList" :key="index">
|
<view v-for="(item,index) in urlList2" :key="index">
|
||||||
<image :src="item.url" mode="widthFix">
|
<image :src="item.url" mode="widthFix" @click="clickImg(urlList2,index)">
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
@ -44,7 +44,7 @@
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>描述:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>描述:</p>
|
||||||
<uni-easyinput type="textarea" autoHeight v-model="remark" placeholder="请输入任务描述(200字以内)"
|
<uni-easyinput type="textarea" autoHeight v-model="remark" placeholder="请输入任务描述(200字以内)"
|
||||||
maxlength="200" v-if="obj.state != 'look'"></uni-easyinput>
|
maxlength="200" v-if="obj.state != 'look'"></uni-easyinput>
|
||||||
<text v-else>{{remark}}</text>
|
<text v-else>{{patrolRow.remark}}</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<uni-popup ref="popup" type="dialog">
|
<uni-popup ref="popup" type="dialog">
|
||||||
|
@ -79,10 +79,14 @@
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
title: "",
|
title: "",
|
||||||
|
vtpId: "",
|
||||||
|
portObj: {},
|
||||||
|
loginObj: {},
|
||||||
shipInfo: {},
|
shipInfo: {},
|
||||||
obj: {},
|
obj: {},
|
||||||
|
importExportName: "",
|
||||||
|
spmTradeName: "",
|
||||||
patrolRow: {},
|
patrolRow: {},
|
||||||
patrolRowIndex: {},
|
|
||||||
// 航次下拉框
|
// 航次下拉框
|
||||||
vvyId: "",
|
vvyId: "",
|
||||||
vvyName: "",
|
vvyName: "",
|
||||||
|
@ -91,7 +95,7 @@
|
||||||
type: "",
|
type: "",
|
||||||
// 图片
|
// 图片
|
||||||
urlList: [],
|
urlList: [],
|
||||||
ysUrl: [],
|
urlList2: [],
|
||||||
delUrlList: [],
|
delUrlList: [],
|
||||||
// 描述
|
// 描述
|
||||||
remark: "",
|
remark: "",
|
||||||
|
@ -108,6 +112,12 @@
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
that = this
|
that = this
|
||||||
|
this.portObj = uni.getStorageSync('portObj')
|
||||||
|
this.loginObj = uni.getStorageSync('loginObj')
|
||||||
|
this.vtpId = uni.getStorageSync('vtpId')
|
||||||
|
this.executeSql1('shipOption')
|
||||||
|
this.executeSql1("shipInfoTable")
|
||||||
|
this.executeSql1("voyageScheduleDataDetailRespDTOList")
|
||||||
if (this.obj.state == 'edit' || this.obj.state == 'look') {
|
if (this.obj.state == 'edit' || this.obj.state == 'look') {
|
||||||
this.getRow();
|
this.getRow();
|
||||||
}
|
}
|
||||||
|
@ -118,14 +128,11 @@
|
||||||
} else {
|
} else {
|
||||||
this.title = "新增安全巡检"
|
this.title = "新增安全巡检"
|
||||||
}
|
}
|
||||||
this.executeSql1('shipOption')
|
|
||||||
this.executeSql1("shipInfoTable")
|
|
||||||
|
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
// 查
|
// 查
|
||||||
executeSql1(tableName) {
|
executeSql1(tableName) {
|
||||||
let sql = `select * from ${tableName}`
|
let sql = `select * from ${tableName} WHERE vtpId = '${this.vtpId}'`
|
||||||
sqlite.executeSqlCeshi(sql).then((value) => {
|
sqlite.executeSqlCeshi(sql).then((value) => {
|
||||||
// 在resolve时执行的回调函数
|
// 在resolve时执行的回调函数
|
||||||
if (tableName == 'shipOption') {
|
if (tableName == 'shipOption') {
|
||||||
|
@ -133,22 +140,21 @@
|
||||||
this.getShip()
|
this.getShip()
|
||||||
} else if (tableName == 'shipInfoTable') {
|
} else if (tableName == 'shipInfoTable') {
|
||||||
this.shipInfo = value[0]
|
this.shipInfo = value[0]
|
||||||
|
} else if (tableName == 'voyageScheduleDataDetailRespDTOList') {
|
||||||
|
this.vvyInfo = value
|
||||||
|
this.hcList = []
|
||||||
|
value.forEach((v, index) => {
|
||||||
|
this.hcList.push({
|
||||||
|
text: v.vvyName,
|
||||||
|
value: v.vvyId
|
||||||
|
})
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}).catch((error) => {
|
}).catch((error) => {
|
||||||
// 在reject时执行的回调函数
|
// 在reject时执行的回调函数
|
||||||
console.error(error);
|
console.error(error);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
executeSql2(tableName) {
|
|
||||||
let sql = `select * from ${tableName} WHERE contactId = '${this.patrolRow.webId}'`
|
|
||||||
sqlite.executeSqlCeshi(sql).then((value) => {
|
|
||||||
// 在resolve时执行的回调函数
|
|
||||||
this.urlList = value
|
|
||||||
}).catch((error) => {
|
|
||||||
// 在reject时执行的回调函数
|
|
||||||
console.error(error);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
// 获取船只信息
|
// 获取船只信息
|
||||||
getShip() {
|
getShip() {
|
||||||
// this.shipInfo = uni.getStorageSync('shipInfo')
|
// this.shipInfo = uni.getStorageSync('shipInfo')
|
||||||
|
@ -168,13 +174,49 @@
|
||||||
},
|
},
|
||||||
// 获取当前行信息
|
// 获取当前行信息
|
||||||
getRow() {
|
getRow() {
|
||||||
this.patrolRow = uni.getStorageSync('patrolRow');
|
uni.request({
|
||||||
this.patrolRowIndex = uni.getStorageSync('patrolRowIndex');
|
url: `${this.$local}/api/safetyInspection/details/${this.obj.id}`,
|
||||||
this.executeSql2('safetyInspectionRespUrlList')
|
header: {
|
||||||
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
|
},
|
||||||
|
method: 'GET', //请求方式,必须为大写
|
||||||
|
success: (res) => {
|
||||||
|
this.patrolRow = res.data.data
|
||||||
this.vvyId = this.patrolRow.vvyId
|
this.vvyId = this.patrolRow.vvyId
|
||||||
this.vvyName = this.patrolRow.vvyName
|
this.vvyName = this.patrolRow.vvyName
|
||||||
|
this.importExportName = this.patrolRow.importExportName
|
||||||
|
this.spmTradeName = this.patrolRow.tradeTypeName
|
||||||
this.type = this.patrolRow.type
|
this.type = this.patrolRow.type
|
||||||
this.remark = this.patrolRow.remark
|
this.remark = this.patrolRow.remark
|
||||||
|
if (this.patrolRow.url != null) {
|
||||||
|
this.urlList = this.patrolRow.url
|
||||||
|
} else {
|
||||||
|
this.urlList = []
|
||||||
|
}
|
||||||
|
if (this.patrolRow.url.length > 0 || this.patrolRow.url != null) {
|
||||||
|
this.patrolRow.url.forEach(v => {
|
||||||
|
this.initImg(v)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
fail: (error) => {}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
initImg(fileName) {
|
||||||
|
uni.request({
|
||||||
|
url: `${this.$local}/api/file/url/?fileName=${fileName}`,
|
||||||
|
header: {
|
||||||
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
|
},
|
||||||
|
method: 'GET', //请求方式,必须为大写
|
||||||
|
success: (res) => {
|
||||||
|
if (res.statusCode == 200) {
|
||||||
|
this.urlList2.push(res.data)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
},
|
},
|
||||||
// 航次下拉
|
// 航次下拉
|
||||||
hcChange(e) {
|
hcChange(e) {
|
||||||
|
@ -184,38 +226,52 @@
|
||||||
this.vvyName = v.text
|
this.vvyName = v.text
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
this.vvyInfo.forEach(v => {
|
||||||
|
if (v.vvyId == e) {
|
||||||
|
this.importExportName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
|
}
|
||||||
|
})
|
||||||
},
|
},
|
||||||
// 图片上传
|
// 图片上传
|
||||||
select(e) {
|
select(e) {
|
||||||
pathToBase64(e.tempFiles[0].path)
|
uni.uploadFile({
|
||||||
.then(base64 => {
|
url: `${this.$local}/api/file/upload`, //上传图片api
|
||||||
compressImgBySize(base64, 1200).then(baseImg => {
|
header: {
|
||||||
// 输出图片base64
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
that.ysUrl.push(baseImg)
|
},
|
||||||
|
filePath: e.tempFilePaths[0],
|
||||||
|
name: 'file',
|
||||||
|
success: (res) => {
|
||||||
|
this.urlList2.push(e.tempFiles[0])
|
||||||
|
let data = JSON.parse(res.data).data
|
||||||
|
this.urlList.push(data.filePath)
|
||||||
|
},
|
||||||
|
fail(e) {
|
||||||
|
uni.showToast({
|
||||||
|
title: `${e.errMsg}`,
|
||||||
|
icon: 'none',
|
||||||
|
duration: 2000
|
||||||
})
|
})
|
||||||
})
|
}
|
||||||
.catch(error => {
|
|
||||||
console.error(error)
|
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
// 移除图片
|
// 移除图片
|
||||||
delUrl(e) {
|
delUrl(e) {
|
||||||
if (this.obj.state == 'edit') {
|
let delIndex = 0
|
||||||
this.delUrlList.push(e.tempFile.webId)
|
this.urlList2.forEach((v, index) => {
|
||||||
}
|
if (v.uuid == e.tempFile.uuid) {
|
||||||
pathToBase64(e.tempFile.path)
|
delIndex = index
|
||||||
.then(base64 => {
|
|
||||||
compressImgBySize(base64, 1200).then(baseImg => {
|
|
||||||
// 输出图片base64
|
|
||||||
this.ysUrl.forEach((v, index) => {
|
|
||||||
if (v == base64) {
|
|
||||||
this.ysUrl.splice(index, 1)
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
})
|
this.urlList.splice(delIndex, 1)
|
||||||
})
|
this.urlList2.splice(delIndex, 1)
|
||||||
.catch(error => {
|
},
|
||||||
console.error(error)
|
// 点击图片放大
|
||||||
|
clickImg(urlList, index) {
|
||||||
|
uni.previewImage({
|
||||||
|
current: index,
|
||||||
|
urls: urlList,
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
// 取消
|
// 取消
|
||||||
|
@ -226,15 +282,23 @@
|
||||||
},
|
},
|
||||||
// 弹框删除
|
// 弹框删除
|
||||||
delConfirm() {
|
delConfirm() {
|
||||||
for (var i = 0; i < this.urlList.length; i++) {
|
uni.request({
|
||||||
let delSql = `DELETE FROM safetyInspectionRespUrlList WHERE webId = '${this.urlList[i].webId}';`
|
url: `${this.$local}/api/safetyInspection/${this.obj.id}`,
|
||||||
this.executeSql(delSql)
|
header: {
|
||||||
}
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
let sql = `DELETE FROM safetyInspectionRespList WHERE webId = '${this.patrolRow.webId}';`
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
this.executeSql(sql)
|
},
|
||||||
|
method: 'DELETE', //请求方式,必须为大写
|
||||||
|
success: (res) => {
|
||||||
|
console.log(res)
|
||||||
|
if (res.statusCode == 200) {
|
||||||
uni.navigateTo({
|
uni.navigateTo({
|
||||||
url: '/pages/shipWork/patrol'
|
url: '/pages/shipWork/patrol'
|
||||||
})
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
fail: () => {}
|
||||||
|
})
|
||||||
},
|
},
|
||||||
// 删除
|
// 删除
|
||||||
del() {
|
del() {
|
||||||
|
@ -242,43 +306,70 @@
|
||||||
},
|
},
|
||||||
// 保存
|
// 保存
|
||||||
save() {
|
save() {
|
||||||
let date = new Date().getTime()
|
let spmTradeName = ""
|
||||||
let webId = uuidv4()
|
if (this.spmTradeName == "内贸") {
|
||||||
let webStatus = 0
|
spmTradeName = "N"
|
||||||
let webDate = api.getDate(date)
|
} else {
|
||||||
if (this.obj.state == "edit") {
|
spmTradeName = "W"
|
||||||
let sql =
|
|
||||||
`UPDATE safetyInspectionRespList SET vvyId = '${this.vvyId}', vvyName = '${this.vvyName}',
|
|
||||||
type = '${this.type}',remark = '${this.remark}', webStatus = '${webStatus}',
|
|
||||||
webDate = '${webDate}' WHERE webId = '${this.patrolRow.webId}';`
|
|
||||||
this.executeSql(sql)
|
|
||||||
if (this.delUrlList.length > 0) {
|
|
||||||
for (var i = 0; i < this.delUrlList.length; i++) {
|
|
||||||
let delSql = `DELETE FROM safetyInspectionRespUrlList WHERE webId = '${this.delUrlList[i]}';`
|
|
||||||
this.executeSql(delSql)
|
|
||||||
}
|
}
|
||||||
|
let importExportName = ""
|
||||||
|
if (this.importExportName == "进口") {
|
||||||
|
importExportName = "I"
|
||||||
|
} else {
|
||||||
|
importExportName = "E"
|
||||||
}
|
}
|
||||||
for (var i = 0; i < this.ysUrl.length; i++) {
|
let dto = {
|
||||||
let webId2 = uuidv4()
|
"importExport": importExportName,
|
||||||
let sql2 = `insert into safetyInspectionRespUrlList values('${webId2}','${this.patrolRow.webId}','${this.ysUrl[i]}',
|
"remark": this.remark,
|
||||||
'${webStatus}','${webDate}')`
|
"spmId": this.vtpId,
|
||||||
this.executeSql(sql2)
|
"tradeType": spmTradeName,
|
||||||
}
|
"type": this.type,
|
||||||
} else if (this.obj.state == "add") {
|
"uploadStatus": "",
|
||||||
let sql = `insert into safetyInspectionRespList values('${webId}','${this.vvyId}','${this.vvyName}',
|
"url": this.urlList,
|
||||||
'${this.type}','${this.remark}','${this.shipInfo.tradeTypeName}','${this.shipInfo.importExportFlagName}',
|
"vsiId": this.obj.id,
|
||||||
'${this.shipInfo.spmName}','${webStatus}','${webDate}')`
|
"vvyId": this.vvyId
|
||||||
this.executeSql(sql)
|
|
||||||
for (var i = 0; i < this.ysUrl.length; i++) {
|
|
||||||
let webId2 = uuidv4()
|
|
||||||
let sql2 = `insert into safetyInspectionRespUrlList values('${webId2}','${webId}','${this.ysUrl[i]}',
|
|
||||||
'${webStatus}','${webDate}')`
|
|
||||||
this.executeSql(sql2)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
if (this.obj.state == "add") {
|
||||||
|
uni.request({
|
||||||
|
url: `${this.$local}/api/safetyInspection`,
|
||||||
|
data: dto,
|
||||||
|
header: {
|
||||||
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
|
},
|
||||||
|
method: 'POST', //请求方式,必须为大写
|
||||||
|
success: (res) => {
|
||||||
|
if (res.statusCode == 200) {
|
||||||
uni.navigateTo({
|
uni.navigateTo({
|
||||||
url: '/pages/shipWork/patrol'
|
url: '/pages/shipWork/patrol'
|
||||||
})
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
fail: (e) => {
|
||||||
|
console.log(e)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
} else if (this.obj.state == 'edit') {
|
||||||
|
uni.request({
|
||||||
|
url: `${this.$local}/api/safetyInspection/${this.obj.id}`,
|
||||||
|
data: dto,
|
||||||
|
header: {
|
||||||
|
'Content-Type': 'application/json', //自定义请求头信息
|
||||||
|
'Authorization': `Bearer ${this.loginObj.access_token}`
|
||||||
|
},
|
||||||
|
method: 'PUT', //请求方式,必须为大写
|
||||||
|
success: (res) => {
|
||||||
|
if (res.statusCode == 200) {
|
||||||
|
uni.navigateTo({
|
||||||
|
url: '/pages/shipWork/patrol'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
fail: (e) => {
|
||||||
|
console.log(e)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
},
|
},
|
||||||
// 编辑
|
// 编辑
|
||||||
toGo(state) {
|
toGo(state) {
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>天气:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>天气:</p>
|
||||||
|
@ -131,6 +131,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
optionData: [],
|
optionData: [],
|
||||||
|
@ -250,6 +251,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
|
@ -19,7 +19,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required">*</text>装卸类型:</p>
|
<p><text class="required">*</text>装卸类型:</p>
|
||||||
|
@ -311,6 +311,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
contactId: "",
|
contactId: "",
|
||||||
aId: "",
|
aId: "",
|
||||||
|
@ -625,6 +626,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>泊位:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>泊位:</p>
|
||||||
|
@ -103,6 +103,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
optionData: [],
|
optionData: [],
|
||||||
|
@ -211,6 +212,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
|
@ -25,7 +25,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>泊位:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>泊位:</p>
|
||||||
|
@ -120,6 +120,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
optionData: [],
|
optionData: [],
|
||||||
|
@ -227,6 +228,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p>贸易类型:</p>
|
<p>贸易类型:</p>
|
||||||
<text>{{shipInfo.spmTradeName}}</text>
|
<text>{{spmTradeName}}</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="li">
|
<view class="li">
|
||||||
<p><text class="required" v-if="obj.state != 'look'">*</text>泊位:</p>
|
<p><text class="required" v-if="obj.state != 'look'">*</text>泊位:</p>
|
||||||
|
@ -125,6 +125,7 @@
|
||||||
|
|
||||||
vvyInfo: [],
|
vvyInfo: [],
|
||||||
importExportFlagName: "",
|
importExportFlagName: "",
|
||||||
|
spmTradeName: "",
|
||||||
|
|
||||||
// 下拉数据
|
// 下拉数据
|
||||||
optionData: [],
|
optionData: [],
|
||||||
|
@ -251,6 +252,7 @@
|
||||||
this.vvyInfo.forEach(v => {
|
this.vvyInfo.forEach(v => {
|
||||||
if (v.vvyId == e) {
|
if (v.vvyId == e) {
|
||||||
this.importExportFlagName = v.importExportFlagName
|
this.importExportFlagName = v.importExportFlagName
|
||||||
|
this.spmTradeName = v.tradeTypeName
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
Before Width: | Height: | Size: 65 KiB After Width: | Height: | Size: 30 KiB |
After Width: | Height: | Size: 159 KiB |
|
@ -0,0 +1,17 @@
|
||||||
|
## 1.0.7(2022-05-26)
|
||||||
|
1. 优化局部改变数据更新问题,避免重新加载数据,只改变局部
|
||||||
|
## 1.0.6(2022-04-18)
|
||||||
|
1. 修改tab快速切换时会出现的BUG
|
||||||
|
## 1.0.5(2022-04-18)
|
||||||
|
1. 修复可能存在数据错误的BUG;
|
||||||
|
2. 兼容,今后可以无需调用refresh()就可以更新数据;
|
||||||
|
## 1.0.4(2022-04-18)
|
||||||
|
1. 修复BUG;
|
||||||
|
## 1.0.3(2022-04-15)
|
||||||
|
1. 优化代码;
|
||||||
|
2. 修改懒加载数据存在的BUG;
|
||||||
|
## 1.0.1(2022-03-11)
|
||||||
|
1. 增加隐藏图片字段的键名字段hideImageKey,默认hide
|
||||||
|
2. 支持在列表中配置hide参数进行隐藏图片
|
||||||
|
## 1.0.0(2022-03-09)
|
||||||
|
使用最简单的思想实现瀑布流
|
|
@ -0,0 +1,323 @@
|
||||||
|
<template>
|
||||||
|
<view class="waterfalls-flow">
|
||||||
|
<view v-for="(item,index) in data.column" :key="index" class="waterfalls-flow-column" :id="`waterfalls_flow_column_${index+1}`" :msg="msg" :style="{'width':w,'margin-left':index==0?0:m}">
|
||||||
|
<view :class="['column-value',{'column-value-show':item2.o}]" v-for="(item2,index2) in columnValue(index)" :key="index2" :style="[s1]" @click.stop="wapperClick(item2)">
|
||||||
|
<view class="inner" v-if="data.seat==1">
|
||||||
|
<!-- #ifdef MP-WEIXIN -->
|
||||||
|
<!-- #ifdef VUE2 -->
|
||||||
|
<slot name="slot{{item2.index}}"></slot>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifdef VUE3 -->
|
||||||
|
<slot :name="`slot${item2.index}`"></slot>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifndef MP-WEIXIN -->
|
||||||
|
<slot v-bind="item2"></slot>
|
||||||
|
<!-- #endif -->
|
||||||
|
</view>
|
||||||
|
<image :class="['img',{'img-hide':item2[hideImageKey]==true||item2[hideImageKey]==1},{'img-error':!item2[data.imageKey]}]" :src="item2[data.imageKey]" mode="widthFix" @load="imgLoad(item2,index+1)" @error="imgError(item2,index+1)" @click.stop="imageClick(item2)"></image>
|
||||||
|
<view class="inner" v-if="data.seat==2">
|
||||||
|
<!-- #ifdef MP-WEIXIN -->
|
||||||
|
<!-- #ifdef VUE2 -->
|
||||||
|
<slot name="slot{{item2.index}}"></slot>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifdef VUE3 -->
|
||||||
|
<slot :name="`slot${item2.index}`"></slot>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifndef MP-WEIXIN -->
|
||||||
|
<slot v-bind="item2"></slot>
|
||||||
|
<!-- #endif -->
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
value: Array,
|
||||||
|
column: { // 列的数量
|
||||||
|
type: [String, Number],
|
||||||
|
default: 2
|
||||||
|
},
|
||||||
|
maxColumn: { // 最大列数
|
||||||
|
type: [String, Number],
|
||||||
|
default: 5
|
||||||
|
},
|
||||||
|
columnSpace: { // 列之间的间距 百分比
|
||||||
|
type: [String, Number],
|
||||||
|
default: 2
|
||||||
|
},
|
||||||
|
imageKey: { // 图片key
|
||||||
|
type: [String],
|
||||||
|
default: 'image'
|
||||||
|
},
|
||||||
|
hideImageKey: { // 隐藏图片key
|
||||||
|
type: [String],
|
||||||
|
default: 'hide'
|
||||||
|
},
|
||||||
|
seat: { // 文本的位置,1图片之上 2图片之下
|
||||||
|
type: [String, Number],
|
||||||
|
default: 2
|
||||||
|
},
|
||||||
|
listStyle: { // 单个展示项的样式:eg:{'background':'red'}
|
||||||
|
type: Object
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
data: {
|
||||||
|
list: this.value ? this.value : [],
|
||||||
|
column: this.column < 2 ? 2 : this.column,
|
||||||
|
columnSpace: this.columnSpace <= 5 ? this.columnSpace : 5,
|
||||||
|
imageKey: this.imageKey,
|
||||||
|
seat: this.seat
|
||||||
|
},
|
||||||
|
msg: 0,
|
||||||
|
listInitStyle: {
|
||||||
|
'border-radius': '12rpx',
|
||||||
|
'margin-bottom': '20rpx',
|
||||||
|
'background-color': '#fff'
|
||||||
|
},
|
||||||
|
adds: [], //预置数据
|
||||||
|
isLoaded: true,
|
||||||
|
curIndex: 0,
|
||||||
|
isRefresh: true,
|
||||||
|
flag: false,
|
||||||
|
refreshDatas: []
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
// 计算列宽
|
||||||
|
w() {
|
||||||
|
const column_rate = `${100 / this.data.column - (+this.data.columnSpace)}%`;
|
||||||
|
return column_rate;
|
||||||
|
},
|
||||||
|
// 计算margin
|
||||||
|
m() {
|
||||||
|
const column_margin = `${(100-(100 / this.data.column - (+this.data.columnSpace)).toFixed(5)*this.data.column)/(this.data.column-1)}%`;
|
||||||
|
return column_margin;
|
||||||
|
},
|
||||||
|
// list样式
|
||||||
|
s1() {
|
||||||
|
return { ...this.listInitStyle, ...this.listStyle };
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
// 初始化
|
||||||
|
this.refresh();
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
// 预加载图片
|
||||||
|
loadImages(idx = 0) {
|
||||||
|
let count = 0;
|
||||||
|
const newList = this.data.list.filter((item, index) => index >= idx);
|
||||||
|
for (let i = 0; i < newList.length; i++) {
|
||||||
|
// #ifndef APP-PLUS
|
||||||
|
uni.getImageInfo({
|
||||||
|
src: `${newList[i][this.imageKey]}.jpg`,
|
||||||
|
complete: res => {
|
||||||
|
count++;
|
||||||
|
if (count == newList.length) this.initValue(idx);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
// #ifdef APP-PLUS
|
||||||
|
plus.io.getImageInfo({
|
||||||
|
src: `${newList[i][this.imageKey]}.jpg`,
|
||||||
|
complete: res => {
|
||||||
|
count++;
|
||||||
|
if (count == newList.length) this.initValue(idx);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// 刷新
|
||||||
|
refresh() {
|
||||||
|
if (!this.isLoaded) {
|
||||||
|
this.refreshDatas = this.value;
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
setTimeout(() => {
|
||||||
|
this.refreshDatas = [];
|
||||||
|
this.isRefresh = true;
|
||||||
|
this.adds = [];
|
||||||
|
this.data.list = this.value ? this.value : [];
|
||||||
|
this.data.column = this.column < 2 ? 2 : this.column >= this.maxColumn ? this.maxColumn : this.column;
|
||||||
|
this.data.columnSpace = this.columnSpace <= 5 ? this.columnSpace : 5;
|
||||||
|
this.data.imageKey = this.imageKey;
|
||||||
|
this.data.seat = this.seat;
|
||||||
|
this.curIndex = 0;
|
||||||
|
// 每列的数据初始化
|
||||||
|
for (let i = 1; i <= this.data.column; i++) {
|
||||||
|
this.data[`column_${i}_values`] = [];
|
||||||
|
this.msg++;
|
||||||
|
}
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.initValue(this.curIndex, 'refresh==>');
|
||||||
|
})
|
||||||
|
}, 1)
|
||||||
|
},
|
||||||
|
columnValue(index) {
|
||||||
|
return this.data[`column_${index+1}_values`];
|
||||||
|
},
|
||||||
|
change(newValue) {
|
||||||
|
for (let i = 0; i < this.data.list.length; i++) {
|
||||||
|
const cv = this.data[`column_${this.data.list[i].column}_values`];
|
||||||
|
for (let j = 0; j < cv.length; j++) {
|
||||||
|
if (newValue[i] && i === cv[j].index) {
|
||||||
|
this.data[`column_${this.data.list[i].column}_values`][j] = Object.assign(cv[j], newValue[i]);
|
||||||
|
this.msg++;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getMin(a, s) {
|
||||||
|
let m = a[0][s];
|
||||||
|
let mo = a[0];
|
||||||
|
for (var i = a.length - 1; i >= 0; i--) {
|
||||||
|
if (a[i][s] < m) {
|
||||||
|
m = a[i][s];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
mo = a.filter(i => i[s] == m);
|
||||||
|
return mo[0];
|
||||||
|
},
|
||||||
|
// 计算每列的高度
|
||||||
|
getMinColumnHeight() {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
const heightArr = [];
|
||||||
|
for (let i = 1; i <= this.data.column; i++) {
|
||||||
|
const query = uni.createSelectorQuery().in(this);
|
||||||
|
query.select(`#waterfalls_flow_column_${i}`).boundingClientRect(data => {
|
||||||
|
heightArr.push({ column: i, height: data.height });
|
||||||
|
}).exec(() => {
|
||||||
|
if (this.data.column <= heightArr.length) {
|
||||||
|
resolve(this.getMin(heightArr, 'height'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
async initValue(i, from) {
|
||||||
|
this.isLoaded = false;
|
||||||
|
if (i >= this.data.list.length || this.refreshDatas.length) {
|
||||||
|
this.msg++;
|
||||||
|
this.loaded();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const minHeightRes = await this.getMinColumnHeight();
|
||||||
|
const c = this.data[`column_${minHeightRes.column}_values`];
|
||||||
|
this.data.list[i].column = minHeightRes.column;
|
||||||
|
c.push({ ...this.data.list[i], cIndex: c.length, index: i, o: 0 });
|
||||||
|
this.msg++;
|
||||||
|
},
|
||||||
|
// 图片加载完成
|
||||||
|
imgLoad(item, c) {
|
||||||
|
const i = item.index;
|
||||||
|
item.o = 1;
|
||||||
|
this.$set(this.data[`column_${c}_values`], item.cIndex, JSON.parse(JSON.stringify(item)));
|
||||||
|
this.initValue(i + 1);
|
||||||
|
},
|
||||||
|
// 图片加载失败
|
||||||
|
imgError(item, c) {
|
||||||
|
const i = item.index;
|
||||||
|
item.o = 1;
|
||||||
|
item[this.data.imageKey] = null;
|
||||||
|
this.$set(this.data[`column_${c}_values`], item.cIndex, JSON.parse(JSON.stringify(item)));
|
||||||
|
this.initValue(i + 1);
|
||||||
|
},
|
||||||
|
// 渲染结束
|
||||||
|
loaded() {
|
||||||
|
if (this.refreshDatas.length) {
|
||||||
|
this.isLoaded = true;
|
||||||
|
this.refresh();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
this.curIndex = this.data.list.length;
|
||||||
|
if (this.adds.length) {
|
||||||
|
this.data.list = this.adds[0];
|
||||||
|
this.adds.splice(0, 1);
|
||||||
|
this.initValue(this.curIndex);
|
||||||
|
} else {
|
||||||
|
if (this.data.list.length) this.$emit('loaded');
|
||||||
|
this.isLoaded = true;
|
||||||
|
this.isRefresh = false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// 单项点击事件
|
||||||
|
wapperClick(item) {
|
||||||
|
this.$emit('wapperClick', item);
|
||||||
|
},
|
||||||
|
// 图片点击事件
|
||||||
|
imageClick(item) {
|
||||||
|
this.$emit('imageClick', item);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
value: {
|
||||||
|
deep: true,
|
||||||
|
handler(newValue, oldValue) {
|
||||||
|
setTimeout(() => {
|
||||||
|
this.$nextTick(() => {
|
||||||
|
if (this.isRefresh) return false;
|
||||||
|
if (this.isLoaded) {
|
||||||
|
// if (newValue.length <= this.curIndex) return this.refresh();
|
||||||
|
if (newValue.length <= this.curIndex) return this.change(newValue);
|
||||||
|
this.data.list = newValue;
|
||||||
|
this.$nextTick(() => {
|
||||||
|
this.initValue(this.curIndex, 'watch==>');
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
this.adds.push(newValue);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, 10)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
column(newValue) {
|
||||||
|
this.refresh();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.waterfalls-flow {
|
||||||
|
overflow: hidden;
|
||||||
|
|
||||||
|
&-column {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.column-value {
|
||||||
|
width: 100%;
|
||||||
|
font-size: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
transition: opacity .4s;
|
||||||
|
opacity: 0;
|
||||||
|
|
||||||
|
&-show {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.inner {
|
||||||
|
font-size: 30rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.img {
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
&-hide {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
&-error {
|
||||||
|
background: #f2f2f2 url() no-repeat center center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,80 @@
|
||||||
|
{
|
||||||
|
"id": "custom-waterfalls-flow",
|
||||||
|
"displayName": "瀑布流 灵活配置 简单易用 兼容vue2vue3小程序、H5、app等多端",
|
||||||
|
"version": "1.0.7",
|
||||||
|
"description": "瀑布流,根据内容自动计算进行流式布局,简单参数配置,实现兼容多端及vue2和vue3的瀑布流布局;uv-ui发布https://ext.dcloud.net.cn/plugin?name=uv-ui",
|
||||||
|
"keywords": [
|
||||||
|
"瀑布流",
|
||||||
|
"瀑布流式布局"
|
||||||
|
],
|
||||||
|
"repository": "https://gitee.com/my_dear_li_pan/my-uni-modules.git",
|
||||||
|
"engines": {
|
||||||
|
},
|
||||||
|
"dcloudext": {
|
||||||
|
"category": [
|
||||||
|
"前端组件",
|
||||||
|
"通用组件"
|
||||||
|
],
|
||||||
|
"sale": {
|
||||||
|
"regular": {
|
||||||
|
"price": "0.00"
|
||||||
|
},
|
||||||
|
"sourcecode": {
|
||||||
|
"price": "0.00"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"contact": {
|
||||||
|
"qq": ""
|
||||||
|
},
|
||||||
|
"declaration": {
|
||||||
|
"ads": "无",
|
||||||
|
"data": "插件不采集任何数据",
|
||||||
|
"permissions": "无"
|
||||||
|
},
|
||||||
|
"npmurl": ""
|
||||||
|
},
|
||||||
|
"uni_modules": {
|
||||||
|
"dependencies": [],
|
||||||
|
"encrypt": [],
|
||||||
|
"platforms": {
|
||||||
|
"cloud": {
|
||||||
|
"tcb": "y",
|
||||||
|
"aliyun": "y"
|
||||||
|
},
|
||||||
|
"client": {
|
||||||
|
"Vue": {
|
||||||
|
"vue2": "y",
|
||||||
|
"vue3": "y"
|
||||||
|
},
|
||||||
|
"App": {
|
||||||
|
"app-vue": "y",
|
||||||
|
"app-nvue": "n"
|
||||||
|
},
|
||||||
|
"H5-mobile": {
|
||||||
|
"Safari": "y",
|
||||||
|
"Android Browser": "y",
|
||||||
|
"微信浏览器(Android)": "y",
|
||||||
|
"QQ浏览器(Android)": "y"
|
||||||
|
},
|
||||||
|
"H5-pc": {
|
||||||
|
"Chrome": "y",
|
||||||
|
"IE": "u",
|
||||||
|
"Edge": "u",
|
||||||
|
"Firefox": "y",
|
||||||
|
"Safari": "u"
|
||||||
|
},
|
||||||
|
"小程序": {
|
||||||
|
"微信": "y",
|
||||||
|
"阿里": "u",
|
||||||
|
"百度": "y",
|
||||||
|
"字节跳动": "y",
|
||||||
|
"QQ": "u"
|
||||||
|
},
|
||||||
|
"快应用": {
|
||||||
|
"华为": "u",
|
||||||
|
"联盟": "u"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,445 @@
|
||||||
|
- <a href="#c1" title="概要">概要</a>
|
||||||
|
- <a href="#c2" title="支持的平台">支持的平台</a>
|
||||||
|
- <a href="#c3" title="使用方式">使用方式</a>
|
||||||
|
- <a href="#c4" title="属性说明">属性说明</a>
|
||||||
|
- <a href="#c5" title="事件说明">事件说明</a>
|
||||||
|
- <a href="#c6" title="组件方法">组件方法</a>
|
||||||
|
- <a href="#c7" title="refresh的使用示例">refresh的使用示例</a>
|
||||||
|
- <a href="#c8" title="隐藏单项图片示例">隐藏单项图片示例</a>
|
||||||
|
- <a href="#c9" title="完整示例">完整示例</a>
|
||||||
|
- <a href="#c10" title="温馨提示">温馨提示</a>
|
||||||
|
- <a href="#c11" title="关注我,不迷路">关注我,不迷路</a>
|
||||||
|
- <a href="#c12" title="个人作品展示">个人作品展示</a>
|
||||||
|
|
||||||
|
<div id="c1"></div>
|
||||||
|
|
||||||
|
#### 概要
|
||||||
|
|
||||||
|
custom-waterfalls-flow是一个瀑布流插件,灵活配置、简单易用、兼容多端、同时兼容vue2和vue3。
|
||||||
|
|
||||||
|
最近在做项目的时候需要用到瀑布流,于是在插件市场找了一些,下载量最高的是用了定位来做的,我认为瀑布流可以不用定位去实现,于是我就自己写了该插件。经过反复的测试优化,最终搞定!
|
||||||
|
|
||||||
|
**设置列数:** 瀑布流的列数可以通过参数直接控制,实时监听,随改随生效。列数最小为2,最大默认为5,可以通过maxColumn参数去控制最大列数,理论上可以设置无限大,具体值自己拿捏。
|
||||||
|
|
||||||
|
**更新数据:** 瀑布流的每项数据,可以直接通过修改value,随改随生效,这样可以实现加载更多数据。已经渲染过的数据不会再次渲染,每次只会渲染新增的数据,这样避免了数据越多渲染越慢的情况。可以调用组件的```refresh()```方法进行数据刷新,注意vue2和vue3中调用子组件的方法有区别,也会在下面进行说明。
|
||||||
|
|
||||||
|
**展示方式:** 瀑布流可以是纯图片,可以使用插槽自定义文字描述,微信小程序与app、h5使用会有些区别,也会在下面具体说明。内容高度及排序都不用担心,会根据每项的内容高度自动计算。
|
||||||
|
|
||||||
|
**实现思路:** 通过配置列数,先渲染出每列,再计算每列的高度,最小的那列就加入一条数据进行渲染,然后再重复计算每列,高度小的加入数据...其实思路是很简单的。
|
||||||
|
|
||||||
|
uniapp插件市场地址:[https://ext.dcloud.net.cn/plugin?id=7594](https://ext.dcloud.net.cn/plugin?id=7594)
|
||||||
|
|
||||||
|
<div id="c2"></div>
|
||||||
|
|
||||||
|
#### 支持的平台
|
||||||
|
|
||||||
|
H5、app、微信小程序(这三个平台经过反复测试优化,兼容vue2和vue3)。
|
||||||
|
|
||||||
|
百度小程序:由于插槽不能循环渲染的限制,只支持纯图片瀑布流。
|
||||||
|
|
||||||
|
其他小程序:暂未测试,需要的可以自己测试和修改,思路肯定是没错的,主要是兼容插槽的问题。
|
||||||
|
|
||||||
|
nvue:暂不支持,后期可能会支持,目前需要的可以自己修改源码。
|
||||||
|
|
||||||
|
<div id="c3"></div>
|
||||||
|
|
||||||
|
#### 使用方式
|
||||||
|
|
||||||
|
**1、导入插件**
|
||||||
|
|
||||||
|
该组件符合uni_modules规范,使用Hbuilderx导入插件,导入到项目根目录中的uni_modules文件夹中。
|
||||||
|
|
||||||
|
**2、template中使用**
|
||||||
|
|
||||||
|
uni_modules规范在项目页面中直接使用,不需要单独引入注册组件。
|
||||||
|
|
||||||
|
***纯图片瀑布流使用***
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<custom-waterfalls-flow :value="data.list"></custom-waterfalls-flow>
|
||||||
|
</template>
|
||||||
|
```
|
||||||
|
|
||||||
|
***微信小程序自定义内容使用***
|
||||||
|
|
||||||
|
微信小程序没有动态模板,使用for循环的方式进行渲染。
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<custom-waterfalls-flow :value="data.list">
|
||||||
|
<view class="item" v-for="(item,index) in data.list" :key="index" slot="slot{{index}}">
|
||||||
|
<view class="title">{{item.title}}</view>
|
||||||
|
<view class="desc">{{item.desc}}</view>
|
||||||
|
</view>
|
||||||
|
</custom-waterfalls-flow>
|
||||||
|
</template>
|
||||||
|
```
|
||||||
|
|
||||||
|
***h5、app端自定义内容使用***
|
||||||
|
|
||||||
|
使用作用域插槽实现
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<custom-waterfalls-flow :value="data.list">
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="item">
|
||||||
|
<view class="title">{{item.title}}</view>
|
||||||
|
<view class="desc">{{item.desc}}</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
</custom-waterfalls-flow>
|
||||||
|
</template>
|
||||||
|
```
|
||||||
|
|
||||||
|
***小程序、h5、app等多端自定义内容使用***
|
||||||
|
|
||||||
|
条件渲染-多端同时兼容
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<custom-waterfalls-flow :value="data.list">
|
||||||
|
<!-- #ifdef MP-WEIXIN -->
|
||||||
|
<view class="item" v-for="(item,index) in data.list" :key="index" slot="slot{{index}}">
|
||||||
|
<view class="title">{{item.title}}</view>
|
||||||
|
<view class="desc">{{item.desc}}</view>
|
||||||
|
</view>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifndef MP-WEIXIN -->
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="item">
|
||||||
|
<view class="title">{{item.title}}</view>
|
||||||
|
<view class="desc">{{item.desc}}</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<!-- #endif -->
|
||||||
|
</custom-waterfalls-flow>
|
||||||
|
</template>
|
||||||
|
```
|
||||||
|
|
||||||
|
<div id="c4"></div>
|
||||||
|
|
||||||
|
#### 属性说明
|
||||||
|
|
||||||
|
参数|说明|类型|是否必填|可选值|默认值
|
||||||
|
-|-|-|-|-|-|
|
||||||
|
value|渲染的列表|Array|是|-|-
|
||||||
|
column|列数|Number|否|2-maxColumn|2
|
||||||
|
maxColumn|最大列数|Number|否|>2|5
|
||||||
|
columnSpace|列之间的间距(单位是百分比)|Number|否|-|2
|
||||||
|
imageKey|列表中的图片字段的键名|String|否|-|image
|
||||||
|
hideImageKey|隐藏图片字段的键名|String|否|-|hide
|
||||||
|
seat|自定义文字的位置,1-图片上方,2-图片下方|Number|否|1/2|2
|
||||||
|
listStyle|单个展示项的样式|Object|否|示例:```{'background':'red'}```|-
|
||||||
|
|
||||||
|
<div id="c5"></div>
|
||||||
|
|
||||||
|
#### 事件说明
|
||||||
|
|
||||||
|
事件名称|说明|回调参数
|
||||||
|
-|-|-|
|
||||||
|
@loaded|图片加载完成事件|-
|
||||||
|
@wapperClick|单项点击事件|单项对应参数
|
||||||
|
@imageClick|图片点击事件|单项对应参数
|
||||||
|
|
||||||
|
<div id="c6"></div>
|
||||||
|
|
||||||
|
#### 组件方法
|
||||||
|
|
||||||
|
事件名称|说明|参数|使用场景
|
||||||
|
-|-|-|-
|
||||||
|
refresh|刷新数据,数据初始化,vue2中使用:```this.$refs.waterfallsFlowRef.refresh();```;vue3中使用:```const waterfallsFlowRef = ref(null);waterfallsFlowRef.value.refresh();```|-|下拉刷新等
|
||||||
|
|
||||||
|
<div id="c7"></div>
|
||||||
|
|
||||||
|
#### refresh的使用示例
|
||||||
|
|
||||||
|
***vue2中使用***
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<button class="btn" type="default" @click="reset()">刷新数据</button>
|
||||||
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="data.list"></custom-waterfalls-flow>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
data:{
|
||||||
|
list: [
|
||||||
|
{ image: 'https://via.placeholder.com/200x500.png/ff0000', title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x200.png/2878ff', title: '我是标题2', desc: '描述描述描述描述描述描述描述描述2' }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
reset(){
|
||||||
|
this.data.list = [{ image: 'https://via.placeholder.com/200x500.png/ff0000', title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' }]
|
||||||
|
this.$refs.waterfallsFlowRef.refresh();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
```
|
||||||
|
|
||||||
|
***vue3中使用***
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<button class="btn" type="default" @click="reset()">刷新数据</button>
|
||||||
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="data.list"></custom-waterfalls-flow>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import { reactive, ref } from 'vue';
|
||||||
|
const data = reactive({
|
||||||
|
list: [
|
||||||
|
{ image: 'https://via.placeholder.com/200x500.png/ff0000', title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x200.png/2878ff', title: '我是标题2', desc: '描述描述描述描述描述描述描述描述2' }
|
||||||
|
]
|
||||||
|
});
|
||||||
|
const waterfallsFlowRef = ref(null);
|
||||||
|
function reset(){
|
||||||
|
data.list = [{ image: 'https://via.placeholder.com/200x500.png/ff0000', title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' }]
|
||||||
|
waterfallsFlowRef.value.refresh();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
```
|
||||||
|
|
||||||
|
<div id="c8"></div>
|
||||||
|
|
||||||
|
#### 隐藏单项图片示例
|
||||||
|
|
||||||
|
在数据列表中配置```hide:true```或者```hide:1```,就可以达到不显示图片的效果。支持使用参数hideImageKey自定义键名称,那就使用:```定义的键名称:true```或者```定义的键名称:1```。
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<custom-waterfalls-flow :value="data.list">
|
||||||
|
<!-- #ifdef MP-WEIXIN -->
|
||||||
|
<view class="item" v-for="(item,index) in data.list" :key="index" slot="slot{{index}}">
|
||||||
|
<view class="title">{{item.title}}</view>
|
||||||
|
<view class="desc">{{item.desc}}</view>
|
||||||
|
</view>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifndef MP-WEIXIN -->
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="item">
|
||||||
|
<view class="title">{{item.title}}</view>
|
||||||
|
<view class="desc">{{item.desc}}</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<!-- #endif -->
|
||||||
|
</custom-waterfalls-flow>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
import { reactive, ref } from 'vue';
|
||||||
|
const data = reactive({
|
||||||
|
list: [
|
||||||
|
{ image: 'https://via.placeholder.com/200x500.png/ff0000',
|
||||||
|
hide:1,title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x200.png/2878ff', title: '我是标题2', desc: '描述描述描述描述描述描述描述描述2' }
|
||||||
|
]
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
```
|
||||||
|
|
||||||
|
<div id="c9"></div>
|
||||||
|
|
||||||
|
#### 完整示例
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<view style="padding: 0 10rpx;">
|
||||||
|
<view class="handle">
|
||||||
|
<button class="btn" type="default" @click="add()">增加数据</button>
|
||||||
|
<button class="btn" type="default" @click="changeColumn(1)">+列数({{column}})</button>
|
||||||
|
<button class="btn" type="default" @click="changeColumn(0)">-列数({{column}})</button>
|
||||||
|
<button class="btn" type="default" @click="reset()">刷新数据</button>
|
||||||
|
</view>
|
||||||
|
<custom-waterfalls-flow ref="waterfallsFlowRef" :value="data.list" :column="column" :columnSpace="1.5" :seat="2" @wapperClick="wapperClick" @imageClick="imageClick" @loaded="loaded">
|
||||||
|
<!-- #ifdef MP-WEIXIN -->
|
||||||
|
<view class="item" v-for="(item,index) in data.list" :key="index" slot="slot{{index}}">
|
||||||
|
<view class="title">{{item.title}}</view>
|
||||||
|
<view class="desc">{{item.desc}}</view>
|
||||||
|
</view>
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifndef MP-WEIXIN -->
|
||||||
|
<template v-slot:default="item">
|
||||||
|
<view class="item">
|
||||||
|
<view class="title">{{item.title}}</view>
|
||||||
|
<view class="desc">{{item.desc}}</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<!-- #endif -->
|
||||||
|
</custom-waterfalls-flow>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
<script setup>
|
||||||
|
// #ifdef VUE3
|
||||||
|
import { reactive, ref, onMounted } from 'vue';
|
||||||
|
const data = reactive({
|
||||||
|
list: [{ image: 'https://via.placeholder.com/200x500.png/ff0000', title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x200.png/2878ff', title: '我是标题2', desc: '描述描述描述描述描述描述描述描述2' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x100.png/FFB6C1', title: '我是标题3', desc: '描述描述描述描述描述描述描述描述3' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x300.png/9400D3', title: '我是标题4', desc: '描述描述描述描述描述描述描述描述4' },
|
||||||
|
{ image: 'https://via.placeholder.com/100x240.png/B0E0E6', title: '我是标题5', desc: '描述描述描述描述描述描述描述描述5' },
|
||||||
|
{ image: 'https://via.placeholder.com/140x280.png/7FFFAA', title: '我是标题6', desc: '描述描述描述描述描述描述描述描述6' },
|
||||||
|
{ image: 'https://via.placeholder.com/40x60.png/EEE8AA', title: '我是标题7', desc: '描述描述描述描述描述描述描述描述7' }]
|
||||||
|
});
|
||||||
|
const column = ref(3);
|
||||||
|
|
||||||
|
function add() {
|
||||||
|
const newArr = [{ image: 'https://via.placeholder.com/58x100.png/FF7F50', title: '我是标题8', desc: '描述描述描述描述描述描述描述描述8' },
|
||||||
|
{ image: 'https://via.placeholder.com/59x100.png/C0C0C0', title: '我是标题9', desc: '描述描述描述描述描述描述描述描述9' },
|
||||||
|
{ image: 'https://via.placeholder.com/60x100.png/FAEBD7', title: '我是标题10', desc: '描述描述描述描述描述描述描述描述10' }];
|
||||||
|
data.list = data.list.concat(newArr);
|
||||||
|
}
|
||||||
|
|
||||||
|
function changeColumn(h) {
|
||||||
|
column.value = !h ? column.value - 1 : column.value + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
function loaded() {
|
||||||
|
console.log('加载完成')
|
||||||
|
}
|
||||||
|
|
||||||
|
function wapperClick(item) {
|
||||||
|
console.log('单项点击事件', item)
|
||||||
|
}
|
||||||
|
|
||||||
|
function imageClick(item) {
|
||||||
|
console.log('图片点击事件', item)
|
||||||
|
}
|
||||||
|
const waterfallsFlowRef = ref(null);
|
||||||
|
|
||||||
|
function reset() {
|
||||||
|
data.list = [{ image: 'https://via.placeholder.com/200x500.png/ff0000', title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' }]
|
||||||
|
waterfallsFlowRef.value.refresh();
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
</script>
|
||||||
|
<script>
|
||||||
|
// #ifdef VUE2
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
data: {
|
||||||
|
list: [{ image: 'https://via.placeholder.com/200x500.png/ff0000', title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x200.png/2878ff', title: '我是标题2', desc: '描述描述描述描述描述描述描述描述2' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x100.png/FFB6C1', title: '我是标题3', desc: '描述描述描述描述描述描述描述描述3' },
|
||||||
|
{ image: 'https://via.placeholder.com/200x300.png/9400D3', title: '我是标题4', desc: '描述描述描述描述描述描述描述描述4' },
|
||||||
|
{ image: 'https://via.placeholder.com/100x240.png/B0E0E6', title: '我是标题5', desc: '描述描述描述描述描述描述描述描述5' },
|
||||||
|
{ image: 'https://via.placeholder.com/140x280.png/7FFFAA', title: '我是标题6', desc: '描述描述描述描述描述描述描述描述6' },
|
||||||
|
{ image: 'https://via.placeholder.com/40x60.png/EEE8AA', title: '我是标题7', desc: '描述描述描述描述描述描述描述描述7' }]
|
||||||
|
},
|
||||||
|
column: 3
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
add() {
|
||||||
|
const newArr = [{ image: 'https://via.placeholder.com/58x100.png/FF7F50', title: '我是标题8', desc: '描述描述描述描述描述描述描述描述8' },
|
||||||
|
{ image: 'https://via.placeholder.com/59x100.png/C0C0C0', title: '我是标题9', desc: '描述描述描述描述描述描述描述描述9' },
|
||||||
|
{ image: 'https://via.placeholder.com/60x100.png/FAEBD7', title: '我是标题10', desc: '描述描述描述描述描述描述描述描述10' }]
|
||||||
|
this.data.list = this.data.list.concat(newArr);
|
||||||
|
},
|
||||||
|
changeColumn(h) {
|
||||||
|
this.column = !h ? this.column - 1 : this.column + 1;
|
||||||
|
},
|
||||||
|
loaded() {
|
||||||
|
console.log('加载完成')
|
||||||
|
},
|
||||||
|
wapperClick(item) {
|
||||||
|
console.log('单项点击事件', item)
|
||||||
|
},
|
||||||
|
imageClick(item) {
|
||||||
|
console.log('图片点击事件', item)
|
||||||
|
},
|
||||||
|
reset() {
|
||||||
|
this.data.list = [{ image: 'https://via.placeholder.com/200x500.png/ff0000', title: '我是标题1', desc: '描述描述描述描述描述描述描述描述1' }]
|
||||||
|
this.$refs.waterfallsFlowRef.refresh();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
page {
|
||||||
|
background-color: #f2f5f9;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.handle {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
margin-bottom: 20rpx;
|
||||||
|
padding: 10rpx;
|
||||||
|
|
||||||
|
.btn {
|
||||||
|
margin: 20rpx 10rpx;
|
||||||
|
padding: 0 20rpx;
|
||||||
|
background: #2878FF;
|
||||||
|
font-size: 28rpx;
|
||||||
|
color: #fff;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.item {
|
||||||
|
padding: 10rpx 10rpx 20rpx;
|
||||||
|
|
||||||
|
.title {
|
||||||
|
line-height: 48rpx;
|
||||||
|
font-size: 28rpx;
|
||||||
|
color: #222;
|
||||||
|
}
|
||||||
|
|
||||||
|
.desc {
|
||||||
|
font-size: 24rpx;
|
||||||
|
color: #666;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
```
|
||||||
|
|
||||||
|
<div id="c10"></div>
|
||||||
|
|
||||||
|
#### 温馨提示
|
||||||
|
|
||||||
|
1、该插件反复测试过微信小程序、h5、app-vue三个端,vue2和vue3都兼容,其他端可能需要测试改进。
|
||||||
|
|
||||||
|
2、该插件的使用hbuilderx版本最好升级到较新版本,我开发的版本是hbuilderx3.3.11.20220209。
|
||||||
|
|
||||||
|
3、对此插件或相关问题有好的建议,可以直接在评论区进行讨论。
|
||||||
|
|
||||||
|
4、希望遇到问题不要喷,也不要骂人,其实这种心情我能理解,写该插件也不是一时半会就完成了的,所以希望互相理解。只要有问题,我会第一时间回复解决。
|
||||||
|
|
||||||
|
5、对此插件有任何问题的可以在下方留言,我会第一时间回复和解决问题。还可以加QQ群进行前端技术交流 568984539,加群备注‘地区-名字-技术类型’。
|
||||||
|
|
||||||
|
#### 最后我想说:认为该插件对你有帮助的,记得收藏、好评,这样可以帮助到更多人哟!
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<div id="c11"></div>
|
||||||
|
|
||||||
|
#### 关注我,不迷路
|
||||||
|
|
||||||
|
如果任何疑问的可以在评论区留言,还可以加QQ群交流:568984539,加群备注‘地区-名字-技术类型’。
|
||||||
|
|
||||||
|
更多前端等相关知识可关注我个人博客:https://blog.csdn.net/qq_42961150?spm=1011.2124.3001.5343
|
||||||
|
|
||||||
|
<div id="c12"></div>
|
||||||
|
|
||||||
|
#### 个人作品展示
|
||||||
|
|
||||||
|
uniapp+vue3.2+unicloud开发微信小程序:**皮皮虎去水印**。
|
||||||
|
|
||||||
|
关注下方公众号:【**全网免费网盘资源**】、【**美团外卖饿了么天天领红包**】、【**去水印**】
|
||||||
|
|
||||||
|
![image](https://vkceyugu.cdn.bspapp.com/VKCEYUGU-bb657efd-fece-483e-a715-5daea480fde8/6e029310-aec8-46e9-9883-1c88dc1925ad.jpg)
|
|
@ -0,0 +1,22 @@
|
||||||
|
## 1.1.6(2022-10-24)
|
||||||
|
修复 初始化渲染变量取值错误问题
|
||||||
|
## 1.1.5(2022-10-24)
|
||||||
|
初始化渲染增加状态条件判断
|
||||||
|
## 1.1.4(2022-10-24)
|
||||||
|
修改空数据提示
|
||||||
|
## 1.1.3(2022-10-24)
|
||||||
|
1、增加内容插槽
|
||||||
|
|
||||||
|
2、删除状态文本属性
|
||||||
|
|
||||||
|
3、组件创建时可触发渲染条件
|
||||||
|
## 1.1.2(2022-09-26)
|
||||||
|
修改了开启布局的判断条件
|
||||||
|
## 1.1.1(2022-08-28)
|
||||||
|
1、加强组件化封装
|
||||||
|
|
||||||
|
2、 完善注释和优化使用逻辑
|
||||||
|
## 1.1.0(2022-08-22)
|
||||||
|
重写渲染列表逻辑
|
||||||
|
## 1.0.1(2021-06-08)
|
||||||
|
修改插入方向计算方式
|
|
@ -0,0 +1,98 @@
|
||||||
|
<template>
|
||||||
|
<view class="waterfall-item-container">
|
||||||
|
<view class="waterfall-item" @tap="onTap">
|
||||||
|
<image :src="params.url" mode="widthFix" @load="emitHeight" @error="emitHeight"></image>
|
||||||
|
<view class="content">
|
||||||
|
<view>{{params.title}}</view>
|
||||||
|
<view class="money">{{params.money}}元</view>
|
||||||
|
<view style="margin: 0 0 8rpx 0;">
|
||||||
|
<text class="label">{{params.label}}</text>
|
||||||
|
</view>
|
||||||
|
<view class="shop-name">{{params.shop}}</view>
|
||||||
|
123
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name:"helangWaterfallItem",
|
||||||
|
options:{
|
||||||
|
virtualHost: true
|
||||||
|
},
|
||||||
|
props:{
|
||||||
|
params:{
|
||||||
|
type: Object,
|
||||||
|
default(){
|
||||||
|
return {}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
tag:{
|
||||||
|
type:String | Number,
|
||||||
|
default:''
|
||||||
|
},
|
||||||
|
index:{
|
||||||
|
type:Number,
|
||||||
|
default:-1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
|
||||||
|
};
|
||||||
|
},
|
||||||
|
methods:{
|
||||||
|
// 发出组件高度信息,在此处可以区分正确和错误的加载,给予错误的提示图片
|
||||||
|
emitHeight(e){
|
||||||
|
const query = uni.createSelectorQuery().in(this);
|
||||||
|
query.select('.waterfall-item-container').boundingClientRect(data => {
|
||||||
|
let height = Math.floor(data.height);
|
||||||
|
this.$emit("height",height,this.$props.tag);
|
||||||
|
}).exec();
|
||||||
|
},
|
||||||
|
onTap(){
|
||||||
|
this.$emit("click",this.$props.index,this.$props.tag);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="less" scoped>
|
||||||
|
.waterfall-item{
|
||||||
|
padding: 16rpx;
|
||||||
|
background-color: #fff;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 28rpx;
|
||||||
|
color: #666;
|
||||||
|
|
||||||
|
image{
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
// 默认设置一个图片的大约值
|
||||||
|
height: 350rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content{
|
||||||
|
margin-top: 16rpx;
|
||||||
|
|
||||||
|
.money{
|
||||||
|
color: #fa3534;
|
||||||
|
margin-top: 8rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.label{
|
||||||
|
background-color: #fa3534;
|
||||||
|
color: #fff;
|
||||||
|
font-size: 20rpx;
|
||||||
|
padding: 4rpx 16rpx;
|
||||||
|
border-radius: 20rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shop-name{
|
||||||
|
font-size: 20rpx;
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,241 @@
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<view class="waterfall-box h-flex-x h-flex-2">
|
||||||
|
<view>
|
||||||
|
<view v-for="(item,index) in leftList" :key="item._render_id"
|
||||||
|
class="list-item"
|
||||||
|
>
|
||||||
|
<!-- :class="{'show': showPage > item._current_page }" -->
|
||||||
|
456
|
||||||
|
<helang-waterfall-item
|
||||||
|
:params="item"
|
||||||
|
tag="left"
|
||||||
|
:index="index"
|
||||||
|
@height="onHeight"
|
||||||
|
@click="onClick"
|
||||||
|
></helang-waterfall-item>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view>
|
||||||
|
<view v-for="(item,index) in rightList" :key="item._render_id"
|
||||||
|
class="list-item"
|
||||||
|
>
|
||||||
|
456
|
||||||
|
<helang-waterfall-item
|
||||||
|
:params="item"
|
||||||
|
@height="onHeight"
|
||||||
|
@click="onClick"
|
||||||
|
tag="right"
|
||||||
|
:index="index"
|
||||||
|
></helang-waterfall-item>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<slot name="default"></slot>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import helangWaterfallItem from "./waterfall-item.vue"
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name:"helangWaterfallList",
|
||||||
|
options:{
|
||||||
|
virtualHost: true
|
||||||
|
},
|
||||||
|
components: {
|
||||||
|
"helang-waterfall-item": helangWaterfallItem
|
||||||
|
},
|
||||||
|
props:{
|
||||||
|
// 组件状态
|
||||||
|
status:{
|
||||||
|
type: String,
|
||||||
|
default:''
|
||||||
|
},
|
||||||
|
// 待渲染的数据
|
||||||
|
list:{
|
||||||
|
type: Array,
|
||||||
|
default(){
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// 重置列表,设置为 true 时,瀑布流会自动重新渲染列表
|
||||||
|
reset:{
|
||||||
|
type: Boolean,
|
||||||
|
default:false
|
||||||
|
},
|
||||||
|
},
|
||||||
|
watch:{
|
||||||
|
"$props.status"(newValue,oldValue){
|
||||||
|
// 状态变更为 加载成功 时,执行瀑布流数据渲染
|
||||||
|
if(newValue == 'success'){
|
||||||
|
this.startRender();
|
||||||
|
}else if(!this.showList){
|
||||||
|
this.resetData();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed:{
|
||||||
|
showList(){
|
||||||
|
return !["fail","empty"].includes(this.$props.status);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
// 左侧列表高度
|
||||||
|
leftHeight: 0,
|
||||||
|
// 右侧列表高度
|
||||||
|
rightHeight: 0,
|
||||||
|
// 左侧列表数据
|
||||||
|
leftList: [],
|
||||||
|
// 右侧列表数据
|
||||||
|
rightList: [],
|
||||||
|
// 待渲染列表
|
||||||
|
awaitRenderList:[],
|
||||||
|
// 当前展示页码数据
|
||||||
|
showPage:1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
if(this.$props.status == 'success'){
|
||||||
|
this.startRender();
|
||||||
|
}
|
||||||
|
console.log('-----',this.list);
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
// 监听高度变化
|
||||||
|
onHeight(height, tag) {
|
||||||
|
/**
|
||||||
|
* 这个为实际渲染后 CSS 中 margin-buttom 的值,本示例默认为20rpx
|
||||||
|
* 用于解决实际渲染后因为数据条数关系,高度差计算偏差的问题
|
||||||
|
* */
|
||||||
|
let marginBottom = uni.upx2px(20);
|
||||||
|
|
||||||
|
// console.log(`左高:${this.leftHeight},右高:${this.rightHeight},当前高:${height},插入方向:${tag}`)
|
||||||
|
|
||||||
|
if (tag == 'left') {
|
||||||
|
this.leftHeight += (height + marginBottom);
|
||||||
|
} else {
|
||||||
|
this.rightHeight += (height + marginBottom);
|
||||||
|
}
|
||||||
|
this.renderList();
|
||||||
|
},
|
||||||
|
// 组件点击事件
|
||||||
|
onClick(index, tag){
|
||||||
|
// 对应的数据
|
||||||
|
if(tag == 'left'){
|
||||||
|
this.$emit("click",this.leftList[index],index,tag);
|
||||||
|
}else{
|
||||||
|
this.$emit("click",this.rightList[index],index,tag);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// 渲染列表,这里实现瀑布流的左右分栏
|
||||||
|
renderList() {
|
||||||
|
// 待渲染长度为 0 时表示已渲染完成
|
||||||
|
if(this.awaitRenderList.length < 1){
|
||||||
|
this.showPage++;
|
||||||
|
this.$emit("done");
|
||||||
|
|
||||||
|
// 为防止 js 数值类型最大值溢出,当高度值大于 1亿时重置高度
|
||||||
|
if(this.leftHeight > 100000000){
|
||||||
|
if(this.leftHeight > this.rightHeight){
|
||||||
|
this.leftHeight = 2;
|
||||||
|
this.rightHeight = 1;
|
||||||
|
}else{
|
||||||
|
this.leftHeight = 1;
|
||||||
|
this.rightHeight = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let item = {
|
||||||
|
...this.awaitRenderList.splice(0,1)[0],
|
||||||
|
// 当前数据添加当前页面标识
|
||||||
|
_current_page:this.showPage,
|
||||||
|
// 当前数据添加一个渲染id,解决 v-for 重复会出现不执行 load 的 BUG
|
||||||
|
_render_id:new Date().getTime()
|
||||||
|
};
|
||||||
|
|
||||||
|
if(this.leftHeight > this.rightHeight){
|
||||||
|
this.rightList.push(item);
|
||||||
|
}else{
|
||||||
|
this.leftList.push(item);
|
||||||
|
}
|
||||||
|
console.log('r',this.rightList,this.leftList);
|
||||||
|
},
|
||||||
|
// 重置数据
|
||||||
|
resetData(){
|
||||||
|
this.leftHeight = 0;
|
||||||
|
this.rightHeight = 0;
|
||||||
|
this.leftList = [];
|
||||||
|
this.rightList = [];
|
||||||
|
this.awaitRenderList = [];
|
||||||
|
// 当前展示页码数据
|
||||||
|
this.showPage = 1;
|
||||||
|
},
|
||||||
|
// 启动渲染
|
||||||
|
startRender(){
|
||||||
|
if(!this.showList){
|
||||||
|
this.resetData();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!this.$props.list || this.$props.list.length < 1){
|
||||||
|
console.log('河浪瀑布流插件提示:当前数据为空,不会触发列表渲染');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 若本次渲染为 重置 则先恢复组件的默认参数
|
||||||
|
if(this.$props.reset){
|
||||||
|
this.resetData();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.awaitRenderList = [...this.$props.list];
|
||||||
|
this.renderList();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="less" scoped>
|
||||||
|
.waterfall-box {
|
||||||
|
padding: 20rpx 10rpx;
|
||||||
|
box-sizing: border-box;
|
||||||
|
|
||||||
|
>view {
|
||||||
|
padding: 0 10rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-item{
|
||||||
|
margin-bottom: 0;
|
||||||
|
// 设置透明,默认是可视的
|
||||||
|
opacity: 0;
|
||||||
|
// 默认超出隐藏,不影响加载中的文字显示效果
|
||||||
|
overflow: hidden;
|
||||||
|
height: 0;
|
||||||
|
|
||||||
|
.show{
|
||||||
|
margin-bottom: 20rpx;
|
||||||
|
opacity: 1;
|
||||||
|
overflow: auto;
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.h-flex-x {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
flex-wrap: nowrap;
|
||||||
|
justify-content: flex-start;
|
||||||
|
align-items: flex-start;
|
||||||
|
align-content: flex-start;
|
||||||
|
|
||||||
|
&.h-flex-2 {
|
||||||
|
>view {
|
||||||
|
width: 50%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,39 @@
|
||||||
|
let list = ()=>{
|
||||||
|
return new Promise((resolve,reject)=>{
|
||||||
|
setTimeout(() => {
|
||||||
|
// 生成随机数方法
|
||||||
|
let random = (min = 0, max) => {
|
||||||
|
return Math.floor(Math.random() * max) + min;
|
||||||
|
}
|
||||||
|
// 待选的图片数据
|
||||||
|
let imgs = [];
|
||||||
|
// 待选的标题数据
|
||||||
|
let titles = [
|
||||||
|
'桃花坞里桃花庵,桃花庵里桃花仙;',
|
||||||
|
'桃花仙人种桃树,又摘桃花卖酒钱。',
|
||||||
|
'酒醒只在花前坐,酒醉还来花下眠;半醒半醉日复日,花落花开年复年。',
|
||||||
|
'但愿老死花酒间,不愿鞠躬车马前;',
|
||||||
|
'车尘马足富者趣,酒盏花枝贫者缘。若将富贵比贫贱,',
|
||||||
|
'一在平地一在天;若将贫贱比车马,他得驱驰我得闲。',
|
||||||
|
'别人笑我太疯癫,我笑他人看不穿;不见五陵豪杰墓,无花无酒锄作田。'
|
||||||
|
];
|
||||||
|
|
||||||
|
let res = [];
|
||||||
|
for (let i = 0; i < 10; i++) {
|
||||||
|
res.push({
|
||||||
|
id:i+1,
|
||||||
|
url:`/uni_modules/helang-waterfall/static/waterfall/${random(0,3)}.jpg?t=${new Date().getTime()}`,
|
||||||
|
title: titles[random(0, titles.length)],
|
||||||
|
money: random(9, 9999),
|
||||||
|
label:'官方自营',
|
||||||
|
shop:'唐诗三百首旗舰店'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
resolve(res);
|
||||||
|
}, 500);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
getList:list
|
||||||
|
}
|
|
@ -0,0 +1,76 @@
|
||||||
|
{
|
||||||
|
"id": "helang-waterfall",
|
||||||
|
"displayName": "瀑布流布局-waterfall",
|
||||||
|
"version": "1.1.6",
|
||||||
|
"description": "这是一款简单又好用的瀑布流布局模板,通过页面模板+组件的方式。充分利用组件的复用性和页面的生命周期。",
|
||||||
|
"keywords": [
|
||||||
|
"瀑布流",
|
||||||
|
"布局",
|
||||||
|
"列表",
|
||||||
|
"waterfall"
|
||||||
|
],
|
||||||
|
"repository": "https://gitee.com/myDarling/uniapp-extend",
|
||||||
|
"engines": {
|
||||||
|
"HBuilderX": "^3.2.11"
|
||||||
|
},
|
||||||
|
"dcloudext": {
|
||||||
|
"sale": {
|
||||||
|
"regular": {
|
||||||
|
"price": "0.00"
|
||||||
|
},
|
||||||
|
"sourcecode": {
|
||||||
|
"price": "0.00"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"contact": {
|
||||||
|
"qq": ""
|
||||||
|
},
|
||||||
|
"declaration": {
|
||||||
|
"ads": "无",
|
||||||
|
"data": "无",
|
||||||
|
"permissions": "无"
|
||||||
|
},
|
||||||
|
"npmurl": "",
|
||||||
|
"type": "uniapp-template-page"
|
||||||
|
},
|
||||||
|
"uni_modules": {
|
||||||
|
"dependencies": [],
|
||||||
|
"encrypt": [],
|
||||||
|
"platforms": {
|
||||||
|
"cloud": {
|
||||||
|
"tcb": "y",
|
||||||
|
"aliyun": "y"
|
||||||
|
},
|
||||||
|
"client": {
|
||||||
|
"App": {
|
||||||
|
"app-vue": "y",
|
||||||
|
"app-nvue": "u"
|
||||||
|
},
|
||||||
|
"H5-mobile": {
|
||||||
|
"Safari": "y",
|
||||||
|
"Android Browser": "y",
|
||||||
|
"微信浏览器(Android)": "y",
|
||||||
|
"QQ浏览器(Android)": "y"
|
||||||
|
},
|
||||||
|
"H5-pc": {
|
||||||
|
"Chrome": "y",
|
||||||
|
"IE": "y",
|
||||||
|
"Edge": "y",
|
||||||
|
"Firefox": "y",
|
||||||
|
"Safari": "y"
|
||||||
|
},
|
||||||
|
"小程序": {
|
||||||
|
"微信": "y",
|
||||||
|
"阿里": "y",
|
||||||
|
"百度": "y",
|
||||||
|
"字节跳动": "y",
|
||||||
|
"QQ": "y"
|
||||||
|
},
|
||||||
|
"快应用": {
|
||||||
|
"华为": "y",
|
||||||
|
"联盟": "y"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,247 @@
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<helang-waterfall-list
|
||||||
|
:status="waterfall.status"
|
||||||
|
:list="waterfall.list"
|
||||||
|
:reset="waterfall.reset"
|
||||||
|
@click="onClick"
|
||||||
|
@done="onDone"
|
||||||
|
>
|
||||||
|
<template>
|
||||||
|
<view v-if="waterfall.status == 'await'">
|
||||||
|
<view class="load-txt">上拉加载更多</view>
|
||||||
|
</view>
|
||||||
|
<view v-else-if="waterfall.status == 'loading'">
|
||||||
|
<view class="load-txt">加载中</view>
|
||||||
|
</view>
|
||||||
|
<view v-else-if="waterfall.status == 'success'">
|
||||||
|
<view class="load-txt">加载中</view>
|
||||||
|
</view>
|
||||||
|
<view v-else-if="waterfall.status == 'finish'">
|
||||||
|
<view class="load-txt">没有更多了</view>
|
||||||
|
</view>
|
||||||
|
<view v-else-if="waterfall.status == 'fail'">
|
||||||
|
<image class="load-icon" src="../../static/waterfall/fail.png"></image>
|
||||||
|
<view class="load-txt">出错了,请刷新重试</view>
|
||||||
|
</view>
|
||||||
|
<view v-else-if="waterfall.status == 'empty'">
|
||||||
|
<image class="load-icon" src="../../static/waterfall/empty.png"></image>
|
||||||
|
<view class="load-txt">暂无数据</view>
|
||||||
|
</view>
|
||||||
|
<view v-else><!-- 别问我为什么要写一个 v-else 的空 view,不写H5平台就会有CSS生效的离谱BUG --></view>
|
||||||
|
</template>
|
||||||
|
</helang-waterfall-list>
|
||||||
|
|
||||||
|
<view class="status-change" @tap="onStatusChange">
|
||||||
|
<view>切换<br />状态</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import helangWaterfallList from "@/uni_modules/helang-waterfall/components/waterfall/waterfall-list"
|
||||||
|
|
||||||
|
// 列表接口模拟数据
|
||||||
|
import mockData from '../../mock-data/waterfall-list.js'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
components: {
|
||||||
|
"helang-waterfall-list": helangWaterfallList
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
// 异步请求相关
|
||||||
|
ajax: {
|
||||||
|
// 是否可以加载
|
||||||
|
load: true,
|
||||||
|
// 每页的请求条件
|
||||||
|
rows:10,
|
||||||
|
// 页码
|
||||||
|
page:1,
|
||||||
|
// 数据列表
|
||||||
|
dataList:[]
|
||||||
|
},
|
||||||
|
// 瀑布流组件相关
|
||||||
|
waterfall:{
|
||||||
|
status:"",
|
||||||
|
reset:false,
|
||||||
|
list:[]
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onReady() {
|
||||||
|
this.getList();
|
||||||
|
},
|
||||||
|
// 触底触发
|
||||||
|
onReachBottom() {
|
||||||
|
this.getList();
|
||||||
|
},
|
||||||
|
// 下拉刷新
|
||||||
|
onPullDownRefresh(){
|
||||||
|
// 正常情况下接口返回应该很会很快。故意延迟调用,让用户有在刷新的体验感
|
||||||
|
setTimeout(()=>{
|
||||||
|
this.ajax.page = 1;
|
||||||
|
this.ajax.load = true;
|
||||||
|
this.getList();
|
||||||
|
},800);
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
// 瀑布流组件点击事件
|
||||||
|
onClick(data,index, tag){
|
||||||
|
console.log(data);
|
||||||
|
let direction = {
|
||||||
|
"left":'左',
|
||||||
|
"right":'右'
|
||||||
|
}
|
||||||
|
uni.showToast({
|
||||||
|
title:`${direction[tag]}侧列表第${index+1}个被点击`,
|
||||||
|
icon:'none'
|
||||||
|
})
|
||||||
|
},
|
||||||
|
// 瀑布流组件渲染完成
|
||||||
|
onDone(){
|
||||||
|
// 设置组件为 非重置,这行代码保留不删即可
|
||||||
|
this.waterfall.reset = false;
|
||||||
|
|
||||||
|
// 恢复 getList 方法的调用
|
||||||
|
this.ajax.load = true;
|
||||||
|
this.ajax.page++;
|
||||||
|
|
||||||
|
// 设置组件状态为 等待加载
|
||||||
|
this.waterfall.status = 'await';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 如果你是一个追求完美的开发者,可以通过判断当前数据的长度和请求的数据长度来优化前端请求,减少不必要请求
|
||||||
|
* 不需要则删除
|
||||||
|
* */
|
||||||
|
/**
|
||||||
|
if(this.ajax.dataCount >= this.ajax.rows){
|
||||||
|
this.ajax.load = true;
|
||||||
|
this.ajax.page++;
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
},
|
||||||
|
// 获取数据
|
||||||
|
getList() {
|
||||||
|
if (!this.ajax.load) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.ajax.load = false;
|
||||||
|
|
||||||
|
// 设置状态为加载中
|
||||||
|
this.waterfall.status = 'loading';
|
||||||
|
|
||||||
|
// 请求数据, mockData.getList 示例一个 ajax 请求
|
||||||
|
mockData.getList({
|
||||||
|
pageNum:this.ajax.page,
|
||||||
|
pageSize:this.ajax.rows
|
||||||
|
}).then(res=>{
|
||||||
|
// 获取到的数据,请注意数据结构
|
||||||
|
console.log(res);
|
||||||
|
|
||||||
|
// 第一页数据执行以下代码
|
||||||
|
if(this.ajax.page == 1){
|
||||||
|
// 关闭下拉
|
||||||
|
uni.stopPullDownRefresh();
|
||||||
|
|
||||||
|
// 设置组件状态为 重置,可供下拉刷新这类需要重置列表功能时使用
|
||||||
|
this.waterfall.reset = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 数据无效时处理
|
||||||
|
if(!res || res.length < 1){
|
||||||
|
// 设置组件为 加载结束 状态
|
||||||
|
this.waterfall.status = 'finish';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将数据赋值给瀑布流 list 属性
|
||||||
|
this.waterfall.list = res;
|
||||||
|
// 设置组件为 加载成功 状态,此时瀑布流组件开始计算当前数据的布局
|
||||||
|
this.waterfall.status = 'success';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 下方的代码为扩展其他功能的示例代码 可做参考,不需要可删除
|
||||||
|
* */
|
||||||
|
|
||||||
|
// 缓存当前数据给其他需要该数据的功能使用
|
||||||
|
if(this.ajax.page == 1){
|
||||||
|
this.ajax.dataList = res;
|
||||||
|
}else{
|
||||||
|
this.ajax.dataList = [...this.ajax.dataList,...res];
|
||||||
|
}
|
||||||
|
// 记录本次数据长度,意义请看 done 事件的回调
|
||||||
|
this.ajax.dataCount = res.length || 0;
|
||||||
|
|
||||||
|
// 。。。下面不需要写代码了,等待组件渲染完成
|
||||||
|
})
|
||||||
|
},
|
||||||
|
// 导航状态切换演示监听
|
||||||
|
onStatusChange(){
|
||||||
|
uni.showActionSheet({
|
||||||
|
itemList: ['常规', '加载异常', '加载错误'],
|
||||||
|
success: (res)=> {
|
||||||
|
switch(res.tapIndex){
|
||||||
|
case 0:
|
||||||
|
this.ajax.page = 1;
|
||||||
|
this.ajax.load = true;
|
||||||
|
this.getList();
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
// alert(111)
|
||||||
|
this.waterfall.status = 'fail';
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
this.waterfall.status = 'empty';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss">
|
||||||
|
page {
|
||||||
|
background-color: #f3f3f3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.load-txt{
|
||||||
|
padding: 0 0 20rpx 0;
|
||||||
|
text-align: center;
|
||||||
|
color: #999;
|
||||||
|
font-size: 24rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.load-icon{
|
||||||
|
width: 300rpx;
|
||||||
|
height: 300rpx;
|
||||||
|
margin: 0 auto 20rpx auto;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.status-change{
|
||||||
|
position: fixed;
|
||||||
|
right: 10rpx;
|
||||||
|
top: 60%;
|
||||||
|
width: 80rpx;
|
||||||
|
height: 80rpx;
|
||||||
|
z-index: 100;
|
||||||
|
font-size: 24rpx;
|
||||||
|
border-radius: 50%;
|
||||||
|
background-color: #0089ff;
|
||||||
|
color: #fff;
|
||||||
|
line-height: 1;
|
||||||
|
opacity: .33;
|
||||||
|
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
flex-wrap: nowrap;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
align-content: center;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1 @@
|
||||||
|
# helang-waterfall
|
After Width: | Height: | Size: 22 KiB |
After Width: | Height: | Size: 22 KiB |
After Width: | Height: | Size: 21 KiB |
After Width: | Height: | Size: 17 KiB |
After Width: | Height: | Size: 18 KiB |
|
@ -0,0 +1,24 @@
|
||||||
|
## 1.1.1(2023-08-01)
|
||||||
|
优化文档
|
||||||
|
## 1.1.0(2023-08-01)
|
||||||
|
优化文档
|
||||||
|
## 1.0.9(2023-07-10)
|
||||||
|
优化文档
|
||||||
|
## 1.0.8(2023-06-25)
|
||||||
|
优化文档
|
||||||
|
## 1.0.7(2023-06-25)
|
||||||
|
优化文档
|
||||||
|
## 1.0.6(2023-05-26)
|
||||||
|
优化文档
|
||||||
|
## 1.0.5(2023-05-22)
|
||||||
|
优化文档
|
||||||
|
## 1.0.4(2023-04-30)
|
||||||
|
新增图片放大功能,解决原生组件和tabbar导航栏等无法覆盖的问题
|
||||||
|
## 1.0.3(2023-04-28)
|
||||||
|
优化文档
|
||||||
|
## 1.0.2(2023-04-28)
|
||||||
|
优化文档
|
||||||
|
## 1.0.1(2023-04-28)
|
||||||
|
新增长按事件
|
||||||
|
## 1.0.0(2023-04-28)
|
||||||
|
插件上线
|
|
@ -0,0 +1,121 @@
|
||||||
|
<template>
|
||||||
|
<view class="previewImage" v-if="show" @tap="close">
|
||||||
|
<view class="page" v-if="urls.length > 0">
|
||||||
|
<text class="text">{{ current + 1 }} / {{ urls.length }}</text>
|
||||||
|
</view>
|
||||||
|
<swiper class="swiper" :current="current" @change="swiperChange" @touchstart="handleTouchStart" @touchend="handleTouchEnd">
|
||||||
|
<swiper-item v-for="(item, index) in urls" :key="index">
|
||||||
|
<movable-area class="movable-area" scale-area>
|
||||||
|
<movable-view class="movable-view" direction="all" :inertia="true" damping="100" scale="true" scale-min="1" scale-max="4" :scale-value="scale">
|
||||||
|
<scroll-view scroll-y="true" class="uni-scroll-view">
|
||||||
|
<view class="scroll-view"><image :key="index" class="image" :src="item" mode="widthFix" @longpress="onLongpress(item)" /></view>
|
||||||
|
</scroll-view>
|
||||||
|
</movable-view>
|
||||||
|
</movable-area>
|
||||||
|
</swiper-item>
|
||||||
|
</swiper>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
urls: {
|
||||||
|
type: Array,
|
||||||
|
required: true,
|
||||||
|
default: () => {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
show: false,
|
||||||
|
current: 0, //当前页
|
||||||
|
scale: 1,
|
||||||
|
isZooming: false // 是否处于缩放状态
|
||||||
|
};
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
//打开
|
||||||
|
open(current) {
|
||||||
|
this.current = this.urls.findIndex(item => item === current);
|
||||||
|
this.show = true;
|
||||||
|
this.$emit('open');
|
||||||
|
},
|
||||||
|
//关闭
|
||||||
|
close() {
|
||||||
|
if (!this.isZooming) {
|
||||||
|
this.show = false;
|
||||||
|
this.current = 0;
|
||||||
|
this.$emit('close');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
//图片改变
|
||||||
|
swiperChange(e) {
|
||||||
|
this.current = e.detail.current;
|
||||||
|
},
|
||||||
|
//监听长按
|
||||||
|
onLongpress(e) {
|
||||||
|
this.$emit('onLongpress', e);
|
||||||
|
},
|
||||||
|
handleTouchStart() {
|
||||||
|
this.isZooming = true;
|
||||||
|
},
|
||||||
|
handleTouchEnd() {
|
||||||
|
this.isZooming = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.previewImage {
|
||||||
|
z-index: 9999;
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background-color: #000000;
|
||||||
|
.swiper {
|
||||||
|
width: 100%;
|
||||||
|
height: 100vh;
|
||||||
|
swiper-item {
|
||||||
|
.movable-area {
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
.movable-view {
|
||||||
|
width: 100%;
|
||||||
|
min-height: 100%;
|
||||||
|
.scroll-view {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
height: 100vh;
|
||||||
|
.image {
|
||||||
|
width: 100%;
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.page {
|
||||||
|
position: absolute;
|
||||||
|
z-index: 9999;
|
||||||
|
width: 100%;
|
||||||
|
top: 60rpx;
|
||||||
|
text-align: center;
|
||||||
|
.text {
|
||||||
|
color: #fff;
|
||||||
|
font-size: 32rpx;
|
||||||
|
background-color: rgba(0, 0, 0, 0.5);
|
||||||
|
padding: 3rpx 16rpx;
|
||||||
|
border-radius: 20rpx;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -0,0 +1,81 @@
|
||||||
|
{
|
||||||
|
"id": "q-previewImage",
|
||||||
|
"displayName": "图片预览、多图左右滑动、图片放大、支持覆盖原生组件、原生导航栏、tabbar",
|
||||||
|
"version": "1.1.1",
|
||||||
|
"description": "最简洁的模拟图片预览,支持长按事件,多图左右滑动,大图上下滑动查看,支持图片放大,支持覆盖原生组件/原生导航栏/tabbar 支持vue2/vue3/app/小程序/h5",
|
||||||
|
"keywords": [
|
||||||
|
"图片预览"
|
||||||
|
],
|
||||||
|
"repository": "",
|
||||||
|
"engines": {
|
||||||
|
"HBuilderX": "^3.4.14"
|
||||||
|
},
|
||||||
|
"dcloudext": {
|
||||||
|
"type": "component-vue",
|
||||||
|
"sale": {
|
||||||
|
"regular": {
|
||||||
|
"price": "0.00"
|
||||||
|
},
|
||||||
|
"sourcecode": {
|
||||||
|
"price": "0.00"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"contact": {
|
||||||
|
"qq": ""
|
||||||
|
},
|
||||||
|
"declaration": {
|
||||||
|
"ads": "无",
|
||||||
|
"data": "无",
|
||||||
|
"permissions": "无"
|
||||||
|
},
|
||||||
|
"npmurl": ""
|
||||||
|
},
|
||||||
|
"uni_modules": {
|
||||||
|
"dependencies": [],
|
||||||
|
"encrypt": [],
|
||||||
|
"platforms": {
|
||||||
|
"cloud": {
|
||||||
|
"tcb": "y",
|
||||||
|
"aliyun": "y"
|
||||||
|
},
|
||||||
|
"client": {
|
||||||
|
"Vue": {
|
||||||
|
"vue2": "y",
|
||||||
|
"vue3": "y"
|
||||||
|
},
|
||||||
|
"App": {
|
||||||
|
"app-vue": "y",
|
||||||
|
"app-nvue": "n"
|
||||||
|
},
|
||||||
|
"H5-mobile": {
|
||||||
|
"Safari": "y",
|
||||||
|
"Android Browser": "u",
|
||||||
|
"微信浏览器(Android)": "u",
|
||||||
|
"QQ浏览器(Android)": "u"
|
||||||
|
},
|
||||||
|
"H5-pc": {
|
||||||
|
"Chrome": "u",
|
||||||
|
"IE": "u",
|
||||||
|
"Edge": "u",
|
||||||
|
"Firefox": "u",
|
||||||
|
"Safari": "u"
|
||||||
|
},
|
||||||
|
"小程序": {
|
||||||
|
"微信": "y",
|
||||||
|
"阿里": "u",
|
||||||
|
"百度": "u",
|
||||||
|
"字节跳动": "u",
|
||||||
|
"QQ": "u",
|
||||||
|
"钉钉": "u",
|
||||||
|
"快手": "u",
|
||||||
|
"飞书": "u",
|
||||||
|
"京东": "u"
|
||||||
|
},
|
||||||
|
"快应用": {
|
||||||
|
"华为": "u",
|
||||||
|
"联盟": "u"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,244 @@
|
||||||
|
# 最简洁的模拟图片预览,支持长按事件,多图左右滑动,大图上下滑动查看,支持图片放大,支持覆盖原生组件/原生导航栏/tabbar 支持vue2/vue3/app/小程序/h5
|
||||||
|
|
||||||
|
- 为了解决项目中因一些特殊原因无法使用uni.previewImage,例如App.onShow或者页面的oShow中写了方法。
|
||||||
|
- 如果用uni.previewImage,每次预览图片都会进到onShow的方法里
|
||||||
|
- 可以基本实现官方的预览图片功能,但是体验不如uni.previewImage()
|
||||||
|
- 如没有特殊原因,还是推荐官方的uni.previewImage()
|
||||||
|
|
||||||
|
## 安装指引
|
||||||
|
|
||||||
|
##1. 在插件市场打开本插件页面,在右侧点击`使用 HBuilderX 导入插件`,选择要导入的项目点击确定
|
||||||
|
|
||||||
|
##2. 使用方法 vue2写法
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<video v-if="videoShow" id="myVideo" src="https://img.cdn.aliyun.dcloud.net.cn/guide/uniapp/%E7%AC%AC1%E8%AE%B2%EF%BC%88uni-app%E4%BA%A7%E5%93%81%E4%BB%8B%E7%BB%8D%EF%BC%89-%20DCloud%E5%AE%98%E6%96%B9%E8%A7%86%E9%A2%91%E6%95%99%E7%A8%8B@20200317.mp4" controls></video>
|
||||||
|
<image v-for="(item, index) in imgs" :key="index" :src="item" @click="preview(item)"></image>
|
||||||
|
<q-previewImage ref="previewImage" :urls="imgs" @onLongpress="onLongpress" @open="open" @close="close"></q-previewImage>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
videoShow:true,//video组件是否显示
|
||||||
|
imgs: [],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
preview(url) {
|
||||||
|
this.imgs = ['https://web-assets.dcloud.net.cn/unidoc/zh/multiport-20210812.png', 'https://web-assets.dcloud.net.cn/unidoc/zh/uni-function-diagram.png'] //设置图片数组
|
||||||
|
// #ifdef MP-WEIXIN
|
||||||
|
this.$nextTick(()=>{
|
||||||
|
this.$refs.previewImage.open(url); // 传入当前选中的图片地址(小程序必须添加$nextTick,解决组件首次加载无图)
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifndef MP-WEIXIN
|
||||||
|
this.$refs.previewImage.open(url); // 传入当前选中的图片地址
|
||||||
|
// #endif
|
||||||
|
},
|
||||||
|
onLongpress(e){ //长按事件
|
||||||
|
console.log('当前长按的图片是' + e);
|
||||||
|
uni.showActionSheet({
|
||||||
|
itemList: ['转发给朋友', '保存到手机'],
|
||||||
|
success: function (res) {
|
||||||
|
console.log('选中了第' + (res.tapIndex + 1) + '个按钮');
|
||||||
|
},
|
||||||
|
fail: function (res) {
|
||||||
|
console.log(res.errMsg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
/* open和close方法一般用不到,但是在一些特殊场景会用到,
|
||||||
|
* 比如预览图片时你需要覆盖 NavigationBar和 TabBar,
|
||||||
|
* 或者在app中需要预览图片时覆盖住原生组件,比如video或者map等,
|
||||||
|
* 你可以根据open和close去做一些操作,例如隐藏导航栏或者隐藏一些原生组件等
|
||||||
|
*/
|
||||||
|
open(){ //监听组件显示 (隐藏TabBar和NavigationBar,隐藏video原生组件)
|
||||||
|
// uni.hideTabBar()
|
||||||
|
// uni.setNavigationBarColor({
|
||||||
|
// frontColor: '#000000', // 设置前景色为黑色
|
||||||
|
// backgroundColor: '#000000', // 设置背景色为黑色
|
||||||
|
// })
|
||||||
|
// this.videoShow = false
|
||||||
|
},
|
||||||
|
close(){ //监听组件隐藏 (显示TabBar和NavigationBar,显示video原生组件)
|
||||||
|
// uni.showTabBar()
|
||||||
|
// uni.setNavigationBarColor({
|
||||||
|
// frontColor: '#ffffff', // 设置前景色为白色
|
||||||
|
// backgroundColor: '#000000', // 设置背景色为黑色
|
||||||
|
// })
|
||||||
|
// this.videoShow = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
##3. vue3 setup写法
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<video v-if="videoShow" id="myVideo" src="https://img.cdn.aliyun.dcloud.net.cn/guide/uniapp/%E7%AC%AC1%E8%AE%B2%EF%BC%88uni-app%E4%BA%A7%E5%93%81%E4%BB%8B%E7%BB%8D%EF%BC%89-%20DCloud%E5%AE%98%E6%96%B9%E8%A7%86%E9%A2%91%E6%95%99%E7%A8%8B@20200317.mp4" controls></video>
|
||||||
|
<image v-for="(item, index) in imgs" :key="index" :src="item" @click="preview(item)"></image>
|
||||||
|
<q-previewImage ref="previewImage" :urls="imgs" @onLongpress="onLongpress" @open="open" @close="close"></q-previewImage>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import { reactive, ref, toRefs,nextTick } from 'vue';
|
||||||
|
|
||||||
|
const data = reactive({
|
||||||
|
videoShow:true,//video组件是否显示
|
||||||
|
imgs: [],
|
||||||
|
});
|
||||||
|
const previewImage = ref(null);
|
||||||
|
|
||||||
|
const { imgs,videoShow } = toRefs(data)// 解构
|
||||||
|
|
||||||
|
const preview = url => {
|
||||||
|
data.imgs = ['https://web-assets.dcloud.net.cn/unidoc/zh/multiport-20210812.png', 'https://web-assets.dcloud.net.cn/unidoc/zh/uni-function-diagram.png'] //设置图片数组
|
||||||
|
|
||||||
|
|
||||||
|
// #ifdef MP-WEIXIN
|
||||||
|
nextTick(()=>{
|
||||||
|
previewImage.value.open(url); // 传入当前选中的图片地址(小程序必须添加nextTick,解决组件首次加载无图)
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifndef MP-WEIXIN
|
||||||
|
previewImage.value.open(url); // 传入当前选中的图片地址
|
||||||
|
// #endif
|
||||||
|
};
|
||||||
|
|
||||||
|
const onLongpress = e =>{
|
||||||
|
console.log('当前长按的图片是' + e);
|
||||||
|
uni.showActionSheet({
|
||||||
|
itemList: ['转发给朋友', '保存到手机'],
|
||||||
|
success: function (res) {
|
||||||
|
console.log('选中了第' + (res.tapIndex + 1) + '个按钮');
|
||||||
|
},
|
||||||
|
fail: function (res) {
|
||||||
|
console.log(res.errMsg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/* open和close方法一般用不到,但是在一些特殊场景会用到,
|
||||||
|
* 比如预览图片时你需要覆盖 NavigationBar和 TabBar,
|
||||||
|
* 或者在app中需要预览图片时覆盖住原生组件,比如video或者map等,
|
||||||
|
* 你可以根据open和close去做一些操作,例如隐藏导航栏或者隐藏一些原生组件等
|
||||||
|
*/
|
||||||
|
const open = () => { //监听组件显示 (隐藏TabBar和NavigationBar,隐藏video原生组件)
|
||||||
|
// uni.hideTabBar()
|
||||||
|
// uni.setNavigationBarColor({
|
||||||
|
// frontColor: '#000000', // 设置前景色为黑色
|
||||||
|
// backgroundColor: '#000000', // 设置背景色为黑色
|
||||||
|
// })
|
||||||
|
// data.videoShow = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const close = () => { //监听组件隐藏 (显示TabBar和NavigationBar,显示video原生组件)
|
||||||
|
// uni.showTabBar()
|
||||||
|
// uni.setNavigationBarColor({
|
||||||
|
// frontColor: '#ffffff', // 设置前景色为白色
|
||||||
|
// backgroundColor: '#000000', // 设置背景色为黑色
|
||||||
|
// })
|
||||||
|
// data.videoShow = true
|
||||||
|
}
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
##4. 项目示例 (一般返回的数据图片是以逗号或特殊字符分割的字符串,点击时就需要传两个参数,一个是图片数组,一个是当前图片的index)
|
||||||
|
## 注意q-previewImage不要写在循环体中,imgs其实就是用来存放当前图片的数组,每次点击每次赋值就行
|
||||||
|
|
||||||
|
```
|
||||||
|
<template>
|
||||||
|
<view>
|
||||||
|
<video v-if="videoShow" id="myVideo" src="https://img.cdn.aliyun.dcloud.net.cn/guide/uniapp/%E7%AC%AC1%E8%AE%B2%EF%BC%88uni-app%E4%BA%A7%E5%93%81%E4%BB%8B%E7%BB%8D%EF%BC%89-%20DCloud%E5%AE%98%E6%96%B9%E8%A7%86%E9%A2%91%E6%95%99%E7%A8%8B@20200317.mp4" controls></video>
|
||||||
|
<view v-for="(item, index) in list" :key="index" class="list">
|
||||||
|
<image :src="i" mode="aspectFill" v-for="(i,imgindex) in item.urls.split(',')" @click.stop="preimg(item.urls.split(','),imgindex)"></image>
|
||||||
|
<view>
|
||||||
|
<q-previewImage ref="previewImage" :urls="imgs" @onLongpress="onLongpress" @open="open" @close="close"></q-previewImage>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
videoShow:true,//是否显示video组件
|
||||||
|
imgs: [],//imgs其实就是用来存放当前图片的数组,每次点击每次赋值就行
|
||||||
|
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
preimg(urls,index){
|
||||||
|
this.imgs = urls //imgs其实就是用来存放当前图片的数组,每次点击每次赋值就行
|
||||||
|
// #ifdef MP-WEIXIN
|
||||||
|
this.$nextTick(()=>{
|
||||||
|
this.$refs.previewImage.open(this.imgs[index]); // 传入当前选中的图片地址(小程序必须添加$nextTick,解决组件首次加载无图)
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifndef MP-WEIXIN
|
||||||
|
this.$refs.previewImage.open(this.imgs[index]); // 传入当前选中的图片地址
|
||||||
|
// #endif
|
||||||
|
},
|
||||||
|
onLongpress(e){ //长按事件
|
||||||
|
console.log('当前长按的图片是' + e);
|
||||||
|
uni.showActionSheet({
|
||||||
|
itemList: ['转发给朋友', '保存到手机'],
|
||||||
|
success: function (res) {
|
||||||
|
console.log('选中了第' + (res.tapIndex + 1) + '个按钮');
|
||||||
|
},
|
||||||
|
fail: function (res) {
|
||||||
|
console.log(res.errMsg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
/* open和close方法一般用不到,但是在一些特殊场景会用到,
|
||||||
|
* 比如预览图片时你需要覆盖 NavigationBar和 TabBar,
|
||||||
|
* 或者在app中需要预览图片时覆盖住原生组件,比如video或者map等,
|
||||||
|
* 你可以根据open和close去做一些操作,例如隐藏导航栏或者隐藏一些原生组件等
|
||||||
|
*/
|
||||||
|
open(){ //监听组件显示 (隐藏TabBar和NavigationBar,隐藏video原生组件)
|
||||||
|
// uni.hideTabBar()
|
||||||
|
// uni.setNavigationBarColor({
|
||||||
|
// frontColor: '#000000', // 设置前景色为黑色
|
||||||
|
// backgroundColor: '#000000', // 设置背景色为黑色
|
||||||
|
// })
|
||||||
|
// this.videoShow = false
|
||||||
|
},
|
||||||
|
close(){ //监听组件隐藏 (显示TabBar和NavigationBar,显示video原生组件)
|
||||||
|
// uni.showTabBar()
|
||||||
|
// uni.setNavigationBarColor({
|
||||||
|
// frontColor: '#ffffff', // 设置前景色为白色
|
||||||
|
// backgroundColor: '#000000', // 设置背景色为黑色
|
||||||
|
// })
|
||||||
|
// this.videoShow = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
## 如果插件对您有一点帮助,请给个五星好评,感谢支持
|
||||||
|
|
||||||
|
|
||||||
|
## 如有问题,请加qq 965969604
|