30 changed files with 1402 additions and 166 deletions
@ -0,0 +1,39 @@ |
|||||
|
package com.dreamchaser.depository_manage.entity; |
||||
|
|
||||
|
import lombok.Data; |
||||
|
|
||||
|
@Data |
||||
|
public class MaterialAndPlace { |
||||
|
/** |
||||
|
* id |
||||
|
*/ |
||||
|
private Integer id; |
||||
|
/** |
||||
|
* 物料id |
||||
|
*/ |
||||
|
private Integer mid; |
||||
|
/** |
||||
|
* 库位id |
||||
|
*/ |
||||
|
private Integer pid; |
||||
|
/** |
||||
|
* 当前库位所处仓库 |
||||
|
*/ |
||||
|
private Integer did; |
||||
|
/** |
||||
|
* 库位编码 |
||||
|
*/ |
||||
|
private String code; |
||||
|
/** |
||||
|
* 库位最大存储量 |
||||
|
*/ |
||||
|
private Integer max; |
||||
|
/** |
||||
|
* 库位最小存储量 |
||||
|
*/ |
||||
|
private Integer min; |
||||
|
/** |
||||
|
* 当前存放额度 |
||||
|
*/ |
||||
|
private Integer quantity; |
||||
|
} |
||||
@ -0,0 +1,143 @@ |
|||||
|
<!DOCTYPE html> |
||||
|
<html lang="en" xmlns:th="http://www.thymeleaf.org"> |
||||
|
<head> |
||||
|
<meta charset="utf-8"> |
||||
|
<title>layui</title> |
||||
|
<meta name="renderer" content="webkit"> |
||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> |
||||
|
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"> |
||||
|
<link rel="stylesheet" href="/static/lib/layui-v2.6.3/css/layui.css" media="all"> |
||||
|
<link rel="stylesheet" href="/static/css/public.css" media="all"> |
||||
|
</head> |
||||
|
<body> |
||||
|
<div class="layuimini-container"> |
||||
|
<div class="layuimini-main"> |
||||
|
<fieldset class="table-search-fieldset"> |
||||
|
<legend>库位创建</legend> |
||||
|
<div class="layui-fluid"> |
||||
|
<div class="layui-card"> |
||||
|
<div class="layui-card-body" style="padding-top: 40px;"> |
||||
|
<div> |
||||
|
<form class="layui-form" |
||||
|
style="margin: 0 auto;max-width: 700px;padding-top: 100px; padding-bottom: 200px" lay-filter="form1"> |
||||
|
<input style="display: none" th:value="${depositoryId}" name="depositoryId" id="depositoryID"> |
||||
|
<div class="layui-form-item"> |
||||
|
<label class="layui-form-label">起始行:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写行数" class="layui-input" |
||||
|
name="place_start_x" lay-verify="required"/> |
||||
|
</div> |
||||
|
<label class="layui-form-label">结束行:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写行数" class="layui-input" |
||||
|
name="place_end_x" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<!-- <div class="layui-form-item"> |
||||
|
<label class="layui-form-label">起始列:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写列数" class="layui-input" |
||||
|
name="place_start_y" lay-verify="required"/> |
||||
|
</div> |
||||
|
<label class="layui-form-label">结束列:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写列数" class="layui-input" |
||||
|
name="place_end_y" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div>--> |
||||
|
<div class="layui-form-item"> |
||||
|
<label class="layui-form-label">起始层:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写层数" class="layui-input" |
||||
|
name="place_start_z" lay-verify="required"/> |
||||
|
</div> |
||||
|
<label class="layui-form-label">结束层:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写层数" class="layui-input" |
||||
|
name="place_end_z" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="layui-form-item"> |
||||
|
<label class="layui-form-label">最大存放量:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写最大存放量" class="layui-input" |
||||
|
name="max" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="layui-form-item"> |
||||
|
<label class="layui-form-label">最小存放量:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写最小存放量" class="layui-input" |
||||
|
name="min" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="layui-form-item"> |
||||
|
<div class="layui-input-block"> |
||||
|
<button class="layui-btn" lay-submit lay-filter="formStep"> |
||||
|
 创建库位  |
||||
|
</button> |
||||
|
</div> |
||||
|
</div> |
||||
|
</form> |
||||
|
</div> |
||||
|
</div> |
||||
|
<hr> |
||||
|
</div> |
||||
|
</div> |
||||
|
</fieldset> |
||||
|
|
||||
|
</div> |
||||
|
</div> |
||||
|
|
||||
|
<script src="/static/lib/layui-v2.6.3/layui.js" charset="utf-8"></script> |
||||
|
|
||||
|
<script> |
||||
|
var data; |
||||
|
|
||||
|
layui.use(['form', 'layer','dropdown','tree'], function () { |
||||
|
var $ = layui.jquery, |
||||
|
form = layui.form, |
||||
|
layer = layui.layer; |
||||
|
|
||||
|
var depositoryId = $("#depositoryID").val(); |
||||
|
form.on('submit(formStep)', function (data) { |
||||
|
var req = data.field; |
||||
|
req.type = "list"; |
||||
|
$.ajax({ |
||||
|
url: "/place/addPlace", |
||||
|
type: 'post', |
||||
|
dataType: 'json', |
||||
|
contentType: "application/json;charset=utf-8", |
||||
|
data: JSON.stringify(req), |
||||
|
beforeSend: function () { |
||||
|
this.layerIndex = layer.load(0, {shade: [0.5, '#393D49']}); |
||||
|
}, |
||||
|
success: function (data) { |
||||
|
layer.close(this.layerIndex); |
||||
|
if (data.status >= 300) { |
||||
|
layer.msg(data.statusInfo.message);//失败的表情 |
||||
|
return; |
||||
|
} else { |
||||
|
layer.msg("添加成功!", { |
||||
|
icon: 6,//成功的表情 |
||||
|
time: 1000 |
||||
|
}, //1秒关闭(如果不配置,默认是3秒) |
||||
|
function(){ |
||||
|
//do something |
||||
|
window.location="/insertListPlace?depositoryId="+depositoryId |
||||
|
}); |
||||
|
} |
||||
|
}, |
||||
|
complete: function () { |
||||
|
form.val("form1", { |
||||
|
}) |
||||
|
} |
||||
|
}); |
||||
|
return false; |
||||
|
}); |
||||
|
|
||||
|
}); |
||||
|
</script> |
||||
|
|
||||
|
</body> |
||||
|
</html> |
||||
@ -0,0 +1,315 @@ |
|||||
|
<template> |
||||
|
<div class="page-scan"> |
||||
|
<div class="base-line"> |
||||
|
<div class="pullWrap"> |
||||
|
<div class="topTitle"> |
||||
|
<div class="pullTitle"> |
||||
|
<div class="pullName">二维码绑定</div> |
||||
|
<!-- 绑定状态图标 --> |
||||
|
<img class="left-icon" src="/static/img/noBind.svg" alt="" v-if="!dataObj.qrCodeId"> |
||||
|
<img class="left-icon" src="/static/img/binded.svg" alt="" v-else> |
||||
|
<div class="right-part"> |
||||
|
<input :disabled="dataObj.qrCodeId" type="text" v-model="dataObj.qrCodeId" placeholder="请输入二维码ID" v-if="dataObj.qrCodeId"> |
||||
|
<input type="text" v-model="scanTextData.scanText" placeholder="请输入二维码ID" v-else> |
||||
|
<img src="/static/img/scan.svg" alt="" v-if="!dataObj.qrCodeId"> |
||||
|
<span @click="toScanCode" v-if="!dataObj.qrCodeId">扫码填入</span> |
||||
|
<span v-if="dataObj.qrCodeId">解绑</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
<!-- 扫描盒子 --> |
||||
|
<div class="scan-box" v-if="scanTextData.showScanBox"> |
||||
|
<div class="scan-cacel" @click="cancelScan" v-show="scanTextData.showScanBoxInfo"> |
||||
|
<!-- <img src="" alt=""> --> |
||||
|
取消 |
||||
|
</div> |
||||
|
<video ref="video" id="video" class="scan-video" v-show="scanTextData.showScanBoxInfo" autoplay></video> |
||||
|
<div class="scan-img" v-show="scanTextData.showScanBoxInfo"> |
||||
|
<div class="scan-frame"> |
||||
|
<span class="left-t"></span> |
||||
|
<span class="right-t"></span> |
||||
|
<span class="left-b"></span> |
||||
|
<span class="right-b"></span> |
||||
|
<span class="cross-line"></span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<!-- <img src="/static/img/scan.svg" alt="" v-show="scanTextData.showScanBoxInfo"> --> |
||||
|
<div class="scan-tip" v-show="scanTextData.showScanBoxInfo"> {{scanTextData.tipMsg}} </div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</template> |
||||
|
|
||||
|
|
||||
|
<script> |
||||
|
import { BrowserMultiFormatReader } from '@zxing/library'; |
||||
|
let scanTextData = { |
||||
|
loadingShow: false, |
||||
|
codeReader: null, |
||||
|
scanText: '', |
||||
|
vin: null, |
||||
|
tipMsg: '将二维码置于屏幕中,即可识别', |
||||
|
tipShow: false |
||||
|
} |
||||
|
export default { |
||||
|
name: 'scanCodePage', |
||||
|
data() { |
||||
|
return { |
||||
|
scanTextData:{ |
||||
|
loadingShow: false, |
||||
|
codeReader: null, |
||||
|
scanText: '', |
||||
|
vin: null, |
||||
|
tipMsg: '将二维码置于屏幕中,即可识别', |
||||
|
tipShow: false, |
||||
|
|
||||
|
showScanBox:false, |
||||
|
showScanBoxInfo:false, |
||||
|
}, |
||||
|
hasBind:false |
||||
|
} |
||||
|
}, |
||||
|
methods: { |
||||
|
toScanCode(){ |
||||
|
console.log('识别二维码',this.dataObj) |
||||
|
scanTextData.codeReader = new BrowserMultiFormatReader(); |
||||
|
this.scanTextData.showScanBox = true |
||||
|
this.openScan(); |
||||
|
|
||||
|
}, |
||||
|
cancelScan(){ |
||||
|
//识别完停止使用摄像头 |
||||
|
let thisVideo = document.getElementById("video"); |
||||
|
thisVideo.srcObject.getTracks()[0].stop() |
||||
|
scanTextData.codeReader.reset(); // 重置 |
||||
|
this.scanTextData.showScanBox = false |
||||
|
setTimeout(()=>{ |
||||
|
this.scanTextData.showScanBoxInfo = false |
||||
|
},1000) |
||||
|
}, |
||||
|
|
||||
|
async openScan() { |
||||
|
scanTextData.codeReader.getVideoInputDevices().then((videoInputDevices) => { |
||||
|
scanTextData.tipShow = true; |
||||
|
scanTextData.tipMsg = '正在调用摄像头...'; |
||||
|
console.log('videoInputDevices', videoInputDevices); |
||||
|
// 默认获取第一个摄像头设备id |
||||
|
let firstDeviceId = videoInputDevices[0].deviceId; |
||||
|
// 获取第一个摄像头设备的名称 |
||||
|
const videoInputDeviceslablestr = JSON.stringify(videoInputDevices[0].label); |
||||
|
if (videoInputDevices.length > 1) { |
||||
|
// 判断是否后置摄像头 |
||||
|
if (videoInputDeviceslablestr.indexOf('back') > -1) { |
||||
|
firstDeviceId = videoInputDevices[0].deviceId; |
||||
|
} else { |
||||
|
firstDeviceId = videoInputDevices[1].deviceId; |
||||
|
} |
||||
|
} |
||||
|
this.decodeFromInputVideoFunc(firstDeviceId); |
||||
|
}).catch(err => { |
||||
|
scanTextData.tipShow = false; |
||||
|
console.error(err); |
||||
|
}); |
||||
|
}, |
||||
|
decodeFromInputVideoFunc(firstDeviceId) { |
||||
|
scanTextData.codeReader.reset(); // 重置 |
||||
|
scanTextData.scanText = ''; |
||||
|
scanTextData.codeReader.decodeFromInputVideoDeviceContinuously(firstDeviceId, 'video', (result, err) => { |
||||
|
scanTextData.tipMsg = '将二维码置于屏幕中,即可识别'; |
||||
|
scanTextData.scanText = ''; |
||||
|
setTimeout(()=>{ |
||||
|
this.scanTextData.showScanBoxInfo = true |
||||
|
},1000) |
||||
|
if (result) { |
||||
|
console.log('扫描结果', result.text); |
||||
|
if (result.text) { |
||||
|
console.log('扫描结果11', result.text); |
||||
|
this.scanTextData.showScanBox = false |
||||
|
this.scanTextData.showScanBoxInfo = false |
||||
|
this.scanTextData.scanText = result.text |
||||
|
//这里扫描出结果可以调用你想要的方法 |
||||
|
//识别完停止使用摄像头 |
||||
|
let thisVideo = document.getElementById("video"); |
||||
|
thisVideo.srcObject.getTracks()[0].stop() |
||||
|
scanTextData.codeReader.reset(); // 重置 |
||||
|
} |
||||
|
}else{ |
||||
|
// console.log('没出来?',result,err) |
||||
|
} |
||||
|
if (err && !(err)) { |
||||
|
scanTextData.tipMsg = '识别失败'; |
||||
|
setTimeout(() => { |
||||
|
scanTextData.tipShow = false; |
||||
|
}, 2000) |
||||
|
console.error(err); |
||||
|
} |
||||
|
}); |
||||
|
}, |
||||
|
|
||||
|
}, |
||||
|
props:['dataObj'] |
||||
|
} |
||||
|
</script> |
||||
|
|
||||
|
|
||||
|
<style scoped lang="scss"> |
||||
|
.pullWrap { |
||||
|
width: 100%; |
||||
|
height: 100px; |
||||
|
padding-top: 50px; |
||||
|
background: #fff; |
||||
|
} |
||||
|
.topTitle { |
||||
|
width: 100%; |
||||
|
position: relative; |
||||
|
} |
||||
|
.pullTitle { |
||||
|
display: flex; |
||||
|
height: 80px; |
||||
|
line-height: 80px; |
||||
|
margin-top: -40px; |
||||
|
background: #fff; |
||||
|
align-items: center; |
||||
|
justify-content: space-between; |
||||
|
} |
||||
|
.pullName{ |
||||
|
font-size: 30px; |
||||
|
color: rgba(0,0,0,0.85); |
||||
|
} |
||||
|
img{ |
||||
|
width: 40px; |
||||
|
height: 40px; |
||||
|
} |
||||
|
.left-icon{ |
||||
|
margin:0 15px; |
||||
|
} |
||||
|
span{ |
||||
|
text-decoration: underline; |
||||
|
text-decoration-color: #42a5ff; |
||||
|
color: #42a5ff; |
||||
|
margin-left: 5px; |
||||
|
} |
||||
|
.right-part { |
||||
|
flex: 1; |
||||
|
display: flex; |
||||
|
align-items: center; |
||||
|
justify-content: space-around; |
||||
|
border-bottom: 1px solid #e5e5e5; |
||||
|
overflow: hidden; |
||||
|
text-overflow: ellipsis; |
||||
|
} |
||||
|
input{ |
||||
|
border:none; |
||||
|
flex: 1; |
||||
|
} |
||||
|
|
||||
|
|
||||
|
.scan-index-bar{ |
||||
|
background-image: linear-gradient( -45deg, #42a5ff ,#59cfff); |
||||
|
} |
||||
|
.van-nav-bar__title{ |
||||
|
color: #fff !important; |
||||
|
} |
||||
|
.scan-box { |
||||
|
position: fixed; |
||||
|
top: 0; |
||||
|
left: 0; |
||||
|
z-index: 5; |
||||
|
height: 100%; |
||||
|
width: 100vw; |
||||
|
} |
||||
|
.scan-cacel{ |
||||
|
position: absolute; |
||||
|
top: 30px; |
||||
|
left: 30px; |
||||
|
z-index: 9; |
||||
|
color: #fff; |
||||
|
font-size: 35px; |
||||
|
|
||||
|
} |
||||
|
.scan-video{ |
||||
|
height: 100vh; |
||||
|
width: 100vw; |
||||
|
object-fit:cover; |
||||
|
} |
||||
|
.scan-img { |
||||
|
width: 500px; |
||||
|
height: 500px; |
||||
|
position: fixed; |
||||
|
top: 40%; |
||||
|
left: 50%; |
||||
|
margin-top: -200px; |
||||
|
margin-left: -250px; |
||||
|
z-index: 6; |
||||
|
} |
||||
|
.scan-frame { |
||||
|
width: 100%; |
||||
|
height: 100%; |
||||
|
position: relative; |
||||
|
} |
||||
|
.left-t,.right-t,.left-b,.right-b{ |
||||
|
position: absolute; |
||||
|
width: 80px; |
||||
|
height: 80px; |
||||
|
} |
||||
|
.left-t{ |
||||
|
top: 0; |
||||
|
left: 0; |
||||
|
border-top:2px solid #17B1B7; |
||||
|
border-left:2px solid #17B1B7; |
||||
|
} |
||||
|
.right-t{ |
||||
|
top: 0; |
||||
|
right: 0; |
||||
|
border-top:2px solid #17B1B7; |
||||
|
border-right:2px solid #17B1B7; |
||||
|
} |
||||
|
.left-b{ |
||||
|
bottom: 0; |
||||
|
left: 0; |
||||
|
border-bottom:2px solid #17B1B7; |
||||
|
border-left:2px solid #17B1B7; |
||||
|
} |
||||
|
.right-b{ |
||||
|
bottom: 0; |
||||
|
right: 0; |
||||
|
border-bottom:2px solid #17B1B7; |
||||
|
border-right:2px solid #17B1B7; |
||||
|
} |
||||
|
.cross-line{ |
||||
|
width: 600px; |
||||
|
height: 10px; |
||||
|
background: linear-gradient(to right, rgba(255, 255, 255, 0),#5DDDD3,rgba(255,255,255,0)); |
||||
|
position: absolute; |
||||
|
top: 0; |
||||
|
left: -50px; |
||||
|
animation: identifier_p 5s infinite; |
||||
|
} |
||||
|
@keyframes identifier_p { |
||||
|
0%{ |
||||
|
top: 0%; |
||||
|
} |
||||
|
50%{ |
||||
|
top: 100%; |
||||
|
} |
||||
|
100%{ |
||||
|
top: 0; |
||||
|
} |
||||
|
} |
||||
|
.scan-tip{ |
||||
|
width: 100vw; |
||||
|
text-align: center; |
||||
|
margin-bottom: 10vh; |
||||
|
color: white; |
||||
|
font-size: 5vw; |
||||
|
position: absolute; |
||||
|
bottom: 50px; |
||||
|
left: 0; |
||||
|
color: #fff; |
||||
|
} |
||||
|
.page-scan{ |
||||
|
overflow-y: hidden; |
||||
|
// background-color: #363636; |
||||
|
} |
||||
|
</style> |
||||
Binary file not shown.
@ -0,0 +1,143 @@ |
|||||
|
<!DOCTYPE html> |
||||
|
<html lang="en" xmlns:th="http://www.thymeleaf.org"> |
||||
|
<head> |
||||
|
<meta charset="utf-8"> |
||||
|
<title>layui</title> |
||||
|
<meta name="renderer" content="webkit"> |
||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> |
||||
|
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"> |
||||
|
<link rel="stylesheet" href="/static/lib/layui-v2.6.3/css/layui.css" media="all"> |
||||
|
<link rel="stylesheet" href="/static/css/public.css" media="all"> |
||||
|
</head> |
||||
|
<body> |
||||
|
<div class="layuimini-container"> |
||||
|
<div class="layuimini-main"> |
||||
|
<fieldset class="table-search-fieldset"> |
||||
|
<legend>库位创建</legend> |
||||
|
<div class="layui-fluid"> |
||||
|
<div class="layui-card"> |
||||
|
<div class="layui-card-body" style="padding-top: 40px;"> |
||||
|
<div> |
||||
|
<form class="layui-form" |
||||
|
style="margin: 0 auto;max-width: 700px;padding-top: 100px; padding-bottom: 200px" lay-filter="form1"> |
||||
|
<input style="display: none" th:value="${depositoryId}" name="depositoryId" id="depositoryID"> |
||||
|
<div class="layui-form-item"> |
||||
|
<label class="layui-form-label">起始行:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写行数" class="layui-input" |
||||
|
name="place_start_x" lay-verify="required"/> |
||||
|
</div> |
||||
|
<label class="layui-form-label">结束行:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写行数" class="layui-input" |
||||
|
name="place_end_x" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<!-- <div class="layui-form-item"> |
||||
|
<label class="layui-form-label">起始列:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写列数" class="layui-input" |
||||
|
name="place_start_y" lay-verify="required"/> |
||||
|
</div> |
||||
|
<label class="layui-form-label">结束列:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写列数" class="layui-input" |
||||
|
name="place_end_y" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div>--> |
||||
|
<div class="layui-form-item"> |
||||
|
<label class="layui-form-label">起始层:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写层数" class="layui-input" |
||||
|
name="place_start_z" lay-verify="required"/> |
||||
|
</div> |
||||
|
<label class="layui-form-label">结束层:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写层数" class="layui-input" |
||||
|
name="place_end_z" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="layui-form-item"> |
||||
|
<label class="layui-form-label">最大存放量:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写最大存放量" class="layui-input" |
||||
|
name="max" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="layui-form-item"> |
||||
|
<label class="layui-form-label">最小存放量:</label> |
||||
|
<div class="layui-input-block"> |
||||
|
<input type="text" placeholder="请填写最小存放量" class="layui-input" |
||||
|
name="min" lay-verify="required"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<div class="layui-form-item"> |
||||
|
<div class="layui-input-block"> |
||||
|
<button class="layui-btn" lay-submit lay-filter="formStep"> |
||||
|
 创建库位  |
||||
|
</button> |
||||
|
</div> |
||||
|
</div> |
||||
|
</form> |
||||
|
</div> |
||||
|
</div> |
||||
|
<hr> |
||||
|
</div> |
||||
|
</div> |
||||
|
</fieldset> |
||||
|
|
||||
|
</div> |
||||
|
</div> |
||||
|
|
||||
|
<script src="/static/lib/layui-v2.6.3/layui.js" charset="utf-8"></script> |
||||
|
|
||||
|
<script> |
||||
|
var data; |
||||
|
|
||||
|
layui.use(['form', 'layer','dropdown','tree'], function () { |
||||
|
var $ = layui.jquery, |
||||
|
form = layui.form, |
||||
|
layer = layui.layer; |
||||
|
|
||||
|
var depositoryId = $("#depositoryID").val(); |
||||
|
form.on('submit(formStep)', function (data) { |
||||
|
var req = data.field; |
||||
|
req.type = "list"; |
||||
|
$.ajax({ |
||||
|
url: "/place/addPlace", |
||||
|
type: 'post', |
||||
|
dataType: 'json', |
||||
|
contentType: "application/json;charset=utf-8", |
||||
|
data: JSON.stringify(req), |
||||
|
beforeSend: function () { |
||||
|
this.layerIndex = layer.load(0, {shade: [0.5, '#393D49']}); |
||||
|
}, |
||||
|
success: function (data) { |
||||
|
layer.close(this.layerIndex); |
||||
|
if (data.status >= 300) { |
||||
|
layer.msg(data.statusInfo.message);//失败的表情 |
||||
|
return; |
||||
|
} else { |
||||
|
layer.msg("添加成功!", { |
||||
|
icon: 6,//成功的表情 |
||||
|
time: 1000 |
||||
|
}, //1秒关闭(如果不配置,默认是3秒) |
||||
|
function(){ |
||||
|
//do something |
||||
|
window.location="/insertListPlace?depositoryId="+depositoryId |
||||
|
}); |
||||
|
} |
||||
|
}, |
||||
|
complete: function () { |
||||
|
form.val("form1", { |
||||
|
}) |
||||
|
} |
||||
|
}); |
||||
|
return false; |
||||
|
}); |
||||
|
|
||||
|
}); |
||||
|
</script> |
||||
|
|
||||
|
</body> |
||||
|
</html> |
||||
@ -0,0 +1,315 @@ |
|||||
|
<template> |
||||
|
<div class="page-scan"> |
||||
|
<div class="base-line"> |
||||
|
<div class="pullWrap"> |
||||
|
<div class="topTitle"> |
||||
|
<div class="pullTitle"> |
||||
|
<div class="pullName">二维码绑定</div> |
||||
|
<!-- 绑定状态图标 --> |
||||
|
<img class="left-icon" src="/static/img/noBind.svg" alt="" v-if="!dataObj.qrCodeId"> |
||||
|
<img class="left-icon" src="/static/img/binded.svg" alt="" v-else> |
||||
|
<div class="right-part"> |
||||
|
<input :disabled="dataObj.qrCodeId" type="text" v-model="dataObj.qrCodeId" placeholder="请输入二维码ID" v-if="dataObj.qrCodeId"> |
||||
|
<input type="text" v-model="scanTextData.scanText" placeholder="请输入二维码ID" v-else> |
||||
|
<img src="/static/img/scan.svg" alt="" v-if="!dataObj.qrCodeId"> |
||||
|
<span @click="toScanCode" v-if="!dataObj.qrCodeId">扫码填入</span> |
||||
|
<span v-if="dataObj.qrCodeId">解绑</span> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
<!-- 扫描盒子 --> |
||||
|
<div class="scan-box" v-if="scanTextData.showScanBox"> |
||||
|
<div class="scan-cacel" @click="cancelScan" v-show="scanTextData.showScanBoxInfo"> |
||||
|
<!-- <img src="" alt=""> --> |
||||
|
取消 |
||||
|
</div> |
||||
|
<video ref="video" id="video" class="scan-video" v-show="scanTextData.showScanBoxInfo" autoplay></video> |
||||
|
<div class="scan-img" v-show="scanTextData.showScanBoxInfo"> |
||||
|
<div class="scan-frame"> |
||||
|
<span class="left-t"></span> |
||||
|
<span class="right-t"></span> |
||||
|
<span class="left-b"></span> |
||||
|
<span class="right-b"></span> |
||||
|
<span class="cross-line"></span> |
||||
|
</div> |
||||
|
</div> |
||||
|
<!-- <img src="/static/img/scan.svg" alt="" v-show="scanTextData.showScanBoxInfo"> --> |
||||
|
<div class="scan-tip" v-show="scanTextData.showScanBoxInfo"> {{scanTextData.tipMsg}} </div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</template> |
||||
|
|
||||
|
|
||||
|
<script> |
||||
|
import { BrowserMultiFormatReader } from '@zxing/library'; |
||||
|
let scanTextData = { |
||||
|
loadingShow: false, |
||||
|
codeReader: null, |
||||
|
scanText: '', |
||||
|
vin: null, |
||||
|
tipMsg: '将二维码置于屏幕中,即可识别', |
||||
|
tipShow: false |
||||
|
} |
||||
|
export default { |
||||
|
name: 'scanCodePage', |
||||
|
data() { |
||||
|
return { |
||||
|
scanTextData:{ |
||||
|
loadingShow: false, |
||||
|
codeReader: null, |
||||
|
scanText: '', |
||||
|
vin: null, |
||||
|
tipMsg: '将二维码置于屏幕中,即可识别', |
||||
|
tipShow: false, |
||||
|
|
||||
|
showScanBox:false, |
||||
|
showScanBoxInfo:false, |
||||
|
}, |
||||
|
hasBind:false |
||||
|
} |
||||
|
}, |
||||
|
methods: { |
||||
|
toScanCode(){ |
||||
|
console.log('识别二维码',this.dataObj) |
||||
|
scanTextData.codeReader = new BrowserMultiFormatReader(); |
||||
|
this.scanTextData.showScanBox = true |
||||
|
this.openScan(); |
||||
|
|
||||
|
}, |
||||
|
cancelScan(){ |
||||
|
//识别完停止使用摄像头 |
||||
|
let thisVideo = document.getElementById("video"); |
||||
|
thisVideo.srcObject.getTracks()[0].stop() |
||||
|
scanTextData.codeReader.reset(); // 重置 |
||||
|
this.scanTextData.showScanBox = false |
||||
|
setTimeout(()=>{ |
||||
|
this.scanTextData.showScanBoxInfo = false |
||||
|
},1000) |
||||
|
}, |
||||
|
|
||||
|
async openScan() { |
||||
|
scanTextData.codeReader.getVideoInputDevices().then((videoInputDevices) => { |
||||
|
scanTextData.tipShow = true; |
||||
|
scanTextData.tipMsg = '正在调用摄像头...'; |
||||
|
console.log('videoInputDevices', videoInputDevices); |
||||
|
// 默认获取第一个摄像头设备id |
||||
|
let firstDeviceId = videoInputDevices[0].deviceId; |
||||
|
// 获取第一个摄像头设备的名称 |
||||
|
const videoInputDeviceslablestr = JSON.stringify(videoInputDevices[0].label); |
||||
|
if (videoInputDevices.length > 1) { |
||||
|
// 判断是否后置摄像头 |
||||
|
if (videoInputDeviceslablestr.indexOf('back') > -1) { |
||||
|
firstDeviceId = videoInputDevices[0].deviceId; |
||||
|
} else { |
||||
|
firstDeviceId = videoInputDevices[1].deviceId; |
||||
|
} |
||||
|
} |
||||
|
this.decodeFromInputVideoFunc(firstDeviceId); |
||||
|
}).catch(err => { |
||||
|
scanTextData.tipShow = false; |
||||
|
console.error(err); |
||||
|
}); |
||||
|
}, |
||||
|
decodeFromInputVideoFunc(firstDeviceId) { |
||||
|
scanTextData.codeReader.reset(); // 重置 |
||||
|
scanTextData.scanText = ''; |
||||
|
scanTextData.codeReader.decodeFromInputVideoDeviceContinuously(firstDeviceId, 'video', (result, err) => { |
||||
|
scanTextData.tipMsg = '将二维码置于屏幕中,即可识别'; |
||||
|
scanTextData.scanText = ''; |
||||
|
setTimeout(()=>{ |
||||
|
this.scanTextData.showScanBoxInfo = true |
||||
|
},1000) |
||||
|
if (result) { |
||||
|
console.log('扫描结果', result.text); |
||||
|
if (result.text) { |
||||
|
console.log('扫描结果11', result.text); |
||||
|
this.scanTextData.showScanBox = false |
||||
|
this.scanTextData.showScanBoxInfo = false |
||||
|
this.scanTextData.scanText = result.text |
||||
|
//这里扫描出结果可以调用你想要的方法 |
||||
|
//识别完停止使用摄像头 |
||||
|
let thisVideo = document.getElementById("video"); |
||||
|
thisVideo.srcObject.getTracks()[0].stop() |
||||
|
scanTextData.codeReader.reset(); // 重置 |
||||
|
} |
||||
|
}else{ |
||||
|
// console.log('没出来?',result,err) |
||||
|
} |
||||
|
if (err && !(err)) { |
||||
|
scanTextData.tipMsg = '识别失败'; |
||||
|
setTimeout(() => { |
||||
|
scanTextData.tipShow = false; |
||||
|
}, 2000) |
||||
|
console.error(err); |
||||
|
} |
||||
|
}); |
||||
|
}, |
||||
|
|
||||
|
}, |
||||
|
props:['dataObj'] |
||||
|
} |
||||
|
</script> |
||||
|
|
||||
|
|
||||
|
<style scoped lang="scss"> |
||||
|
.pullWrap { |
||||
|
width: 100%; |
||||
|
height: 100px; |
||||
|
padding-top: 50px; |
||||
|
background: #fff; |
||||
|
} |
||||
|
.topTitle { |
||||
|
width: 100%; |
||||
|
position: relative; |
||||
|
} |
||||
|
.pullTitle { |
||||
|
display: flex; |
||||
|
height: 80px; |
||||
|
line-height: 80px; |
||||
|
margin-top: -40px; |
||||
|
background: #fff; |
||||
|
align-items: center; |
||||
|
justify-content: space-between; |
||||
|
} |
||||
|
.pullName{ |
||||
|
font-size: 30px; |
||||
|
color: rgba(0,0,0,0.85); |
||||
|
} |
||||
|
img{ |
||||
|
width: 40px; |
||||
|
height: 40px; |
||||
|
} |
||||
|
.left-icon{ |
||||
|
margin:0 15px; |
||||
|
} |
||||
|
span{ |
||||
|
text-decoration: underline; |
||||
|
text-decoration-color: #42a5ff; |
||||
|
color: #42a5ff; |
||||
|
margin-left: 5px; |
||||
|
} |
||||
|
.right-part { |
||||
|
flex: 1; |
||||
|
display: flex; |
||||
|
align-items: center; |
||||
|
justify-content: space-around; |
||||
|
border-bottom: 1px solid #e5e5e5; |
||||
|
overflow: hidden; |
||||
|
text-overflow: ellipsis; |
||||
|
} |
||||
|
input{ |
||||
|
border:none; |
||||
|
flex: 1; |
||||
|
} |
||||
|
|
||||
|
|
||||
|
.scan-index-bar{ |
||||
|
background-image: linear-gradient( -45deg, #42a5ff ,#59cfff); |
||||
|
} |
||||
|
.van-nav-bar__title{ |
||||
|
color: #fff !important; |
||||
|
} |
||||
|
.scan-box { |
||||
|
position: fixed; |
||||
|
top: 0; |
||||
|
left: 0; |
||||
|
z-index: 5; |
||||
|
height: 100%; |
||||
|
width: 100vw; |
||||
|
} |
||||
|
.scan-cacel{ |
||||
|
position: absolute; |
||||
|
top: 30px; |
||||
|
left: 30px; |
||||
|
z-index: 9; |
||||
|
color: #fff; |
||||
|
font-size: 35px; |
||||
|
|
||||
|
} |
||||
|
.scan-video{ |
||||
|
height: 100vh; |
||||
|
width: 100vw; |
||||
|
object-fit:cover; |
||||
|
} |
||||
|
.scan-img { |
||||
|
width: 500px; |
||||
|
height: 500px; |
||||
|
position: fixed; |
||||
|
top: 40%; |
||||
|
left: 50%; |
||||
|
margin-top: -200px; |
||||
|
margin-left: -250px; |
||||
|
z-index: 6; |
||||
|
} |
||||
|
.scan-frame { |
||||
|
width: 100%; |
||||
|
height: 100%; |
||||
|
position: relative; |
||||
|
} |
||||
|
.left-t,.right-t,.left-b,.right-b{ |
||||
|
position: absolute; |
||||
|
width: 80px; |
||||
|
height: 80px; |
||||
|
} |
||||
|
.left-t{ |
||||
|
top: 0; |
||||
|
left: 0; |
||||
|
border-top:2px solid #17B1B7; |
||||
|
border-left:2px solid #17B1B7; |
||||
|
} |
||||
|
.right-t{ |
||||
|
top: 0; |
||||
|
right: 0; |
||||
|
border-top:2px solid #17B1B7; |
||||
|
border-right:2px solid #17B1B7; |
||||
|
} |
||||
|
.left-b{ |
||||
|
bottom: 0; |
||||
|
left: 0; |
||||
|
border-bottom:2px solid #17B1B7; |
||||
|
border-left:2px solid #17B1B7; |
||||
|
} |
||||
|
.right-b{ |
||||
|
bottom: 0; |
||||
|
right: 0; |
||||
|
border-bottom:2px solid #17B1B7; |
||||
|
border-right:2px solid #17B1B7; |
||||
|
} |
||||
|
.cross-line{ |
||||
|
width: 600px; |
||||
|
height: 10px; |
||||
|
background: linear-gradient(to right, rgba(255, 255, 255, 0),#5DDDD3,rgba(255,255,255,0)); |
||||
|
position: absolute; |
||||
|
top: 0; |
||||
|
left: -50px; |
||||
|
animation: identifier_p 5s infinite; |
||||
|
} |
||||
|
@keyframes identifier_p { |
||||
|
0%{ |
||||
|
top: 0%; |
||||
|
} |
||||
|
50%{ |
||||
|
top: 100%; |
||||
|
} |
||||
|
100%{ |
||||
|
top: 0; |
||||
|
} |
||||
|
} |
||||
|
.scan-tip{ |
||||
|
width: 100vw; |
||||
|
text-align: center; |
||||
|
margin-bottom: 10vh; |
||||
|
color: white; |
||||
|
font-size: 5vw; |
||||
|
position: absolute; |
||||
|
bottom: 50px; |
||||
|
left: 0; |
||||
|
color: #fff; |
||||
|
} |
||||
|
.page-scan{ |
||||
|
overflow-y: hidden; |
||||
|
// background-color: #363636; |
||||
|
} |
||||
|
</style> |
||||
Loading…
Reference in new issue