zhang_li
7 months ago
12 changed files with 17508 additions and 0 deletions
After Width: | Height: | Size: 42 KiB |
@ -0,0 +1,416 @@ |
|||
<template> |
|||
<view class="kk-printer"> |
|||
<view class="kk-printer-btn" @tap="handlePrintTap"> |
|||
{{isPrinting?printingText:defaultText}} |
|||
</view> |
|||
<view class="kk-shadow" :class="isShowSearch?'show':''" @tap="handleSearchClose"> |
|||
<view class="kk-modal" @tap.stop="doNothing"> |
|||
<view class="kk-search-device"> |
|||
<view class="kk-filter-wrap"> |
|||
<view class="filter-title">根据SRRI过滤设备</view> |
|||
<slider @change="handleSRRIChange" max='-20' min='-100' value="-95" show-value/> |
|||
</view> |
|||
<view class="kk-filter-wrap"> |
|||
<view class="filter-title">根据蓝牙名过滤</view> |
|||
<input type="text" placeholder-class="kk-placeholder-class" placeholder="请输入蓝牙名字或设备ID搜索" v-model="filterName" /> |
|||
</view> |
|||
<view class="kk-btn-wrap"> |
|||
<view class="kk-btn-item confirm-btn" @tap="searchBtnTap" v-if="!isSearching"> |
|||
搜索设备 |
|||
</view> |
|||
<view class="kk-btn-item confirm-btn" v-else> |
|||
搜索中... |
|||
</view> |
|||
<view class="kk-btn-item" @tap="stopSearchBtnTap"> |
|||
停止搜索 |
|||
</view> |
|||
</view> |
|||
<view class="kk-devices-wrap"> |
|||
<view class="empty-wrap" v-if="filterDeviceList.length <= 0"> |
|||
<view class="empty-icon"></view> |
|||
<view class="empty-text">~ 无可搜索到的设备 ~</view> |
|||
</view> |
|||
<view class="" v-else> |
|||
<view class="kk-devices-item" v-for="(item,index) in filterDeviceList" :key="index" @tap="handleConnectDevice(item)"> |
|||
<view class="name"> |
|||
<text>设备名称:</text> |
|||
<text>{{item.name?item.name:'未命名'}}</text> |
|||
</view> |
|||
<view class="rssi"> |
|||
<text>信号强度:</text> |
|||
<text>({{Math.max(0, item.RSSI + 100)}}%)</text> |
|||
</view> |
|||
<view class="deviceid"> |
|||
<text>设备ID:</text> |
|||
<text>{{item.deviceId}}</text> |
|||
</view> |
|||
<view class="advmac" v-if="item.advMac"> |
|||
<text>advMac:</text> |
|||
<text>{{item.advMac}}</text> |
|||
</view> |
|||
</view> |
|||
</view> |
|||
</view> |
|||
</view> |
|||
</view> |
|||
</view> |
|||
</view> |
|||
</template> |
|||
|
|||
<script> |
|||
import * as gbk from '@/components/kk-printer/utils/printUtil-GBK.js'; |
|||
import * as blesdk from './utils/bluetoolth'; |
|||
import * as util from './utils/util.js'; |
|||
export default { |
|||
data(){ |
|||
return{ |
|||
//是否正在打印 |
|||
isPrinting:false, |
|||
//是否正在搜索设备 |
|||
isSearching:false, |
|||
//是否显示蓝牙列表 |
|||
isShowSearch:false, |
|||
//按蓝牙名过滤 |
|||
filterName:'', |
|||
//按信号过滤 |
|||
filterRSSI:-95, |
|||
//设备列表 |
|||
devicesList:[], |
|||
//连接的设备ID |
|||
deviceId:'', |
|||
//根据设备ID获取的服务 |
|||
services:'', |
|||
//获取特征值时返回的三要素 |
|||
serviceId: '', |
|||
writeId: '', |
|||
readId: '' |
|||
} |
|||
}, |
|||
props:{ |
|||
//按钮默认文字 |
|||
defaultText:{ |
|||
type:String, |
|||
default:'打印' |
|||
}, |
|||
//按钮打印中的文字 |
|||
printingText:{ |
|||
type:String, |
|||
default:'打印中...' |
|||
}, |
|||
bufferData:{ |
|||
type:String, |
|||
require:true |
|||
} |
|||
}, |
|||
computed:{ |
|||
mapFilterRSSI(){ |
|||
return (0 - this.filterRSSI) |
|||
}, |
|||
filterDeviceList(){ |
|||
let devices = this.devicesList; |
|||
let name = this.filterName; |
|||
let rssi = this.filterRSSI; |
|||
console.log(devices,name,rssi) |
|||
//按RSSI过滤 |
|||
let filterDevices1 = devices.filter((item)=>{ |
|||
return item.RSSI > rssi |
|||
}) |
|||
console.log(filterDevices1) |
|||
// 按名字过滤 |
|||
let filterDevices2 |
|||
if(name!=''){ |
|||
filterDevices2 = filterDevices1.filter((item)=>{ |
|||
return (item.name.indexOf(name) >= 0 || item.deviceId.indexOf(name) >= 0) |
|||
}) |
|||
}else{ |
|||
filterDevices2 = filterDevices1 |
|||
} |
|||
// 根据广播数据提取MAC地址 |
|||
for (let i = 0; i < filterDevices2.length;i++) { |
|||
if (filterDevices2[i].hasOwnProperty('advertisData')){ |
|||
if (filterDevices2[i].advertisData.byteLength == 8) { |
|||
filterDevices2[i].advMac = util.buf2hex(filterDevices2[i].advertisData.slice(2, 7)); |
|||
} |
|||
} |
|||
} |
|||
return filterDevices2 |
|||
} |
|||
}, |
|||
mounted() { |
|||
|
|||
}, |
|||
beforeDestroy(){ |
|||
this.stopSearchBtnTap(); |
|||
}, |
|||
methods:{ |
|||
doNothing(){ |
|||
return; |
|||
}, |
|||
//点击打印按钮 |
|||
handlePrintTap(){ |
|||
console.log(11) |
|||
//打开蓝牙适配器 |
|||
blesdk.openBlue().then((res)=>{ |
|||
console.log(99,res) |
|||
//获取已连接设备 |
|||
blesdk.getConnectedBluetoothDevices().then((res)=>{ |
|||
//若没有已连接设备,弹框搜索设备 |
|||
console.log(66,res,this.deviceId,this.serviceId,this.writeId,this.bufferData,this.onPrintSuccess) |
|||
if(res.devices.length == 0){ |
|||
this.isShowSearch = true |
|||
}else{ |
|||
let datalen=20; |
|||
if (plus.os.name != 'Android') |
|||
{ |
|||
datalen=180; |
|||
} |
|||
this.isPrinting = true; |
|||
this.$emit('onPrint'); |
|||
this.$nextTick(()=>{ |
|||
console.log(1,this.bufferData) |
|||
if(this.bufferData!=''){ |
|||
let buffer = gbk.strToGBKByte(this.bufferData) |
|||
console.log(2,buffer) |
|||
let opt = { |
|||
deviceId: this.deviceId, |
|||
serviceId: this.serviceId, |
|||
characteristicId: this.writeId, |
|||
value:buffer, |
|||
lasterSuccess: this.onPrintSuccess, |
|||
onceLength:datalen |
|||
} |
|||
console.log(3,opt) |
|||
blesdk.sendDataToDevice(opt); |
|||
this.isPrinting = false; |
|||
} |
|||
}) |
|||
} |
|||
}).catch((err)=>{ |
|||
console.log(88,err) |
|||
blesdk.catchToast(err); |
|||
}) |
|||
}).catch((err)=>{ |
|||
console.log(77,err) |
|||
blesdk.catchToast(err); |
|||
}) |
|||
}, |
|||
onGetDevice(res){ |
|||
this.devicesList = res; |
|||
}, |
|||
handleSearchClose(){ |
|||
this.isShowSearch = false |
|||
}, |
|||
handleSRRIChange(e){ |
|||
this.filterRSSI = e.detail.value |
|||
}, |
|||
//开始搜索设备 |
|||
searchBtnTap(){ |
|||
blesdk.startBluetoothDevicesDiscovery(); |
|||
this.isSearching = true; |
|||
blesdk.onfindBlueDevices(this.onGetDevice) |
|||
}, |
|||
//停止搜索设备 |
|||
stopSearchBtnTap(){ |
|||
blesdk.stopBlueDevicesDiscovery(); |
|||
this.isSearching = false; |
|||
}, |
|||
//点击连接设备 |
|||
handleConnectDevice(device){ |
|||
|
|||
let deviceId = device.deviceId; |
|||
let name = device.name; |
|||
this.deviceId = deviceId; |
|||
console.log('deviceId',this.deviceId) |
|||
// uni.setStorageSync('k_curDeviceID',deviceId); |
|||
// uni.setStorageSync('k_curDeviceName',name); |
|||
uni.onBLEConnectionStateChange((res)=>{ |
|||
console.log('连接',res) |
|||
if(res.connected){ |
|||
plus.nativeUI.toast('设备'+ res.deviceId + '已连接',{ |
|||
verticalAlign:'center' |
|||
}) |
|||
}else{ |
|||
plus.nativeUI.toast('设备'+ res.deviceId + '已断开连接',{ |
|||
verticalAlign:'center' |
|||
}) |
|||
} |
|||
}) |
|||
blesdk.createBLEConnection(deviceId, this.onConnectSuccess, this.onConnectFail); |
|||
}, |
|||
onConnectSuccess(res){ |
|||
this.stopSearchBtnTap() |
|||
blesdk.getBLEDeviceServices(this.deviceId, this.onGetServicesSuccess, this.onGetServicesFail); |
|||
}, |
|||
onConnectFail(err){ |
|||
console.log('链接失败',err) |
|||
}, |
|||
onGetServicesSuccess(res){ |
|||
console.log('获取服务',res) |
|||
this.services = res.serviceId; |
|||
blesdk.getDeviceCharacteristics(this.deviceId, this.services, this.onGetCharacterSuccess, this.onGetCharacterFail); |
|||
}, |
|||
onGetServicesFail(err){ |
|||
console.log('获取服务失败') |
|||
}, |
|||
onGetCharacterSuccess(res){ |
|||
console.log('获取特征值成功',res) |
|||
this.serviceId = res.serviceId; |
|||
this.writeId = res.writeId; |
|||
this.readId = res.readId; |
|||
this.isShowSearch = false; |
|||
}, |
|||
onGetCharacterFail(err){ |
|||
console.log('特征值获取失败') |
|||
}, |
|||
onPrintSuccess(){ |
|||
this.isPrinting = false; |
|||
console.log('打印成功') |
|||
this.$emit('onPrintSuccess') |
|||
}, |
|||
onPrintFail(){ |
|||
console.log('打印失败') |
|||
this.isPrinting = false; |
|||
} |
|||
} |
|||
} |
|||
</script> |
|||
|
|||
<style lang="scss" scoped> |
|||
.kk-printer{ |
|||
width:100%; |
|||
height:100%; |
|||
&-btn{ |
|||
width:100%; |
|||
height:100%; |
|||
text-align: center; |
|||
line-height: 50px; |
|||
} |
|||
.kk-shadow{ |
|||
display: none; |
|||
&.show{ |
|||
display: block; |
|||
width:100vw; |
|||
height:100vh; |
|||
background: rgba(0,0,0,0.4); |
|||
position: fixed; |
|||
top: 0; |
|||
left: 0; |
|||
display: flex; |
|||
justify-content: center; |
|||
align-items: center; |
|||
.kk-modal{ |
|||
width:680upx; |
|||
height: 80%; |
|||
padding:24upx; |
|||
box-sizing: border-box; |
|||
overflow-y: auto; |
|||
border-radius: 20upx; |
|||
background: #ffffff; |
|||
display: flex; |
|||
justify-content: center; |
|||
align-items: center; |
|||
.kk-search-device{ |
|||
width:100%; |
|||
height: 100%; |
|||
.kk-filter-wrap{ |
|||
width:100%; |
|||
height: 160upx; |
|||
display: flex; |
|||
flex-direction: column; |
|||
justify-content: flex-start; |
|||
align-items: flex-start; |
|||
.filter-title{ |
|||
line-height: 70upx; |
|||
font-size: 30upx; |
|||
color: #999999; |
|||
} |
|||
&>slider{ |
|||
width:90%; |
|||
height: 90upx; |
|||
} |
|||
&>input{ |
|||
padding:0 20upx; |
|||
box-sizing: border-box; |
|||
border-radius: 10upx; |
|||
height: 90upx; |
|||
width:100%; |
|||
border: 1upx solid #ebebeb; |
|||
} |
|||
} |
|||
.kk-btn-wrap{ |
|||
width:100%; |
|||
height: 140upx; |
|||
display: flex; |
|||
justify-content: space-between; |
|||
align-items: center; |
|||
&>view{ |
|||
flex:1 1 auto; |
|||
height: 100upx; |
|||
line-height: 100upx; |
|||
border-radius: 16upx; |
|||
text-align: center; |
|||
color:#ffffff; |
|||
&.confirm-btn{ |
|||
background: #007AFF; |
|||
margin-right:30upx; |
|||
} |
|||
&:nth-last-child(1){ |
|||
background: #DD524D; |
|||
} |
|||
} |
|||
} |
|||
.kk-devices-wrap{ |
|||
height: calc(100% - 460upx); |
|||
overflow-y:auto; |
|||
padding:10upx 20upx; |
|||
box-sizing: border-box; |
|||
border: 1upx solid #ebebeb; |
|||
box-sizing: border-box; |
|||
border-radius: 20upx; |
|||
.empty-wrap{ |
|||
width:100%; |
|||
height: 100%; |
|||
display: flex; |
|||
flex-direction: column; |
|||
justify-content: center; |
|||
align-items: center; |
|||
.empty-icon{ |
|||
width:268upx; |
|||
height: 240upx; |
|||
background: url('./empty-icon.png') no-repeat; |
|||
background-size:100% 100%; |
|||
margin-bottom: 26upx; |
|||
} |
|||
.empty-text{ |
|||
width: 100%; |
|||
line-height: 50upx; |
|||
font-size: 30upx; |
|||
text-align: center; |
|||
color: #999999; |
|||
} |
|||
} |
|||
.kk-devices-item{ |
|||
width:100%; |
|||
border-bottom: 1upx solid #ebebeb; |
|||
padding:10upx 0; |
|||
box-sizing: border-box; |
|||
&:nth-last-child(1){ |
|||
border-bottom: none; |
|||
} |
|||
&>view{ |
|||
width:100%; |
|||
font-size: 30upx; |
|||
} |
|||
} |
|||
} |
|||
} |
|||
} |
|||
} |
|||
|
|||
} |
|||
} |
|||
.kk-placeholder-class{ |
|||
font-size: 30upx; |
|||
color:#999999; |
|||
} |
|||
</style> |
@ -0,0 +1,402 @@ |
|||
var CHAR_TILDE = 126; |
|||
var CODE_FNC1 = 102; |
|||
|
|||
var SET_STARTA = 103; |
|||
var SET_STARTB = 104; |
|||
var SET_STARTC = 105; |
|||
var SET_SHIFT = 98; |
|||
var SET_CODEA = 101; |
|||
var SET_CODEB = 100; |
|||
var SET_STOP = 106; |
|||
|
|||
|
|||
var REPLACE_CODES = { |
|||
CHAR_TILDE: CODE_FNC1 //~ corresponds to FNC1 in GS1-128 standard
|
|||
} |
|||
|
|||
var CODESET = { |
|||
ANY: 1, |
|||
AB: 2, |
|||
A: 3, |
|||
B: 4, |
|||
C: 5 |
|||
}; |
|||
|
|||
function getBytes(str) { |
|||
var bytes = []; |
|||
for (var i = 0; i < str.length; i++) { |
|||
bytes.push(str.charCodeAt(i)); |
|||
} |
|||
return bytes; |
|||
} |
|||
|
|||
exports.code128 = function (ctx, text, width, height) { |
|||
|
|||
width = parseInt(width); |
|||
|
|||
height = parseInt(height); |
|||
|
|||
var codes = stringToCode128(text); |
|||
|
|||
var g = new Graphics(ctx, width, height); |
|||
|
|||
var barWeight = g.area.width / ((codes.length - 3) * 11 + 35); |
|||
|
|||
var x = g.area.left; |
|||
var y = g.area.top; |
|||
for (var i = 0; i < codes.length; i++) { |
|||
var c = codes[i]; |
|||
//two bars at a time: 1 black and 1 white
|
|||
for (var bar = 0; bar < 8; bar += 2) { |
|||
var barW = PATTERNS[c][bar] * barWeight; |
|||
// var barH = height - y - this.border;
|
|||
var barH = height - y; |
|||
var spcW = PATTERNS[c][bar + 1] * barWeight; |
|||
|
|||
//no need to draw if 0 width
|
|||
if (barW > 0) { |
|||
g.fillFgRect(x, y, barW, barH); |
|||
} |
|||
|
|||
x += barW + spcW; |
|||
} |
|||
} |
|||
|
|||
ctx.draw(); |
|||
} |
|||
|
|||
|
|||
function stringToCode128(text) { |
|||
|
|||
var barc = { |
|||
currcs: CODESET.C |
|||
}; |
|||
|
|||
var bytes = getBytes(text); |
|||
//decide starting codeset
|
|||
var index = bytes[0] == CHAR_TILDE ? 1 : 0; |
|||
|
|||
var csa1 = bytes.length > 0 ? codeSetAllowedFor(bytes[index++]) : CODESET.AB; |
|||
var csa2 = bytes.length > 0 ? codeSetAllowedFor(bytes[index++]) : CODESET.AB; |
|||
barc.currcs = getBestStartSet(csa1, csa2); |
|||
barc.currcs = perhapsCodeC(bytes, barc.currcs); |
|||
|
|||
//if no codeset changes this will end up with bytes.length+3
|
|||
//start, checksum and stop
|
|||
var codes = new Array(); |
|||
|
|||
switch (barc.currcs) { |
|||
case CODESET.A: |
|||
codes.push(SET_STARTA); |
|||
break; |
|||
case CODESET.B: |
|||
codes.push(SET_STARTB); |
|||
break; |
|||
default: |
|||
codes.push(SET_STARTC); |
|||
break; |
|||
} |
|||
|
|||
|
|||
for (var i = 0; i < bytes.length; i++) { |
|||
var b1 = bytes[i]; //get the first of a pair
|
|||
//should we translate/replace
|
|||
if (b1 in REPLACE_CODES) { |
|||
codes.push(REPLACE_CODES[b1]); |
|||
i++ //jump to next
|
|||
b1 = bytes[i]; |
|||
} |
|||
|
|||
//get the next in the pair if possible
|
|||
var b2 = bytes.length > (i + 1) ? bytes[i + 1] : -1; |
|||
|
|||
codes = codes.concat(codesForChar(b1, b2, barc.currcs)); |
|||
//code C takes 2 chars each time
|
|||
if (barc.currcs == CODESET.C) i++; |
|||
} |
|||
|
|||
//calculate checksum according to Code 128 standards
|
|||
var checksum = codes[0]; |
|||
for (var weight = 1; weight < codes.length; weight++) { |
|||
checksum += (weight * codes[weight]); |
|||
} |
|||
codes.push(checksum % 103); |
|||
|
|||
codes.push(SET_STOP); |
|||
|
|||
//encoding should now be complete
|
|||
return codes; |
|||
|
|||
function getBestStartSet(csa1, csa2) { |
|||
//tries to figure out the best codeset
|
|||
//to start with to get the most compact code
|
|||
var vote = 0; |
|||
vote += csa1 == CODESET.A ? 1 : 0; |
|||
vote += csa1 == CODESET.B ? -1 : 0; |
|||
vote += csa2 == CODESET.A ? 1 : 0; |
|||
vote += csa2 == CODESET.B ? -1 : 0; |
|||
//tie goes to B due to my own predudices
|
|||
return vote > 0 ? CODESET.A : CODESET.B; |
|||
} |
|||
|
|||
function perhapsCodeC(bytes, codeset) { |
|||
for (var i = 0; i < bytes.length; i++) { |
|||
var b = bytes[i] |
|||
if ((b < 48 || b > 57) && b != CHAR_TILDE) |
|||
return codeset; |
|||
} |
|||
return CODESET.C; |
|||
} |
|||
|
|||
//chr1 is current byte
|
|||
//chr2 is the next byte to process. looks ahead.
|
|||
function codesForChar(chr1, chr2, currcs) { |
|||
var result = []; |
|||
var shifter = -1; |
|||
|
|||
if (charCompatible(chr1, currcs)) { |
|||
if (currcs == CODESET.C) { |
|||
if (chr2 == -1) { |
|||
shifter = SET_CODEB; |
|||
currcs = CODESET.B; |
|||
} |
|||
else if ((chr2 != -1) && !charCompatible(chr2, currcs)) { |
|||
//need to check ahead as well
|
|||
if (charCompatible(chr2, CODESET.A)) { |
|||
shifter = SET_CODEA; |
|||
currcs = CODESET.A; |
|||
} |
|||
else { |
|||
shifter = SET_CODEB; |
|||
currcs = CODESET.B; |
|||
} |
|||
} |
|||
} |
|||
} |
|||
else { |
|||
//if there is a next char AND that next char is also not compatible
|
|||
if ((chr2 != -1) && !charCompatible(chr2, currcs)) { |
|||
//need to switch code sets
|
|||
switch (currcs) { |
|||
case CODESET.A: |
|||
shifter = SET_CODEB; |
|||
currcs = CODESET.B; |
|||
break; |
|||
case CODESET.B: |
|||
shifter = SET_CODEA; |
|||
currcs = CODESET.A; |
|||
break; |
|||
} |
|||
} |
|||
else { |
|||
//no need to shift code sets, a temporary SHIFT will suffice
|
|||
shifter = SET_SHIFT; |
|||
} |
|||
} |
|||
|
|||
//ok some type of shift is nessecary
|
|||
if (shifter != -1) { |
|||
result.push(shifter); |
|||
result.push(codeValue(chr2)); |
|||
} |
|||
else { |
|||
if (currcs == CODESET.C) { |
|||
//include next as well
|
|||
result.push(codeValue(chr1, chr2)); |
|||
} |
|||
else { |
|||
result.push(codeValue(chr1)); |
|||
} |
|||
} |
|||
barc.currcs = currcs; |
|||
|
|||
return result; |
|||
} |
|||
} |
|||
|
|||
//reduce the ascii code to fit into the Code128 char table
|
|||
function codeValue(chr1, chr2) { |
|||
if (typeof chr2 == "undefined") { |
|||
return chr1 >= 32 ? chr1 - 32 : chr1 + 64; |
|||
} |
|||
else { |
|||
return parseInt(String.fromCharCode(chr1) + String.fromCharCode(chr2)); |
|||
} |
|||
} |
|||
|
|||
function charCompatible(chr, codeset) { |
|||
var csa = codeSetAllowedFor(chr); |
|||
if (csa == CODESET.ANY) return true; |
|||
//if we need to change from current
|
|||
if (csa == CODESET.AB) return true; |
|||
if (csa == CODESET.A && codeset == CODESET.A) return true; |
|||
if (csa == CODESET.B && codeset == CODESET.B) return true; |
|||
return false; |
|||
} |
|||
|
|||
function codeSetAllowedFor(chr) { |
|||
if (chr >= 48 && chr <= 57) { |
|||
//0-9
|
|||
return CODESET.ANY; |
|||
} |
|||
else if (chr >= 32 && chr <= 95) { |
|||
//0-9 A-Z
|
|||
return CODESET.AB; |
|||
} |
|||
else { |
|||
//if non printable
|
|||
return chr < 32 ? CODESET.A : CODESET.B; |
|||
} |
|||
} |
|||
|
|||
var Graphics = function(ctx, width, height) { |
|||
|
|||
this.width = width; |
|||
this.height = height; |
|||
this.quiet = Math.round(this.width / 40); |
|||
|
|||
this.border_size = 0; |
|||
this.padding_width = 0; |
|||
|
|||
this.area = { |
|||
width : width - this.padding_width * 2 - this.quiet * 2, |
|||
height: height - this.border_size * 2, |
|||
top : this.border_size - 4, |
|||
left : this.padding_width + this.quiet |
|||
}; |
|||
|
|||
this.ctx = ctx; |
|||
this.fg = "#000000"; |
|||
this.bg = "#ffffff"; |
|||
|
|||
// fill background
|
|||
this.fillBgRect(0,0, width, height); |
|||
|
|||
// fill center to create border
|
|||
this.fillBgRect(0, this.border_size, width, height - this.border_size * 2); |
|||
} |
|||
|
|||
//use native color
|
|||
Graphics.prototype._fillRect = function(x, y, width, height, color) { |
|||
this.ctx.setFillStyle(color) |
|||
this.ctx.fillRect(x, y, width, height) |
|||
} |
|||
|
|||
Graphics.prototype.fillFgRect = function(x,y, width, height) { |
|||
this._fillRect(x, y, width, height, this.fg); |
|||
} |
|||
|
|||
Graphics.prototype.fillBgRect = function(x,y, width, height) { |
|||
this._fillRect(x, y, width, height, this.bg); |
|||
} |
|||
|
|||
var PATTERNS = [ |
|||
[2, 1, 2, 2, 2, 2, 0, 0], // 0
|
|||
[2, 2, 2, 1, 2, 2, 0, 0], // 1
|
|||
[2, 2, 2, 2, 2, 1, 0, 0], // 2
|
|||
[1, 2, 1, 2, 2, 3, 0, 0], // 3
|
|||
[1, 2, 1, 3, 2, 2, 0, 0], // 4
|
|||
[1, 3, 1, 2, 2, 2, 0, 0], // 5
|
|||
[1, 2, 2, 2, 1, 3, 0, 0], // 6
|
|||
[1, 2, 2, 3, 1, 2, 0, 0], // 7
|
|||
[1, 3, 2, 2, 1, 2, 0, 0], // 8
|
|||
[2, 2, 1, 2, 1, 3, 0, 0], // 9
|
|||
[2, 2, 1, 3, 1, 2, 0, 0], // 10
|
|||
[2, 3, 1, 2, 1, 2, 0, 0], // 11
|
|||
[1, 1, 2, 2, 3, 2, 0, 0], // 12
|
|||
[1, 2, 2, 1, 3, 2, 0, 0], // 13
|
|||
[1, 2, 2, 2, 3, 1, 0, 0], // 14
|
|||
[1, 1, 3, 2, 2, 2, 0, 0], // 15
|
|||
[1, 2, 3, 1, 2, 2, 0, 0], // 16
|
|||
[1, 2, 3, 2, 2, 1, 0, 0], // 17
|
|||
[2, 2, 3, 2, 1, 1, 0, 0], // 18
|
|||
[2, 2, 1, 1, 3, 2, 0, 0], // 19
|
|||
[2, 2, 1, 2, 3, 1, 0, 0], // 20
|
|||
[2, 1, 3, 2, 1, 2, 0, 0], // 21
|
|||
[2, 2, 3, 1, 1, 2, 0, 0], // 22
|
|||
[3, 1, 2, 1, 3, 1, 0, 0], // 23
|
|||
[3, 1, 1, 2, 2, 2, 0, 0], // 24
|
|||
[3, 2, 1, 1, 2, 2, 0, 0], // 25
|
|||
[3, 2, 1, 2, 2, 1, 0, 0], // 26
|
|||
[3, 1, 2, 2, 1, 2, 0, 0], // 27
|
|||
[3, 2, 2, 1, 1, 2, 0, 0], // 28
|
|||
[3, 2, 2, 2, 1, 1, 0, 0], // 29
|
|||
[2, 1, 2, 1, 2, 3, 0, 0], // 30
|
|||
[2, 1, 2, 3, 2, 1, 0, 0], // 31
|
|||
[2, 3, 2, 1, 2, 1, 0, 0], // 32
|
|||
[1, 1, 1, 3, 2, 3, 0, 0], // 33
|
|||
[1, 3, 1, 1, 2, 3, 0, 0], // 34
|
|||
[1, 3, 1, 3, 2, 1, 0, 0], // 35
|
|||
[1, 1, 2, 3, 1, 3, 0, 0], // 36
|
|||
[1, 3, 2, 1, 1, 3, 0, 0], // 37
|
|||
[1, 3, 2, 3, 1, 1, 0, 0], // 38
|
|||
[2, 1, 1, 3, 1, 3, 0, 0], // 39
|
|||
[2, 3, 1, 1, 1, 3, 0, 0], // 40
|
|||
[2, 3, 1, 3, 1, 1, 0, 0], // 41
|
|||
[1, 1, 2, 1, 3, 3, 0, 0], // 42
|
|||
[1, 1, 2, 3, 3, 1, 0, 0], // 43
|
|||
[1, 3, 2, 1, 3, 1, 0, 0], // 44
|
|||
[1, 1, 3, 1, 2, 3, 0, 0], // 45
|
|||
[1, 1, 3, 3, 2, 1, 0, 0], // 46
|
|||
[1, 3, 3, 1, 2, 1, 0, 0], // 47
|
|||
[3, 1, 3, 1, 2, 1, 0, 0], // 48
|
|||
[2, 1, 1, 3, 3, 1, 0, 0], // 49
|
|||
[2, 3, 1, 1, 3, 1, 0, 0], // 50
|
|||
[2, 1, 3, 1, 1, 3, 0, 0], // 51
|
|||
[2, 1, 3, 3, 1, 1, 0, 0], // 52
|
|||
[2, 1, 3, 1, 3, 1, 0, 0], // 53
|
|||
[3, 1, 1, 1, 2, 3, 0, 0], // 54
|
|||
[3, 1, 1, 3, 2, 1, 0, 0], // 55
|
|||
[3, 3, 1, 1, 2, 1, 0, 0], // 56
|
|||
[3, 1, 2, 1, 1, 3, 0, 0], // 57
|
|||
[3, 1, 2, 3, 1, 1, 0, 0], // 58
|
|||
[3, 3, 2, 1, 1, 1, 0, 0], // 59
|
|||
[3, 1, 4, 1, 1, 1, 0, 0], // 60
|
|||
[2, 2, 1, 4, 1, 1, 0, 0], // 61
|
|||
[4, 3, 1, 1, 1, 1, 0, 0], // 62
|
|||
[1, 1, 1, 2, 2, 4, 0, 0], // 63
|
|||
[1, 1, 1, 4, 2, 2, 0, 0], // 64
|
|||
[1, 2, 1, 1, 2, 4, 0, 0], // 65
|
|||
[1, 2, 1, 4, 2, 1, 0, 0], // 66
|
|||
[1, 4, 1, 1, 2, 2, 0, 0], // 67
|
|||
[1, 4, 1, 2, 2, 1, 0, 0], // 68
|
|||
[1, 1, 2, 2, 1, 4, 0, 0], // 69
|
|||
[1, 1, 2, 4, 1, 2, 0, 0], // 70
|
|||
[1, 2, 2, 1, 1, 4, 0, 0], // 71
|
|||
[1, 2, 2, 4, 1, 1, 0, 0], // 72
|
|||
[1, 4, 2, 1, 1, 2, 0, 0], // 73
|
|||
[1, 4, 2, 2, 1, 1, 0, 0], // 74
|
|||
[2, 4, 1, 2, 1, 1, 0, 0], // 75
|
|||
[2, 2, 1, 1, 1, 4, 0, 0], // 76
|
|||
[4, 1, 3, 1, 1, 1, 0, 0], // 77
|
|||
[2, 4, 1, 1, 1, 2, 0, 0], // 78
|
|||
[1, 3, 4, 1, 1, 1, 0, 0], // 79
|
|||
[1, 1, 1, 2, 4, 2, 0, 0], // 80
|
|||
[1, 2, 1, 1, 4, 2, 0, 0], // 81
|
|||
[1, 2, 1, 2, 4, 1, 0, 0], // 82
|
|||
[1, 1, 4, 2, 1, 2, 0, 0], // 83
|
|||
[1, 2, 4, 1, 1, 2, 0, 0], // 84
|
|||
[1, 2, 4, 2, 1, 1, 0, 0], // 85
|
|||
[4, 1, 1, 2, 1, 2, 0, 0], // 86
|
|||
[4, 2, 1, 1, 1, 2, 0, 0], // 87
|
|||
[4, 2, 1, 2, 1, 1, 0, 0], // 88
|
|||
[2, 1, 2, 1, 4, 1, 0, 0], // 89
|
|||
[2, 1, 4, 1, 2, 1, 0, 0], // 90
|
|||
[4, 1, 2, 1, 2, 1, 0, 0], // 91
|
|||
[1, 1, 1, 1, 4, 3, 0, 0], // 92
|
|||
[1, 1, 1, 3, 4, 1, 0, 0], // 93
|
|||
[1, 3, 1, 1, 4, 1, 0, 0], // 94
|
|||
[1, 1, 4, 1, 1, 3, 0, 0], // 95
|
|||
[1, 1, 4, 3, 1, 1, 0, 0], // 96
|
|||
[4, 1, 1, 1, 1, 3, 0, 0], // 97
|
|||
[4, 1, 1, 3, 1, 1, 0, 0], // 98
|
|||
[1, 1, 3, 1, 4, 1, 0, 0], // 99
|
|||
[1, 1, 4, 1, 3, 1, 0, 0], // 100
|
|||
[3, 1, 1, 1, 4, 1, 0, 0], // 101
|
|||
[4, 1, 1, 1, 3, 1, 0, 0], // 102
|
|||
[2, 1, 1, 4, 1, 2, 0, 0], // 103
|
|||
[2, 1, 1, 2, 1, 4, 0, 0], // 104
|
|||
[2, 1, 1, 2, 3, 2, 0, 0], // 105
|
|||
[2, 3, 3, 1, 1, 1, 2, 0] // 106
|
|||
] |
|||
|
@ -0,0 +1,723 @@ |
|||
/** |
|||
* @export |
|||
* @param {string} name 微信api的名称 ,如 uniAsyncPromise("getSystemInfo",options) |
|||
* @param {object} options 除了success 和 fail 的其他参数 |
|||
* @returns |
|||
*/ |
|||
export function uniAsyncPromise(name, options) { |
|||
return new Promise((resolve, reject) => { |
|||
uni[name]({ |
|||
...(options || {}), |
|||
// ...options,
|
|||
success: (res) => { |
|||
resolve(res); |
|||
}, |
|||
fail: (err) => { |
|||
reject(err); |
|||
} |
|||
}); |
|||
}); |
|||
} |
|||
//微信小程序向蓝牙打印机发送数据进行打印的坑:
|
|||
//小程序api向蓝牙打印机发送数据打印,发送的任何内容都应该要转成二进制数据,而且蓝牙打印的文本编码是GBK的,发送中文需转成GBK编码再转成二进制数据发送
|
|||
//发送打印机指令也要转成二进制数据发送
|
|||
//蓝牙打印机一次接收的二级制数据有限制,不同的系统不同的蓝牙设备限制可能不同,微信建议一次20个字节,需做递归分包发送
|
|||
//发送完要打印的内容后,一定要发送一个打印的指令才能顺利打印 (有些指令就不需要)
|
|||
|
|||
//一、初始化蓝牙、开始检索蓝牙设备
|
|||
// { allowDuplicatesKey: true, interval: 500}
|
|||
export function openBlue() { |
|||
return uniAsyncPromise('openBluetoothAdapter') |
|||
} |
|||
|
|||
export function startBluetoothDevicesDiscovery(option) { |
|||
console.log('开始蓝牙扫描'); |
|||
uniAsyncPromise('startBluetoothDevicesDiscovery', option).then((res) => { |
|||
console.log('正在搜寻蓝牙设备', res); |
|||
}); |
|||
} |
|||
|
|||
export function getConnectedBluetoothDevices(option) { |
|||
console.log('开始获取已连接设备'); |
|||
return uniAsyncPromise('getConnectedBluetoothDevices', option) |
|||
} |
|||
//二、
|
|||
/** |
|||
* |
|||
* |
|||
* @export |
|||
* @param {function} getDevices uni.getBluetoothDevices的监听回调函数 |
|||
*/ |
|||
export function onfindBlueDevices(getDevices) { |
|||
//监听寻找到新设备的事件
|
|||
uni.onBluetoothDeviceFound((devices)=>{ |
|||
//获取在蓝牙模块生效期间所有已发现的蓝牙设备
|
|||
uniAsyncPromise('getBluetoothDevices').then((res) => { |
|||
getDevices && getDevices(res.devices); |
|||
}); |
|||
}); |
|||
} |
|||
|
|||
/** |
|||
* @export |
|||
* @param {function} stopBlueDevicesDiscovery 关闭蓝牙扫描 |
|||
*/ |
|||
export function stopBlueDevicesDiscovery() { |
|||
//监听寻找到新设备的事件
|
|||
console.log('停止蓝牙扫描'); |
|||
return uniAsyncPromise('stopBluetoothDevicesDiscovery').then((res) => { |
|||
console.log('停止搜寻蓝牙设备', res); |
|||
}); |
|||
} |
|||
|
|||
|
|||
|
|||
//三、连接蓝牙设备
|
|||
/** |
|||
* @export |
|||
* @param {function} createBLEConnection |
|||
* @param {number} deviceId 蓝牙设备id |
|||
*/ |
|||
export function createBLEConnection(deviceId, sucess, fail) { |
|||
//连接蓝牙设备
|
|||
console.log('连接蓝牙设备', deviceId); |
|||
uniAsyncPromise("createBLEConnection", { |
|||
deviceId |
|||
}) |
|||
.then(res => { |
|||
//连接成功可选择停止搜索蓝牙
|
|||
//stopBlueDevicesDiscovery();
|
|||
console.log('连接成功'); |
|||
sucess && sucess({ |
|||
res: res, |
|||
}); |
|||
}) |
|||
.catch(res => { |
|||
console.log('连接设备异常' + res); |
|||
fail && fail({ |
|||
res: res, |
|||
}); |
|||
}) |
|||
/*.finally(res=>{ |
|||
console.log('连接成功'); |
|||
sucess && sucess({ |
|||
res: res, |
|||
}); |
|||
});*/ |
|||
} |
|||
|
|||
export function closeBLEConnection(deviceId) { |
|||
console.log('断开蓝牙设备', deviceId); |
|||
uniAsyncPromise("closeBLEConnection", { |
|||
deviceId |
|||
}) |
|||
.then(res => { |
|||
console.log('BLEDisconnect complete', res); |
|||
}) |
|||
.catch(res => { |
|||
console.log('断开设备异常' + res); |
|||
}) |
|||
/*.finally(res=>{ |
|||
console.log('BLEDisconnect complete', res); |
|||
}); */ |
|||
} |
|||
|
|||
//四、连接成功后, 获取蓝牙设备的service服务
|
|||
// uniAsyncPromise("getBLEDeviceServices",{deviceId:""}).then(res=>{})
|
|||
export function getBLEDeviceServices(deviceId, success, fail) { |
|||
console.log('获取ServiceId', deviceId); |
|||
//加延迟避免取不到service
|
|||
setTimeout(()=>{ |
|||
uniAsyncPromise("getBLEDeviceServices", { |
|||
deviceId:deviceId |
|||
}) |
|||
.then(res => { |
|||
console.log('服务', res); |
|||
success && success({ |
|||
serviceId: res.services, |
|||
}); |
|||
}) |
|||
.catch((res) => { |
|||
//getBLEDeviceServices(deviceId, success, fail);
|
|||
console.log('获取ServiceId异常' + res); |
|||
fail && fail({ |
|||
res: res, |
|||
}); |
|||
}); |
|||
},1000) |
|||
} |
|||
|
|||
//五、获取的service服务可能有多个,递归获取特征值(最后要用的是能读,能写,能监听的那个值的uuid作为特征值id)
|
|||
/** |
|||
* |
|||
* |
|||
* @export |
|||
* @param {number} deviceId 蓝牙设备id |
|||
* @param {array} services uniAsyncPromise("getBLEDeviceServices",{deviceId:""}).then(res=>{})获取的res.services |
|||
* @param {function} success 成功取得有用特征值uuid的回调函数 |
|||
*/ |
|||
export function getDeviceCharacteristics(deviceId, services, success, fail) { |
|||
//services = services.slice(0);
|
|||
console.log('获取Characteristics', deviceId, services); |
|||
if (services.length) { |
|||
const serviceId = services.shift().uuid; |
|||
console.log('ServceID ', serviceId); |
|||
uniAsyncPromise('getBLEDeviceCharacteristics', { |
|||
deviceId, |
|||
serviceId, |
|||
}) |
|||
.then((res) => { |
|||
console.log('getBLEDeviceCharacteristics', deviceId, serviceId, res); |
|||
let finished = false; |
|||
let write = false; |
|||
let notify = false; |
|||
let indicate = false; |
|||
var readId; |
|||
var writeId; |
|||
//有斑马品牌的一款打印机中res.characteristics的所有uuid都是相同的,找所有的properties存在(notify || indicate) && write这种情况就说明这个uuid是可用的(不确保所有的打印机都能用这种方式取得uuid,在主要测试得凯盛诺打印机res.characteristic只有一个uuid,所以也能用这个方式)
|
|||
for (var i = 0; i < res.characteristics.length; i++) { |
|||
if (!notify) { |
|||
notify = res.characteristics[i].properties.notify; |
|||
if (notify) readId = res.characteristics[i].uuid; |
|||
} |
|||
if (!indicate) { |
|||
indicate = res.characteristics[i].properties.indicate; |
|||
if (indicate) readId = res.characteristics[i].uuid; |
|||
} |
|||
if (!write) { |
|||
write = res.characteristics[i].properties.write; |
|||
writeId = res.characteristics[i].uuid; |
|||
} |
|||
if ((notify || indicate) && write) { |
|||
/* 获取蓝牙特征值uuid */ |
|||
success && |
|||
success({ |
|||
serviceId, |
|||
writeId: writeId, |
|||
readId: readId, |
|||
}); |
|||
finished = true; |
|||
break; |
|||
} |
|||
} |
|||
|
|||
if (!finished) { |
|||
getDeviceCharacteristics(deviceId, services, success, fail); |
|||
} |
|||
}) |
|||
.catch((res) => { |
|||
getDeviceCharacteristics(deviceId, services, success, fail); |
|||
}); |
|||
} else { |
|||
fail && fail(); |
|||
} |
|||
} |
|||
|
|||
//六、启动notify 蓝牙监听功能 然后使用 uni.onBLECharacteristicValueChange用来监听蓝牙设备传递数据
|
|||
/** |
|||
* @export |
|||
* @param {object} options |
|||
* { |
|||
deviceId,//蓝牙设备id
|
|||
serviceId,//服务id
|
|||
characteristicId,//可用特征值uuid
|
|||
} |
|||
* @param {function} onChange 监听蓝牙设备传递数据回调函数 |
|||
*/ |
|||
export function onGetBLECharacteristicValueChange(options, onChange = function() {}) { |
|||
console.log('deviceId ', options.deviceId); |
|||
console.log('serviceId ', options.serviceId); |
|||
console.log('characteristicId ', options.characteristicId); |
|||
uniAsyncPromise('notifyBLECharacteristicValueChange', { |
|||
state: true, |
|||
...options, |
|||
}).then((res) => { |
|||
console.log('onBLECharacteristicValueChange '); |
|||
uni.onBLECharacteristicValueChange(onChange); |
|||
}); |
|||
} |
|||
|
|||
//七、发送数据(递归分包发送)
|
|||
/** |
|||
* @export |
|||
* @param {object} options |
|||
* { |
|||
deviceId, |
|||
serviceId, |
|||
characteristicId, |
|||
value [ArrayBuffer], |
|||
lasterSuccess, |
|||
onceLength |
|||
} |
|||
*/ |
|||
|
|||
export function sendDataToDevice(options) { |
|||
let byteLength = options.value.byteLength; |
|||
//这里默认一次20个字节发送
|
|||
const speed = options.onceLength; //20;
|
|||
console.log("send data 20"); |
|||
console.log(options); |
|||
if (byteLength > 0) { |
|||
uniAsyncPromise('writeBLECharacteristicValue', { |
|||
...options, |
|||
value: options.value.slice(0, byteLength > speed ? speed : byteLength), |
|||
}) |
|||
.then((res) => { |
|||
if (byteLength > speed) { |
|||
sendDataToDevice({ |
|||
...options, |
|||
value: options.value.slice(speed, byteLength), |
|||
}); |
|||
} else { |
|||
options.lasterSuccess && options.lasterSuccess(); |
|||
} |
|||
}) |
|||
.catch((res) => { |
|||
console.log(res); |
|||
}); |
|||
} |
|||
} |
|||
export function charToArrayBuffer(str) { |
|||
var out = new ArrayBuffer(str.length); |
|||
var uint8 = new Uint8Array(out); |
|||
var strs = str.split(''); |
|||
for (var i = 0; i < strs.length; i++) { |
|||
uint8[i] = strs[i].charCodeAt(); |
|||
} |
|||
return uint8; |
|||
} |
|||
export function charToArray(str) { |
|||
var arr = []; |
|||
var strs = str.split(''); |
|||
for (var i = 0; i < strs.length; i++) { |
|||
arr[i] = strs[i].charCodeAt(); |
|||
} |
|||
return arr; |
|||
} |
|||
//打印二维码
|
|||
/** |
|||
* @export |
|||
* @param {object} options |
|||
* { |
|||
deviceId, |
|||
serviceId, |
|||
characteristicId, |
|||
value,//ArrayBuffer:二维码的数据
|
|||
} |
|||
*/ |
|||
export function printQR(options) { |
|||
//打印二维码的十进制指令data:
|
|||
let data = [29, 107, 97, 7, 4, options.value.byteLength, 0]; |
|||
sendDataToDevice({ |
|||
...options, |
|||
value: new Uint8Array(data).buffer, |
|||
lasterSuccess: () => { |
|||
//指令发送成功后,发送二维码的数据
|
|||
sendDataToDevice(options); |
|||
}, |
|||
}); |
|||
} |
|||
|
|||
function grayPixle(pix) { |
|||
return pix[0] * 0.299 + pix[1] * 0.587 + pix[2] * 0.114; |
|||
} |
|||
|
|||
export function overwriteImageData(data) { |
|||
let sendWidth = data.width, |
|||
sendHeight = data.height; |
|||
const threshold = data.threshold || 180; |
|||
let sendImageData = new ArrayBuffer((sendWidth * sendHeight) / 8); |
|||
sendImageData = new Uint8Array(sendImageData); |
|||
let pix = data.imageData; |
|||
const part = []; |
|||
let index = 0; |
|||
for (let i = 0; i < pix.length; i += 32) { |
|||
//横向每8个像素点组成一个字节(8位二进制数)。
|
|||
for (let k = 0; k < 8; k++) { |
|||
const grayPixle1 = grayPixle(pix.slice(i + k * 4, i + k * 4 + (4 - 1))); |
|||
//阈值调整
|
|||
if (grayPixle1 > threshold) { |
|||
//灰度值大于threshold位 白色 为第k位0不打印
|
|||
part[k] = 0; |
|||
} else { |
|||
part[k] = 1; |
|||
} |
|||
} |
|||
let temp = 0; |
|||
for (let a = 0; a < part.length; a++) { |
|||
temp += part[a] * Math.pow(2, part.length - 1 - a); |
|||
} |
|||
//开始不明白以下算法什么意思,了解了字节才知道,一个字节是8位的二进制数,part这个数组存的0和1就是二进制的0和1,传输到打印的位图数据的一个字节是0-255之间的十进制数,以下是用权相加法转十进制数,理解了这个就用上面的for循环替代了
|
|||
// const temp =
|
|||
// part[0] * 128 +
|
|||
// part[1] * 64 +
|
|||
// part[2] * 32 +
|
|||
// part[3] * 16 +
|
|||
// part[4] * 8 +
|
|||
// part[5] * 4 +
|
|||
// part[6] * 2 +
|
|||
// part[7] * 1;
|
|||
sendImageData[index++] = temp; |
|||
} |
|||
return { |
|||
array: Array.from(sendImageData), |
|||
width: sendWidth / 8, |
|||
height: sendHeight, |
|||
}; |
|||
} |
|||
/** |
|||
* printImage |
|||
* @param {object} opt |
|||
* { |
|||
deviceId,//蓝牙设备id
|
|||
serviceId,//服务id
|
|||
characteristicId,//可用特征值uuid
|
|||
lasterSuccess , //最后完成的回调
|
|||
} |
|||
*/ |
|||
export function printImage(opt = {}, imageInfo = {}) { |
|||
let arr = imageInfo.array, |
|||
width = imageInfo.width; |
|||
const writeArray = []; |
|||
const xl = width % 256; |
|||
const xh = width / 256; |
|||
//分行发送图片数据,用的十进制指令
|
|||
const command = [29, 118, 48, 0, xl, xh, 1, 0]; //1D 76 30 00 w h
|
|||
const enter = [13, 10]; |
|||
for (let i = 0; i < arr.length / width; i++) { |
|||
const subArr = arr.slice(i * width, i * width + width); |
|||
const tempArr = command.concat(subArr); |
|||
writeArray.push(new Uint8Array(tempArr)); |
|||
} |
|||
writeArray.push(new Uint8Array(enter)); |
|||
//console.log(writeArray);
|
|||
const print = (options, writeArray) => { |
|||
if (writeArray.length) { |
|||
console.log("send"); |
|||
sendDataToDevice({ |
|||
...options, |
|||
value: writeArray.shift().buffer, |
|||
lasterSuccess: () => { |
|||
if (writeArray.length) { |
|||
print(options, writeArray); |
|||
} else { |
|||
options.lasterSuccess && options.lasterSuccess(); |
|||
} |
|||
}, |
|||
}); |
|||
} |
|||
}; |
|||
console.log("start print"); |
|||
print(opt, writeArray); |
|||
} |
|||
|
|||
/* 16hex insert 0 */ |
|||
function Hex2Str(num) { |
|||
if (num.toString(16).length < 2) return "0" + num.toString(16); |
|||
else |
|||
return num.toString(16); |
|||
} |
|||
/*****CPCL指令接口****/ |
|||
|
|||
/** |
|||
* 配置项如下 |
|||
* |
|||
* width: 标签纸的宽度,单位像素點 |
|||
* height: 标签纸的高度,单位像素點 |
|||
* 8像素=1mm |
|||
* printNum: 打印张数,默认为1 |
|||
* rotation:页面整体旋转 1-90度 2-180度 3-270度 其他-不旋转 |
|||
*/ |
|||
export function CreatCPCLPage(width, height, printNum, rotation = 0, offset = 0) { |
|||
var strCmd = '! ' + offset + ' 200 200 ' + height + ' ' + printNum + '\n'; |
|||
strCmd += "PAGE-WIDTH " + width + '\n'; |
|||
if (rotation == 1) |
|||
strCmd += "ZPROTATE90\n"; |
|||
else if (rotation == 2) |
|||
strCmd += "ZPROTATE180\n"; |
|||
else if (rotation == 3) |
|||
strCmd += "ZPROTATE270\n"; |
|||
else |
|||
strCmd += "ZPROTATE0\n"; |
|||
return strCmd; |
|||
} |
|||
|
|||
/** |
|||
* 打印文字 |
|||
* x: 文字方块左上角X座标,单位dot |
|||
* y: 文字方块左上角Y座标,单位dot |
|||
* fontName,fontSize: 字体,取值: 參考文檔 |
|||
* rotation: 旋转 1-90度 2-180度 3-270度 其他-不旋转 |
|||
* content: 文字内容 |
|||
*/ |
|||
export function addCPCLText(x, y, fontName, fontSize, rotation, content) { |
|||
//console.log(fontName,fontSize,rotation, content);
|
|||
var strCmd = ''; |
|||
if (rotation == 1) { |
|||
strCmd += 'T90 '; |
|||
} |
|||
if (rotation == 2) { |
|||
strCmd += 'T180 '; |
|||
} |
|||
if (rotation == 3) { |
|||
strCmd += 'T270 '; |
|||
} else { |
|||
strCmd += 'T '; |
|||
} |
|||
strCmd += fontName + ' ' + fontSize + ' ' + x + ' ' + y + ' ' + content + '\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 打印一维码 |
|||
* |
|||
* x: 文字方块左上角X座标,单位dot |
|||
* y: 文字方块左上角Y座标,单位dot |
|||
* codeType: 条码类型,取值为128、UPCA、UPCA2、UPCA5、UPCE、UPCE2、UPC5、EAN13、EAN13+2、EAN13+5、 |
|||
* EAN8、EAN8+2、EAN8+5、39、39C、F39、F39C、93、CODABAR、CODABAR16、ITF、I2OF5 |
|||
* h: 条码高度,单位dot |
|||
* rotation: 顺时针旋转角度,取值如下: |
|||
* - 0 不旋转 |
|||
* - 1 顺时针旋转90度 |
|||
* |
|||
* narrow: 窄条码比例因子(dot) 取值: 參考文檔 |
|||
* wide: 宽条码比例因子(dot) 取值: 參考文檔 |
|||
* content: 文字内容 |
|||
* |
|||
*/ |
|||
export function addCPCLBarCode(x, y, codeType, h, rotation, narrow, wide, content) { |
|||
var strCmd = ''; |
|||
if (rotation == 0) |
|||
strCmd += 'B '; |
|||
else |
|||
strCmd += 'VB '; |
|||
strCmd += codeType + ' ' + narrow + ' ' + wide + ' ' + h + ' ' + x + ' ' + y + ' ' + content + '\n' |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 打印二维码 |
|||
* |
|||
* x: 文字方块左上角X座标,单位dot |
|||
* y: 文字方块左上角Y座标,单位dot |
|||
* level: 错误纠正能力等级,取值为L(7%)、M(15%)、Q(25%)、H(30%) |
|||
* ver: 1-10 版本,根据内容调整以获取合适容量 |
|||
* scale: 1-10 放大倍数 |
|||
* content: 文字内容 |
|||
* |
|||
*/ |
|||
export function addCPCLQRCode(x, y, level, ver, scale, content) { |
|||
var strCmd = 'B QR ' + x + ' ' + y + ' M ' + ver + ' U ' + scale + '\n' + level + 'A,' + content + '\n'; |
|||
strCmd += 'ENDQR\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 放大指令 |
|||
* scaleX: 横向放大倍数 1,2,3等整数 |
|||
* scaleY: 纵向放大倍数 1,2,3等整数 |
|||
*/ |
|||
export function addCPCLSETMAG(scaleX, scaleY) { |
|||
var strCmd = 'SETMAG ' + scaleX + ' ' + scaleY + '\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 对齐指令 0-左对齐 1-右对齐 2-居中 |
|||
*/ |
|||
export function addCPCLLocation(set) { |
|||
var strCmd = ''; |
|||
if (set == 1) { |
|||
strCmd += 'RIGHT\n'; |
|||
} else if (set == 2) { |
|||
strCmd += 'CENTER\n'; |
|||
} else { |
|||
strCmd += 'LEFT\n'; |
|||
} |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 反白线 x0,y0,x1,y1,width |
|||
*/ |
|||
export function addCPCLInverseLine(x0, y0, x1, y1, width) { |
|||
var strCmd = 'IL ' + x0 + ' ' + y0 + ' ' + x1 + ' ' + y1 + ' ' + width + '\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 画线 x0,y0,x1,y1,width |
|||
*/ |
|||
export function addCPCLLine(x0, y0, x1, y1, width) { |
|||
var strCmd = 'L ' + x0 + ' ' + y0 + ' ' + x1 + ' ' + y1 + ' ' + width + '\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 画框 x0,y0,x1,y1,width |
|||
*/ |
|||
export function addCPCLBox(x0, y0, x1, y1, width) { |
|||
var strCmd = 'BOX ' + x0 + ' ' + y0 + ' ' + x1 + ' ' + y1 + ' ' + width + '\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 字体加粗 |
|||
*/ |
|||
export function addCPCLSETBOLD(bold) { |
|||
var strCmd = 'SETBOLD ' + bold + '\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 字体下划线 |
|||
*/ |
|||
export function addCPCLUNDERLINE(c) { |
|||
var strCmd = 'UNDERLINE '; |
|||
if (c) strCmd += 'ON\n'; |
|||
else if (c) strCmd += 'OFF\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 水印打印灰度等级 0-255 |
|||
*/ |
|||
export function addCPCLBACKGROUND(level) { |
|||
var strCmd = 'BACKGROUND '; |
|||
if (level > 255 || level < 0) level = 255; |
|||
strCmd += level + '\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 打印水印文字 |
|||
* x: 文字方块左上角X座标,单位dot |
|||
* y: 文字方块左上角Y座标,单位dot |
|||
* fontName,fontSize: 字体,取值: 參考文檔 |
|||
* rotation: 旋转 1-90度 2-180度 3-270度 其他-不旋转 |
|||
* content: 文字内容 |
|||
*/ |
|||
export function addCPCLBKVText(x, y, fontName, fontSize, rotation, content) { |
|||
//console.log(fontName,fontSize,rotation, content);
|
|||
var strCmd = ''; |
|||
if (rotation == 1) { |
|||
strCmd += 'BKT90 '; |
|||
} |
|||
if (rotation == 2) { |
|||
strCmd += 'BKT180 '; |
|||
} |
|||
if (rotation == 3) { |
|||
strCmd += 'BKT270 '; |
|||
} else { |
|||
strCmd += 'BKT '; |
|||
} |
|||
strCmd += fontName + ' ' + fontSize + ' ' + x + ' ' + y + ' ' + content + '\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
|
|||
/** |
|||
* 标签缝隙定位指令 |
|||
*/ |
|||
export function addCPCLGAP() { |
|||
var strCmd = 'GAP-SENSE\nFORM\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 标签右黑标检测指令 |
|||
*/ |
|||
export function addCPCLSENSE() { |
|||
var strCmd = 'BAR-SENSE\nFORM\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 标签左黑标检测指令 |
|||
*/ |
|||
export function addCPCLSENSELEFT() { |
|||
var strCmd = 'BAR-SENSE LEFT\nFORM\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 打印指令 |
|||
*/ |
|||
export function addCPCLPrint() { |
|||
var strCmd = 'PRINT\n'; |
|||
return strCmd; |
|||
}; |
|||
|
|||
/** |
|||
* 图片打印指令 |
|||
* x: 文字方块左上角X座标,单位dot |
|||
* y: 文字方块左上角Y座标,单位dot |
|||
* data{ |
|||
threshold,//0/1提取的灰度级
|
|||
width,//图像宽度
|
|||
height,//图像高度
|
|||
imageData , //图像数据
|
|||
} |
|||
*/ |
|||
export function addCPCLImageCmd(x, y, data) { |
|||
var strImgCmd = ''; |
|||
const threshold = data.threshold || 180; |
|||
let myBitmapWidth = data.width, |
|||
myBitmapHeight = data.height; |
|||
let len = parseInt((myBitmapWidth + 7) / 8); //一行的数据长度
|
|||
//console.log('len=',len);
|
|||
//console.log('myBitmapWidth=',myBitmapWidth);
|
|||
//console.log('myBitmapHeight=',myBitmapHeight);
|
|||
let ndata = 0; |
|||
let i = 0; |
|||
let j = 0; |
|||
let sendImageData = new ArrayBuffer(len * myBitmapHeight); |
|||
sendImageData = new Uint8Array(sendImageData); |
|||
let pix = data.imageData; |
|||
console.log('pix=', pix); |
|||
|
|||
for (i = 0; i < myBitmapHeight; i++) { |
|||
for (j = 0; j < len; j++) { |
|||
sendImageData[ndata + j] = 0; |
|||
} |
|||
for (j = 0; j < myBitmapWidth; j++) { |
|||
const grayPixle1 = grayPixle(pix.slice((i * myBitmapWidth + j) * 4, (i * myBitmapWidth + j) * 4 + 3)); |
|||
if (grayPixle1 < threshold) |
|||
sendImageData[ndata + parseInt(j / 8)] |= (0x80 >> (j % 8)); |
|||
|
|||
} |
|||
ndata += len; |
|||
} |
|||
//console.log('sendImageData=',sendImageData);
|
|||
//CPCL指令图片数据
|
|||
strImgCmd += 'EG ' + len + ' ' + myBitmapHeight + ' ' + x + ' ' + y + ' '; |
|||
for (i = 0; i < sendImageData.length; i++) { |
|||
strImgCmd += Hex2Str(sendImageData[i]); |
|||
} |
|||
strImgCmd += '\n'; |
|||
//console.log(strImgCmd);
|
|||
return strImgCmd; |
|||
} |
|||
/** |
|||
* toast显示捕获的蓝牙异常 |
|||
*/ |
|||
export function catchToast(err) { |
|||
const errMsg = { |
|||
10000: '未初始化蓝牙模块', |
|||
10001: '蓝牙未打开', |
|||
10002: '没有找到指定设备', |
|||
10003: '连接失败', |
|||
10004: '没有找到指定服务', |
|||
10005: '没有找到指定特征值', |
|||
10006: '当前连接已断开', |
|||
10007: '当前特征值不支持此操作', |
|||
10008: '系统上报异常', |
|||
10009: '系统版本低于 4.3 不支持BLE' |
|||
}; |
|||
let coode = err.errCode ? err.errCode.toString() : ''; |
|||
let msg = errMsg[coode]; |
|||
plus.nativeUI.toast(msg || coode, { |
|||
align: 'center', |
|||
verticalAlign: 'center' |
|||
}); |
|||
} |
@ -0,0 +1,23 @@ |
|||
var barcode = require('./barcode'); |
|||
var qrcode = require('./qrcode'); |
|||
|
|||
function convert_length(length) { |
|||
return Math.round(wx.getSystemInfoSync().windowWidth * length / 750); |
|||
} |
|||
|
|||
function barc(id, code, width, height) { |
|||
barcode.code128(wx.createCanvasContext(id), code, convert_length(width), convert_length(height)) |
|||
} |
|||
|
|||
function qrc(id, code, width, height) { |
|||
qrcode.api.draw(code, { |
|||
ctx: wx.createCanvasContext(id), |
|||
width: convert_length(width), |
|||
height: convert_length(height) |
|||
}) |
|||
} |
|||
|
|||
module.exports = { |
|||
barcode: barc, |
|||
qrcode: qrc |
|||
} |
File diff suppressed because it is too large
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -0,0 +1,778 @@ |
|||
var QR = (function () { |
|||
|
|||
// alignment pattern
|
|||
var adelta = [ |
|||
0, 11, 15, 19, 23, 27, 31, // force 1 pat
|
|||
16, 18, 20, 22, 24, 26, 28, 20, 22, 24, 24, 26, 28, 28, 22, 24, 24, |
|||
26, 26, 28, 28, 24, 24, 26, 26, 26, 28, 28, 24, 26, 26, 26, 28, 28 |
|||
]; |
|||
|
|||
// version block
|
|||
var vpat = [ |
|||
0xc94, 0x5bc, 0xa99, 0x4d3, 0xbf6, 0x762, 0x847, 0x60d, |
|||
0x928, 0xb78, 0x45d, 0xa17, 0x532, 0x9a6, 0x683, 0x8c9, |
|||
0x7ec, 0xec4, 0x1e1, 0xfab, 0x08e, 0xc1a, 0x33f, 0xd75, |
|||
0x250, 0x9d5, 0x6f0, 0x8ba, 0x79f, 0xb0b, 0x42e, 0xa64, |
|||
0x541, 0xc69 |
|||
]; |
|||
|
|||
// final format bits with mask: level << 3 | mask
|
|||
var fmtword = [ |
|||
0x77c4, 0x72f3, 0x7daa, 0x789d, 0x662f, 0x6318, 0x6c41, 0x6976, //L
|
|||
0x5412, 0x5125, 0x5e7c, 0x5b4b, 0x45f9, 0x40ce, 0x4f97, 0x4aa0, //M
|
|||
0x355f, 0x3068, 0x3f31, 0x3a06, 0x24b4, 0x2183, 0x2eda, 0x2bed, //Q
|
|||
0x1689, 0x13be, 0x1ce7, 0x19d0, 0x0762, 0x0255, 0x0d0c, 0x083b //H
|
|||
]; |
|||
|
|||
// 4 per version: number of blocks 1,2; data width; ecc width
|
|||
var eccblocks = [ |
|||
1, 0, 19, 7, 1, 0, 16, 10, 1, 0, 13, 13, 1, 0, 9, 17, |
|||
1, 0, 34, 10, 1, 0, 28, 16, 1, 0, 22, 22, 1, 0, 16, 28, |
|||
1, 0, 55, 15, 1, 0, 44, 26, 2, 0, 17, 18, 2, 0, 13, 22, |
|||
1, 0, 80, 20, 2, 0, 32, 18, 2, 0, 24, 26, 4, 0, 9, 16, |
|||
1, 0, 108, 26, 2, 0, 43, 24, 2, 2, 15, 18, 2, 2, 11, 22, |
|||
2, 0, 68, 18, 4, 0, 27, 16, 4, 0, 19, 24, 4, 0, 15, 28, |
|||
2, 0, 78, 20, 4, 0, 31, 18, 2, 4, 14, 18, 4, 1, 13, 26, |
|||
2, 0, 97, 24, 2, 2, 38, 22, 4, 2, 18, 22, 4, 2, 14, 26, |
|||
2, 0, 116, 30, 3, 2, 36, 22, 4, 4, 16, 20, 4, 4, 12, 24, |
|||
2, 2, 68, 18, 4, 1, 43, 26, 6, 2, 19, 24, 6, 2, 15, 28, |
|||
4, 0, 81, 20, 1, 4, 50, 30, 4, 4, 22, 28, 3, 8, 12, 24, |
|||
2, 2, 92, 24, 6, 2, 36, 22, 4, 6, 20, 26, 7, 4, 14, 28, |
|||
4, 0, 107, 26, 8, 1, 37, 22, 8, 4, 20, 24, 12, 4, 11, 22, |
|||
3, 1, 115, 30, 4, 5, 40, 24, 11, 5, 16, 20, 11, 5, 12, 24, |
|||
5, 1, 87, 22, 5, 5, 41, 24, 5, 7, 24, 30, 11, 7, 12, 24, |
|||
5, 1, 98, 24, 7, 3, 45, 28, 15, 2, 19, 24, 3, 13, 15, 30, |
|||
1, 5, 107, 28, 10, 1, 46, 28, 1, 15, 22, 28, 2, 17, 14, 28, |
|||
5, 1, 120, 30, 9, 4, 43, 26, 17, 1, 22, 28, 2, 19, 14, 28, |
|||
3, 4, 113, 28, 3, 11, 44, 26, 17, 4, 21, 26, 9, 16, 13, 26, |
|||
3, 5, 107, 28, 3, 13, 41, 26, 15, 5, 24, 30, 15, 10, 15, 28, |
|||
4, 4, 116, 28, 17, 0, 42, 26, 17, 6, 22, 28, 19, 6, 16, 30, |
|||
2, 7, 111, 28, 17, 0, 46, 28, 7, 16, 24, 30, 34, 0, 13, 24, |
|||
4, 5, 121, 30, 4, 14, 47, 28, 11, 14, 24, 30, 16, 14, 15, 30, |
|||
6, 4, 117, 30, 6, 14, 45, 28, 11, 16, 24, 30, 30, 2, 16, 30, |
|||
8, 4, 106, 26, 8, 13, 47, 28, 7, 22, 24, 30, 22, 13, 15, 30, |
|||
10, 2, 114, 28, 19, 4, 46, 28, 28, 6, 22, 28, 33, 4, 16, 30, |
|||
8, 4, 122, 30, 22, 3, 45, 28, 8, 26, 23, 30, 12, 28, 15, 30, |
|||
3, 10, 117, 30, 3, 23, 45, 28, 4, 31, 24, 30, 11, 31, 15, 30, |
|||
7, 7, 116, 30, 21, 7, 45, 28, 1, 37, 23, 30, 19, 26, 15, 30, |
|||
5, 10, 115, 30, 19, 10, 47, 28, 15, 25, 24, 30, 23, 25, 15, 30, |
|||
13, 3, 115, 30, 2, 29, 46, 28, 42, 1, 24, 30, 23, 28, 15, 30, |
|||
17, 0, 115, 30, 10, 23, 46, 28, 10, 35, 24, 30, 19, 35, 15, 30, |
|||
17, 1, 115, 30, 14, 21, 46, 28, 29, 19, 24, 30, 11, 46, 15, 30, |
|||
13, 6, 115, 30, 14, 23, 46, 28, 44, 7, 24, 30, 59, 1, 16, 30, |
|||
12, 7, 121, 30, 12, 26, 47, 28, 39, 14, 24, 30, 22, 41, 15, 30, |
|||
6, 14, 121, 30, 6, 34, 47, 28, 46, 10, 24, 30, 2, 64, 15, 30, |
|||
17, 4, 122, 30, 29, 14, 46, 28, 49, 10, 24, 30, 24, 46, 15, 30, |
|||
4, 18, 122, 30, 13, 32, 46, 28, 48, 14, 24, 30, 42, 32, 15, 30, |
|||
20, 4, 117, 30, 40, 7, 47, 28, 43, 22, 24, 30, 10, 67, 15, 30, |
|||
19, 6, 118, 30, 18, 31, 47, 28, 34, 34, 24, 30, 20, 61, 15, 30 |
|||
]; |
|||
|
|||
// Galois field log table
|
|||
var glog = [ |
|||
0xff, 0x00, 0x01, 0x19, 0x02, 0x32, 0x1a, 0xc6, 0x03, 0xdf, 0x33, 0xee, 0x1b, 0x68, 0xc7, 0x4b, |
|||
0x04, 0x64, 0xe0, 0x0e, 0x34, 0x8d, 0xef, 0x81, 0x1c, 0xc1, 0x69, 0xf8, 0xc8, 0x08, 0x4c, 0x71, |
|||
0x05, 0x8a, 0x65, 0x2f, 0xe1, 0x24, 0x0f, 0x21, 0x35, 0x93, 0x8e, 0xda, 0xf0, 0x12, 0x82, 0x45, |
|||
0x1d, 0xb5, 0xc2, 0x7d, 0x6a, 0x27, 0xf9, 0xb9, 0xc9, 0x9a, 0x09, 0x78, 0x4d, 0xe4, 0x72, 0xa6, |
|||
0x06, 0xbf, 0x8b, 0x62, 0x66, 0xdd, 0x30, 0xfd, 0xe2, 0x98, 0x25, 0xb3, 0x10, 0x91, 0x22, 0x88, |
|||
0x36, 0xd0, 0x94, 0xce, 0x8f, 0x96, 0xdb, 0xbd, 0xf1, 0xd2, 0x13, 0x5c, 0x83, 0x38, 0x46, 0x40, |
|||
0x1e, 0x42, 0xb6, 0xa3, 0xc3, 0x48, 0x7e, 0x6e, 0x6b, 0x3a, 0x28, 0x54, 0xfa, 0x85, 0xba, 0x3d, |
|||
0xca, 0x5e, 0x9b, 0x9f, 0x0a, 0x15, 0x79, 0x2b, 0x4e, 0xd4, 0xe5, 0xac, 0x73, 0xf3, 0xa7, 0x57, |
|||
0x07, 0x70, 0xc0, 0xf7, 0x8c, 0x80, 0x63, 0x0d, 0x67, 0x4a, 0xde, 0xed, 0x31, 0xc5, 0xfe, 0x18, |
|||
0xe3, 0xa5, 0x99, 0x77, 0x26, 0xb8, 0xb4, 0x7c, 0x11, 0x44, 0x92, 0xd9, 0x23, 0x20, 0x89, 0x2e, |
|||
0x37, 0x3f, 0xd1, 0x5b, 0x95, 0xbc, 0xcf, 0xcd, 0x90, 0x87, 0x97, 0xb2, 0xdc, 0xfc, 0xbe, 0x61, |
|||
0xf2, 0x56, 0xd3, 0xab, 0x14, 0x2a, 0x5d, 0x9e, 0x84, 0x3c, 0x39, 0x53, 0x47, 0x6d, 0x41, 0xa2, |
|||
0x1f, 0x2d, 0x43, 0xd8, 0xb7, 0x7b, 0xa4, 0x76, 0xc4, 0x17, 0x49, 0xec, 0x7f, 0x0c, 0x6f, 0xf6, |
|||
0x6c, 0xa1, 0x3b, 0x52, 0x29, 0x9d, 0x55, 0xaa, 0xfb, 0x60, 0x86, 0xb1, 0xbb, 0xcc, 0x3e, 0x5a, |
|||
0xcb, 0x59, 0x5f, 0xb0, 0x9c, 0xa9, 0xa0, 0x51, 0x0b, 0xf5, 0x16, 0xeb, 0x7a, 0x75, 0x2c, 0xd7, |
|||
0x4f, 0xae, 0xd5, 0xe9, 0xe6, 0xe7, 0xad, 0xe8, 0x74, 0xd6, 0xf4, 0xea, 0xa8, 0x50, 0x58, 0xaf |
|||
]; |
|||
|
|||
// Galios field exponent table
|
|||
var gexp = [ |
|||
0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1d, 0x3a, 0x74, 0xe8, 0xcd, 0x87, 0x13, 0x26, |
|||
0x4c, 0x98, 0x2d, 0x5a, 0xb4, 0x75, 0xea, 0xc9, 0x8f, 0x03, 0x06, 0x0c, 0x18, 0x30, 0x60, 0xc0, |
|||
0x9d, 0x27, 0x4e, 0x9c, 0x25, 0x4a, 0x94, 0x35, 0x6a, 0xd4, 0xb5, 0x77, 0xee, 0xc1, 0x9f, 0x23, |
|||
0x46, 0x8c, 0x05, 0x0a, 0x14, 0x28, 0x50, 0xa0, 0x5d, 0xba, 0x69, 0xd2, 0xb9, 0x6f, 0xde, 0xa1, |
|||
0x5f, 0xbe, 0x61, 0xc2, 0x99, 0x2f, 0x5e, 0xbc, 0x65, 0xca, 0x89, 0x0f, 0x1e, 0x3c, 0x78, 0xf0, |
|||
0xfd, 0xe7, 0xd3, 0xbb, 0x6b, 0xd6, 0xb1, 0x7f, 0xfe, 0xe1, 0xdf, 0xa3, 0x5b, 0xb6, 0x71, 0xe2, |
|||
0xd9, 0xaf, 0x43, 0x86, 0x11, 0x22, 0x44, 0x88, 0x0d, 0x1a, 0x34, 0x68, 0xd0, 0xbd, 0x67, 0xce, |
|||
0x81, 0x1f, 0x3e, 0x7c, 0xf8, 0xed, 0xc7, 0x93, 0x3b, 0x76, 0xec, 0xc5, 0x97, 0x33, 0x66, 0xcc, |
|||
0x85, 0x17, 0x2e, 0x5c, 0xb8, 0x6d, 0xda, 0xa9, 0x4f, 0x9e, 0x21, 0x42, 0x84, 0x15, 0x2a, 0x54, |
|||
0xa8, 0x4d, 0x9a, 0x29, 0x52, 0xa4, 0x55, 0xaa, 0x49, 0x92, 0x39, 0x72, 0xe4, 0xd5, 0xb7, 0x73, |
|||
0xe6, 0xd1, 0xbf, 0x63, 0xc6, 0x91, 0x3f, 0x7e, 0xfc, 0xe5, 0xd7, 0xb3, 0x7b, 0xf6, 0xf1, 0xff, |
|||
0xe3, 0xdb, 0xab, 0x4b, 0x96, 0x31, 0x62, 0xc4, 0x95, 0x37, 0x6e, 0xdc, 0xa5, 0x57, 0xae, 0x41, |
|||
0x82, 0x19, 0x32, 0x64, 0xc8, 0x8d, 0x07, 0x0e, 0x1c, 0x38, 0x70, 0xe0, 0xdd, 0xa7, 0x53, 0xa6, |
|||
0x51, 0xa2, 0x59, 0xb2, 0x79, 0xf2, 0xf9, 0xef, 0xc3, 0x9b, 0x2b, 0x56, 0xac, 0x45, 0x8a, 0x09, |
|||
0x12, 0x24, 0x48, 0x90, 0x3d, 0x7a, 0xf4, 0xf5, 0xf7, 0xf3, 0xfb, 0xeb, 0xcb, 0x8b, 0x0b, 0x16, |
|||
0x2c, 0x58, 0xb0, 0x7d, 0xfa, 0xe9, 0xcf, 0x83, 0x1b, 0x36, 0x6c, 0xd8, 0xad, 0x47, 0x8e, 0x00 |
|||
]; |
|||
|
|||
// Working buffers:
|
|||
// data input and ecc append, image working buffer, fixed part of image, run lengths for badness
|
|||
var strinbuf=[], eccbuf=[], qrframe=[], framask=[], rlens=[]; |
|||
// Control values - width is based on version, last 4 are from table.
|
|||
var version, width, neccblk1, neccblk2, datablkw, eccblkwid; |
|||
var ecclevel = 2; |
|||
// set bit to indicate cell in qrframe is immutable. symmetric around diagonal
|
|||
function setmask(x, y) |
|||
{ |
|||
var bt; |
|||
if (x > y) { |
|||
bt = x; |
|||
x = y; |
|||
y = bt; |
|||
} |
|||
// y*y = 1+3+5...
|
|||
bt = y; |
|||
bt *= y; |
|||
bt += y; |
|||
bt >>= 1; |
|||
bt += x; |
|||
framask[bt] = 1; |
|||
} |
|||
|
|||
// enter alignment pattern - black to qrframe, white to mask (later black frame merged to mask)
|
|||
function putalign(x, y) |
|||
{ |
|||
var j; |
|||
|
|||
qrframe[x + width * y] = 1; |
|||
for (j = -2; j < 2; j++) { |
|||
qrframe[(x + j) + width * (y - 2)] = 1; |
|||
qrframe[(x - 2) + width * (y + j + 1)] = 1; |
|||
qrframe[(x + 2) + width * (y + j)] = 1; |
|||
qrframe[(x + j + 1) + width * (y + 2)] = 1; |
|||
} |
|||
for (j = 0; j < 2; j++) { |
|||
setmask(x - 1, y + j); |
|||
setmask(x + 1, y - j); |
|||
setmask(x - j, y - 1); |
|||
setmask(x + j, y + 1); |
|||
} |
|||
} |
|||
|
|||
//========================================================================
|
|||
// Reed Solomon error correction
|
|||
// exponentiation mod N
|
|||
function modnn(x) |
|||
{ |
|||
while (x >= 255) { |
|||
x -= 255; |
|||
x = (x >> 8) + (x & 255); |
|||
} |
|||
return x; |
|||
} |
|||
|
|||
var genpoly = []; |
|||
|
|||
// Calculate and append ECC data to data block. Block is in strinbuf, indexes to buffers given.
|
|||
function appendrs(data, dlen, ecbuf, eclen) |
|||
{ |
|||
var i, j, fb; |
|||
|
|||
for (i = 0; i < eclen; i++) |
|||
strinbuf[ecbuf + i] = 0; |
|||
for (i = 0; i < dlen; i++) { |
|||
fb = glog[strinbuf[data + i] ^ strinbuf[ecbuf]]; |
|||
if (fb != 255) /* fb term is non-zero */ |
|||
for (j = 1; j < eclen; j++) |
|||
strinbuf[ecbuf + j - 1] = strinbuf[ecbuf + j] ^ gexp[modnn(fb + genpoly[eclen - j])]; |
|||
else |
|||
for( j = ecbuf ; j < ecbuf + eclen; j++ ) |
|||
strinbuf[j] = strinbuf[j + 1]; |
|||
strinbuf[ ecbuf + eclen - 1] = fb == 255 ? 0 : gexp[modnn(fb + genpoly[0])]; |
|||
} |
|||
} |
|||
|
|||
//========================================================================
|
|||
// Frame data insert following the path rules
|
|||
|
|||
// check mask - since symmetrical use half.
|
|||
function ismasked(x, y) |
|||
{ |
|||
var bt; |
|||
if (x > y) { |
|||
bt = x; |
|||
x = y; |
|||
y = bt; |
|||
} |
|||
bt = y; |
|||
bt += y * y; |
|||
bt >>= 1; |
|||
bt += x; |
|||
return framask[bt]; |
|||
} |
|||
|
|||
//========================================================================
|
|||
// Apply the selected mask out of the 8.
|
|||
function applymask(m) |
|||
{ |
|||
var x, y, r3x, r3y; |
|||
|
|||
switch (m) { |
|||
case 0: |
|||
for (y = 0; y < width; y++) |
|||
for (x = 0; x < width; x++) |
|||
if (!((x + y) & 1) && !ismasked(x, y)) |
|||
qrframe[x + y * width] ^= 1; |
|||
break; |
|||
case 1: |
|||
for (y = 0; y < width; y++) |
|||
for (x = 0; x < width; x++) |
|||
if (!(y & 1) && !ismasked(x, y)) |
|||
qrframe[x + y * width] ^= 1; |
|||
break; |
|||
case 2: |
|||
for (y = 0; y < width; y++) |
|||
for (r3x = 0, x = 0; x < width; x++, r3x++) { |
|||
if (r3x == 3) |
|||
r3x = 0; |
|||
if (!r3x && !ismasked(x, y)) |
|||
qrframe[x + y * width] ^= 1; |
|||
} |
|||
break; |
|||
case 3: |
|||
for (r3y = 0, y = 0; y < width; y++, r3y++) { |
|||
if (r3y == 3) |
|||
r3y = 0; |
|||
for (r3x = r3y, x = 0; x < width; x++, r3x++) { |
|||
if (r3x == 3) |
|||
r3x = 0; |
|||
if (!r3x && !ismasked(x, y)) |
|||
qrframe[x + y * width] ^= 1; |
|||
} |
|||
} |
|||
break; |
|||
case 4: |
|||
for (y = 0; y < width; y++) |
|||
for (r3x = 0, r3y = ((y >> 1) & 1), x = 0; x < width; x++, r3x++) { |
|||
if (r3x == 3) { |
|||
r3x = 0; |
|||
r3y = !r3y; |
|||
} |
|||
if (!r3y && !ismasked(x, y)) |
|||
qrframe[x + y * width] ^= 1; |
|||
} |
|||
break; |
|||
case 5: |
|||
for (r3y = 0, y = 0; y < width; y++, r3y++) { |
|||
if (r3y == 3) |
|||
r3y = 0; |
|||
for (r3x = 0, x = 0; x < width; x++, r3x++) { |
|||
if (r3x == 3) |
|||
r3x = 0; |
|||
if (!((x & y & 1) + !(!r3x | !r3y)) && !ismasked(x, y)) |
|||
qrframe[x + y * width] ^= 1; |
|||
} |
|||
} |
|||
break; |
|||
case 6: |
|||
for (r3y = 0, y = 0; y < width; y++, r3y++) { |
|||
if (r3y == 3) |
|||
r3y = 0; |
|||
for (r3x = 0, x = 0; x < width; x++, r3x++) { |
|||
if (r3x == 3) |
|||
r3x = 0; |
|||
if (!(((x & y & 1) + (r3x && (r3x == r3y))) & 1) && !ismasked(x, y)) |
|||
qrframe[x + y * width] ^= 1; |
|||
} |
|||
} |
|||
break; |
|||
case 7: |
|||
for (r3y = 0, y = 0; y < width; y++, r3y++) { |
|||
if (r3y == 3) |
|||
r3y = 0; |
|||
for (r3x = 0, x = 0; x < width; x++, r3x++) { |
|||
if (r3x == 3) |
|||
r3x = 0; |
|||
if (!(((r3x && (r3x == r3y)) + ((x + y) & 1)) & 1) && !ismasked(x, y)) |
|||
qrframe[x + y * width] ^= 1; |
|||
} |
|||
} |
|||
break; |
|||
} |
|||
return; |
|||
} |
|||
|
|||
// Badness coefficients.
|
|||
var N1 = 3, N2 = 3, N3 = 40, N4 = 10; |
|||
|
|||
// Using the table of the length of each run, calculate the amount of bad image
|
|||
// - long runs or those that look like finders; called twice, once each for X and Y
|
|||
function badruns(length) |
|||
{ |
|||
var i; |
|||
var runsbad = 0; |
|||
for (i = 0; i <= length; i++) |
|||
if (rlens[i] >= 5) |
|||
runsbad += N1 + rlens[i] - 5; |
|||
// BwBBBwB as in finder
|
|||
for (i = 3; i < length - 1; i += 2) |
|||
if (rlens[i - 2] == rlens[i + 2] |
|||
&& rlens[i + 2] == rlens[i - 1] |
|||
&& rlens[i - 1] == rlens[i + 1] |
|||
&& rlens[i - 1] * 3 == rlens[i] |
|||
// white around the black pattern? Not part of spec
|
|||
&& (rlens[i - 3] == 0 // beginning
|
|||
|| i + 3 > length // end
|
|||
|| rlens[i - 3] * 3 >= rlens[i] * 4 || rlens[i + 3] * 3 >= rlens[i] * 4) |
|||
) |
|||
runsbad += N3; |
|||
return runsbad; |
|||
} |
|||
|
|||
// Calculate how bad the masked image is - blocks, imbalance, runs, or finders.
|
|||
function badcheck() |
|||
{ |
|||
var x, y, h, b, b1; |
|||
var thisbad = 0; |
|||
var bw = 0; |
|||
|
|||
// blocks of same color.
|
|||
for (y = 0; y < width - 1; y++) |
|||
for (x = 0; x < width - 1; x++) |
|||
if ((qrframe[x + width * y] && qrframe[(x + 1) + width * y] |
|||
&& qrframe[x + width * (y + 1)] && qrframe[(x + 1) + width * (y + 1)]) // all black
|
|||
|| !(qrframe[x + width * y] || qrframe[(x + 1) + width * y] |
|||
|| qrframe[x + width * (y + 1)] || qrframe[(x + 1) + width * (y + 1)])) // all white
|
|||
thisbad += N2; |
|||
|
|||
// X runs
|
|||
for (y = 0; y < width; y++) { |
|||
rlens[0] = 0; |
|||
for (h = b = x = 0; x < width; x++) { |
|||
if ((b1 = qrframe[x + width * y]) == b) |
|||
rlens[h]++; |
|||
else |
|||
rlens[++h] = 1; |
|||
b = b1; |
|||
bw += b ? 1 : -1; |
|||
} |
|||
thisbad += badruns(h); |
|||
} |
|||
|
|||
// black/white imbalance
|
|||
if (bw < 0) |
|||
bw = -bw; |
|||
|
|||
var big = bw; |
|||
var count = 0; |
|||
big += big << 2; |
|||
big <<= 1; |
|||
while (big > width * width) |
|||
big -= width * width, count++; |
|||
thisbad += count * N4; |
|||
|
|||
// Y runs
|
|||
for (x = 0; x < width; x++) { |
|||
rlens[0] = 0; |
|||
for (h = b = y = 0; y < width; y++) { |
|||
if ((b1 = qrframe[x + width * y]) == b) |
|||
rlens[h]++; |
|||
else |
|||
rlens[++h] = 1; |
|||
b = b1; |
|||
} |
|||
thisbad += badruns(h); |
|||
} |
|||
return thisbad; |
|||
} |
|||
|
|||
function genframe(instring) |
|||
{ |
|||
var x, y, k, t, v, i, j, m; |
|||
|
|||
// find the smallest version that fits the string
|
|||
t = instring.length; |
|||
version = 0; |
|||
do { |
|||
version++; |
|||
k = (ecclevel - 1) * 4 + (version - 1) * 16; |
|||
neccblk1 = eccblocks[k++]; |
|||
neccblk2 = eccblocks[k++]; |
|||
datablkw = eccblocks[k++]; |
|||
eccblkwid = eccblocks[k]; |
|||
k = datablkw * (neccblk1 + neccblk2) + neccblk2 - 3 + (version <= 9); |
|||
if (t <= k) |
|||
break; |
|||
} while (version < 40); |
|||
|
|||
// FIXME - insure that it fits insted of being truncated
|
|||
width = 17 + 4 * version; |
|||
|
|||
// allocate, clear and setup data structures
|
|||
v = datablkw + (datablkw + eccblkwid) * (neccblk1 + neccblk2) + neccblk2; |
|||
for( t = 0; t < v; t++ ) |
|||
eccbuf[t] = 0; |
|||
strinbuf = instring.slice(0); |
|||
|
|||
for( t = 0; t < width * width; t++ ) |
|||
qrframe[t] = 0; |
|||
|
|||
for( t = 0 ; t < (width * (width + 1) + 1) / 2; t++) |
|||
framask[t] = 0; |
|||
|
|||
// insert finders - black to frame, white to mask
|
|||
for (t = 0; t < 3; t++) { |
|||
k = 0; |
|||
y = 0; |
|||
if (t == 1) |
|||
k = (width - 7); |
|||
if (t == 2) |
|||
y = (width - 7); |
|||
qrframe[(y + 3) + width * (k + 3)] = 1; |
|||
for (x = 0; x < 6; x++) { |
|||
qrframe[(y + x) + width * k] = 1; |
|||
qrframe[y + width * (k + x + 1)] = 1; |
|||
qrframe[(y + 6) + width * (k + x)] = 1; |
|||
qrframe[(y + x + 1) + width * (k + 6)] = 1; |
|||
} |
|||
for (x = 1; x < 5; x++) { |
|||
setmask(y + x, k + 1); |
|||
setmask(y + 1, k + x + 1); |
|||
setmask(y + 5, k + x); |
|||
setmask(y + x + 1, k + 5); |
|||
} |
|||
for (x = 2; x < 4; x++) { |
|||
qrframe[(y + x) + width * (k + 2)] = 1; |
|||
qrframe[(y + 2) + width * (k + x + 1)] = 1; |
|||
qrframe[(y + 4) + width * (k + x)] = 1; |
|||
qrframe[(y + x + 1) + width * (k + 4)] = 1; |
|||
} |
|||
} |
|||
|
|||
// alignment blocks
|
|||
if (version > 1) { |
|||
t = adelta[version]; |
|||
y = width - 7; |
|||
for (;;) { |
|||
x = width - 7; |
|||
while (x > t - 3) { |
|||
putalign(x, y); |
|||
if (x < t) |
|||
break; |
|||
x -= t; |
|||
} |
|||
if (y <= t + 9) |
|||
break; |
|||
y -= t; |
|||
putalign(6, y); |
|||
putalign(y, 6); |
|||
} |
|||
} |
|||
|
|||
// single black
|
|||
qrframe[8 + width * (width - 8)] = 1; |
|||
|
|||
// timing gap - mask only
|
|||
for (y = 0; y < 7; y++) { |
|||
setmask(7, y); |
|||
setmask(width - 8, y); |
|||
setmask(7, y + width - 7); |
|||
} |
|||
for (x = 0; x < 8; x++) { |
|||
setmask(x, 7); |
|||
setmask(x + width - 8, 7); |
|||
setmask(x, width - 8); |
|||
} |
|||
|
|||
// reserve mask-format area
|
|||
for (x = 0; x < 9; x++) |
|||
setmask(x, 8); |
|||
for (x = 0; x < 8; x++) { |
|||
setmask(x + width - 8, 8); |
|||
setmask(8, x); |
|||
} |
|||
for (y = 0; y < 7; y++) |
|||
setmask(8, y + width - 7); |
|||
|
|||
// timing row/col
|
|||
for (x = 0; x < width - 14; x++) |
|||
if (x & 1) { |
|||
setmask(8 + x, 6); |
|||
setmask(6, 8 + x); |
|||
} |
|||
else { |
|||
qrframe[(8 + x) + width * 6] = 1; |
|||
qrframe[6 + width * (8 + x)] = 1; |
|||
} |
|||
|
|||
// version block
|
|||
if (version > 6) { |
|||
t = vpat[version - 7]; |
|||
k = 17; |
|||
for (x = 0; x < 6; x++) |
|||
for (y = 0; y < 3; y++, k--) |
|||
if (1 & (k > 11 ? version >> (k - 12) : t >> k)) { |
|||
qrframe[(5 - x) + width * (2 - y + width - 11)] = 1; |
|||
qrframe[(2 - y + width - 11) + width * (5 - x)] = 1; |
|||
} |
|||
else { |
|||
setmask(5 - x, 2 - y + width - 11); |
|||
setmask(2 - y + width - 11, 5 - x); |
|||
} |
|||
} |
|||
|
|||
// sync mask bits - only set above for white spaces, so add in black bits
|
|||
for (y = 0; y < width; y++) |
|||
for (x = 0; x <= y; x++) |
|||
if (qrframe[x + width * y]) |
|||
setmask(x, y); |
|||
|
|||
// convert string to bitstream
|
|||
// 8 bit data to QR-coded 8 bit data (numeric or alphanum, or kanji not supported)
|
|||
v = strinbuf.length; |
|||
|
|||
// string to array
|
|||
for( i = 0 ; i < v; i++ ) |
|||
eccbuf[i] = strinbuf.charCodeAt(i); |
|||
strinbuf = eccbuf.slice(0); |
|||
|
|||
// calculate max string length
|
|||
x = datablkw * (neccblk1 + neccblk2) + neccblk2; |
|||
if (v >= x - 2) { |
|||
v = x - 2; |
|||
if (version > 9) |
|||
v--; |
|||
} |
|||
|
|||
// shift and repack to insert length prefix
|
|||
i = v; |
|||
if (version > 9) { |
|||
strinbuf[i + 2] = 0; |
|||
strinbuf[i + 3] = 0; |
|||
while (i--) { |
|||
t = strinbuf[i]; |
|||
strinbuf[i + 3] |= 255 & (t << 4); |
|||
strinbuf[i + 2] = t >> 4; |
|||
} |
|||
strinbuf[2] |= 255 & (v << 4); |
|||
strinbuf[1] = v >> 4; |
|||
strinbuf[0] = 0x40 | (v >> 12); |
|||
} |
|||
else { |
|||
strinbuf[i + 1] = 0; |
|||
strinbuf[i + 2] = 0; |
|||
while (i--) { |
|||
t = strinbuf[i]; |
|||
strinbuf[i + 2] |= 255 & (t << 4); |
|||
strinbuf[i + 1] = t >> 4; |
|||
} |
|||
strinbuf[1] |= 255 & (v << 4); |
|||
strinbuf[0] = 0x40 | (v >> 4); |
|||
} |
|||
// fill to end with pad pattern
|
|||
i = v + 3 - (version < 10); |
|||
while (i < x) { |
|||
strinbuf[i++] = 0xec; |
|||
// buffer has room if (i == x) break;
|
|||
strinbuf[i++] = 0x11; |
|||
} |
|||
|
|||
// calculate and append ECC
|
|||
|
|||
// calculate generator polynomial
|
|||
genpoly[0] = 1; |
|||
for (i = 0; i < eccblkwid; i++) { |
|||
genpoly[i + 1] = 1; |
|||
for (j = i; j > 0; j--) |
|||
genpoly[j] = genpoly[j] |
|||
? genpoly[j - 1] ^ gexp[modnn(glog[genpoly[j]] + i)] : genpoly[j - 1]; |
|||
genpoly[0] = gexp[modnn(glog[genpoly[0]] + i)]; |
|||
} |
|||
for (i = 0; i <= eccblkwid; i++) |
|||
genpoly[i] = glog[genpoly[i]]; // use logs for genpoly[] to save calc step
|
|||
|
|||
// append ecc to data buffer
|
|||
k = x; |
|||
y = 0; |
|||
for (i = 0; i < neccblk1; i++) { |
|||
appendrs(y, datablkw, k, eccblkwid); |
|||
y += datablkw; |
|||
k += eccblkwid; |
|||
} |
|||
for (i = 0; i < neccblk2; i++) { |
|||
appendrs(y, datablkw + 1, k, eccblkwid); |
|||
y += datablkw + 1; |
|||
k += eccblkwid; |
|||
} |
|||
// interleave blocks
|
|||
y = 0; |
|||
for (i = 0; i < datablkw; i++) { |
|||
for (j = 0; j < neccblk1; j++) |
|||
eccbuf[y++] = strinbuf[i + j * datablkw]; |
|||
for (j = 0; j < neccblk2; j++) |
|||
eccbuf[y++] = strinbuf[(neccblk1 * datablkw) + i + (j * (datablkw + 1))]; |
|||
} |
|||
for (j = 0; j < neccblk2; j++) |
|||
eccbuf[y++] = strinbuf[(neccblk1 * datablkw) + i + (j * (datablkw + 1))]; |
|||
for (i = 0; i < eccblkwid; i++) |
|||
for (j = 0; j < neccblk1 + neccblk2; j++) |
|||
eccbuf[y++] = strinbuf[x + i + j * eccblkwid]; |
|||
strinbuf = eccbuf; |
|||
|
|||
// pack bits into frame avoiding masked area.
|
|||
x = y = width - 1; |
|||
k = v = 1; // up, minus
|
|||
/* inteleaved data and ecc codes */ |
|||
m = (datablkw + eccblkwid) * (neccblk1 + neccblk2) + neccblk2; |
|||
for (i = 0; i < m; i++) { |
|||
t = strinbuf[i]; |
|||
for (j = 0; j < 8; j++, t <<= 1) { |
|||
if (0x80 & t) |
|||
qrframe[x + width * y] = 1; |
|||
do { // find next fill position
|
|||
if (v) |
|||
x--; |
|||
else { |
|||
x++; |
|||
if (k) { |
|||
if (y != 0) |
|||
y--; |
|||
else { |
|||
x -= 2; |
|||
k = !k; |
|||
if (x == 6) { |
|||
x--; |
|||
y = 9; |
|||
} |
|||
} |
|||
} |
|||
else { |
|||
if (y != width - 1) |
|||
y++; |
|||
else { |
|||
x -= 2; |
|||
k = !k; |
|||
if (x == 6) { |
|||
x--; |
|||
y -= 8; |
|||
} |
|||
} |
|||
} |
|||
} |
|||
v = !v; |
|||
} while (ismasked(x, y)); |
|||
} |
|||
} |
|||
|
|||
// save pre-mask copy of frame
|
|||
strinbuf = qrframe.slice(0); |
|||
t = 0; // best
|
|||
y = 30000; // demerit
|
|||
// for instead of while since in original arduino code
|
|||
// if an early mask was "good enough" it wouldn't try for a better one
|
|||
// since they get more complex and take longer.
|
|||
for (k = 0; k < 8; k++) { |
|||
applymask(k); // returns black-white imbalance
|
|||
x = badcheck(); |
|||
if (x < y) { // current mask better than previous best?
|
|||
y = x; |
|||
t = k; |
|||
} |
|||
if (t == 7) |
|||
break; // don't increment i to a void redoing mask
|
|||
qrframe = strinbuf.slice(0); // reset for next pass
|
|||
} |
|||
if (t != k) // redo best mask - none good enough, last wasn't t
|
|||
applymask(t); |
|||
|
|||
// add in final mask/ecclevel bytes
|
|||
y = fmtword[t + ((ecclevel - 1) << 3)]; |
|||
// low byte
|
|||
for (k = 0; k < 8; k++, y >>= 1) |
|||
if (y & 1) { |
|||
qrframe[(width - 1 - k) + width * 8] = 1; |
|||
if (k < 6) |
|||
qrframe[8 + width * k] = 1; |
|||
else |
|||
qrframe[8 + width * (k + 1)] = 1; |
|||
} |
|||
// high byte
|
|||
for (k = 0; k < 7; k++, y >>= 1) |
|||
if (y & 1) { |
|||
qrframe[8 + width * (width - 7 + k)] = 1; |
|||
if (k) |
|||
qrframe[(6 - k) + width * 8] = 1; |
|||
else |
|||
qrframe[7 + width * 8] = 1; |
|||
} |
|||
|
|||
// return image
|
|||
return qrframe; |
|||
} |
|||
|
|||
var _canvas = null, |
|||
_size = null; |
|||
|
|||
var api = { |
|||
|
|||
get ecclevel () { |
|||
return ecclevel; |
|||
}, |
|||
|
|||
set ecclevel (val) { |
|||
ecclevel = val; |
|||
}, |
|||
|
|||
get size () { |
|||
return _size; |
|||
}, |
|||
|
|||
set size (val) { |
|||
_size = val |
|||
}, |
|||
|
|||
get canvas () { |
|||
return _canvas; |
|||
}, |
|||
|
|||
set canvas (el) { |
|||
_canvas = el; |
|||
}, |
|||
|
|||
getFrame: function (string) { |
|||
return genframe(string); |
|||
}, |
|||
|
|||
draw: function (string, canvas, size, ecc) { |
|||
|
|||
ecclevel = ecc || ecclevel; |
|||
canvas = canvas || _canvas; |
|||
|
|||
if (!canvas) { |
|||
console.warn('No canvas provided to draw QR code in!') |
|||
return; |
|||
} |
|||
|
|||
size = size || _size || Math.min(canvas.width, canvas.height); |
|||
|
|||
var frame = genframe(string), |
|||
ctx = canvas.ctx, |
|||
px = Math.round(size / (width + 8)); |
|||
|
|||
var roundedSize = px * (width + 8), |
|||
offset = Math.floor((size - roundedSize) / 2); |
|||
|
|||
size = roundedSize; |
|||
|
|||
ctx.clearRect(0, 0, canvas.width, canvas.height); |
|||
ctx.setFillStyle('#000000'); |
|||
|
|||
for (var i = 0; i < width; i++) { |
|||
for (var j = 0; j < width; j++) { |
|||
if (frame[j * width + i]) { |
|||
ctx.fillRect(px * (4 + i) + offset, px * (4 + j) + offset, px, px); |
|||
} |
|||
} |
|||
} |
|||
ctx.draw(); |
|||
} |
|||
} |
|||
|
|||
module.exports = { |
|||
api: api |
|||
} |
|||
|
|||
})() |
@ -0,0 +1,171 @@ |
|||
//const gbk = require('./gbk.js');
|
|||
//console.log("sasas" + gbk);
|
|||
const formatTime = date => { |
|||
const year = date.getFullYear() |
|||
const month = date.getMonth() + 1 |
|||
const day = date.getDate() |
|||
const hour = date.getHours() |
|||
const minute = date.getMinutes() |
|||
const second = date.getSeconds() |
|||
|
|||
return [year, month, day].map(formatNumber).join('/') + ' ' + [hour, minute, second].map(formatNumber).join(':') |
|||
} |
|||
|
|||
const formatNumber = n => { |
|||
n = n.toString() |
|||
return n[1] ? n : '0' + n |
|||
} |
|||
|
|||
/* |
|||
const hexStringToBuff = str => { //str='中国:WXHSH'
|
|||
const buffer = new ArrayBuffer((sumStrLength(str)) * 4) |
|||
const dataView = new DataView(buffer) |
|||
var data = str.toString(); |
|||
var p = 0; //ArrayBuffer 偏移量
|
|||
for (var i = 0; i < data.length; i++) { |
|||
if (isCN(data[i])) { //是中文
|
|||
//调用GBK 转码
|
|||
var t = gbk.encode(data[i]); |
|||
for (var j = 0; j < 2; j++) { |
|||
//var code = t[j * 2] + t[j * 2 + 1];
|
|||
var code = t[j * 3 + 1] + t[j * 3 + 2]; |
|||
var temp = parseInt(code, 16) |
|||
//var temp = strToHexCharCode(code);
|
|||
dataView.setUint8(p++, temp) |
|||
} |
|||
} else { |
|||
var temp = data.charCodeAt(i); |
|||
dataView.setUint8(p++, temp) |
|||
} |
|||
} |
|||
return buffer; |
|||
} |
|||
*/ |
|||
function toUnicode(s) { |
|||
var str = ""; |
|||
for (var i = 0; i < s.length; i++) { |
|||
str += "\\u" + s.charCodeAt(i).toString(16) + "\t"; |
|||
} |
|||
return str; |
|||
} |
|||
|
|||
function strToHexCharCode(str) { |
|||
if (str === "") |
|||
return ""; |
|||
var hexCharCode = []; |
|||
hexCharCode.push("0x"); |
|||
for (var i = 0; i < str.length; i++) { |
|||
hexCharCode.push((str.charCodeAt(i)).toString(16)); |
|||
} |
|||
return hexCharCode.join(""); |
|||
} |
|||
|
|||
function sumStrLength(str) { |
|||
var length = 0; |
|||
var data = str.toString(); |
|||
for (var i = 0; i < data.length; i++) { |
|||
if (isCN(data[i])) { //是中文
|
|||
length += 2; |
|||
} else { |
|||
length += 1; |
|||
} |
|||
} |
|||
return length; |
|||
} |
|||
|
|||
function isCN(str) { |
|||
if (/^[\u3220-\uFA29]+$/.test(str)) { |
|||
return true; |
|||
} else { |
|||
return false; |
|||
} |
|||
} |
|||
|
|||
//汉字转码
|
|||
export function hexStringToArrayBuffer(str) { |
|||
const buffer = new ArrayBuffer((str.length / 2) + 1) |
|||
const dataView = new DataView(buffer) |
|||
for (var i = 0; i < str.length / 2; i++) { |
|||
var temp = parseInt(str[i * 2] + str[i * 2 + 1], 16) |
|||
dataView.setUint8(i, temp) |
|||
} |
|||
dataView.setUint8((str.length / 2), 0x0a) |
|||
return buffer; |
|||
} |
|||
|
|||
//返回八位数组
|
|||
function subString(str) { |
|||
var arr = []; |
|||
if (str.length > 8) { //大于8
|
|||
for (var i = 0; |
|||
(i * 8) < str.length; i++) { |
|||
var temp = str.substring(i * 8, 8 * i + 8); |
|||
arr.push(temp) |
|||
} |
|||
return arr; |
|||
} else { |
|||
return str |
|||
} |
|||
} |
|||
|
|||
//不带有汉字
|
|||
function hexStringToArrayBufferstr(str) { |
|||
let val = "" |
|||
for (let i = 0; i < str.length; i++) { |
|||
if (val === '') { |
|||
val = str.charCodeAt(i).toString(16) |
|||
} else { |
|||
val += ',' + str.charCodeAt(i).toString(16) |
|||
} |
|||
} |
|||
val += "," + "0x0a"; |
|||
console.log(val) |
|||
// 将16进制转化为ArrayBuffer
|
|||
return new Uint8Array(val.match(/[\da-f]{2}/gi).map(function(h) { |
|||
return parseInt(h, 16) |
|||
})).buffer |
|||
} |
|||
|
|||
export function ab2hex(buffer) { |
|||
let hexArr = Array.prototype.map.call( |
|||
new Uint8Array(buffer), |
|||
function (bit) { |
|||
return ('00' + bit.toString(16)).slice(-2) |
|||
}) |
|||
return hexArr.join(''); |
|||
} |
|||
|
|||
// ArrayBuffer转为字符串,参数为ArrayBuffer对象
|
|||
export function ab2str(buf) { |
|||
return String.fromCharCode.apply(null, new Uint8Array(buf)); |
|||
} |
|||
|
|||
// 字符串转为ArrayBuffer对象,参数为字符串
|
|||
export function str2ab(str) { |
|||
var buf = new ArrayBuffer(str.length+1); // 补充/0
|
|||
var bufView = new Uint8Array(buf); |
|||
for (var i = 0, strLen = str.length; i < strLen; i++) { |
|||
bufView[i] = str.charCodeAt(i); |
|||
} |
|||
return buf; |
|||
} |
|||
|
|||
export function send0X0A() { |
|||
const buffer = new ArrayBuffer(1) |
|||
const dataView = new DataView(buffer) |
|||
dataView.setUint8(0, 0x0a) |
|||
return buffer; |
|||
} |
|||
|
|||
export function buf2hex(buffer) { |
|||
return Array.prototype.map.call(new Uint8Array(buffer), x => ('00' + x.toString(16)).slice(-2)).join(''); |
|||
} |
|||
|
|||
// module.exports = {
|
|||
// hexStringToArrayBuffer: hexStringToArrayBuffer,
|
|||
// send0X0A: send0X0A,
|
|||
// ab2hex: ab2hex,
|
|||
// str2ab: str2ab,
|
|||
// ab2str: ab2str,
|
|||
// buf2hex: buf2hex
|
|||
// }
|
@ -0,0 +1,70 @@ |
|||
<template> |
|||
<view class="content"> |
|||
<view class="btn-wrap"> |
|||
<kk-printer ref="kkprinter" :bufferData="bufferData" @onPrint="onPrint"></kk-printer> |
|||
</view> |
|||
</view> |
|||
</template> |
|||
|
|||
<script> |
|||
import util from '@/components/kk-printer/utils/util.js'; |
|||
import * as blesdk from '@/components/kk-printer/utils/bluetoolth.js'; |
|||
import kkPrinter from '@/components/kk-printer/index.vue'; |
|||
export default { |
|||
data() { |
|||
return { |
|||
bufferData:'' |
|||
} |
|||
}, |
|||
components:{ |
|||
kkPrinter |
|||
}, |
|||
mounted() { |
|||
|
|||
}, |
|||
methods: { |
|||
onPrint(opt){ |
|||
let arr = ['2','3'] |
|||
let y = 450 |
|||
let strCmd =blesdk.CreatCPCLPage(400,y*arr.length,1,0); |
|||
arr.forEach((item,index)=>{ |
|||
strCmd += blesdk.addCPCLLine(0,210 + (y*index),400,210+ (y*index),3); |
|||
strCmd += blesdk.addCPCLText(10,0 + (y*index),'4','3',0,'8.14'); |
|||
strCmd += blesdk.addCPCLBarCode(270,0 + (y*index),'128',80,0,1,1,'00051'); |
|||
strCmd += blesdk.addCPCLText(290,80 + (y*index),'7','2',0,'00051'); |
|||
strCmd += blesdk.addCPCLText(40,110 + (y*index),'3','0',0,'CHICKEN FEET (BONELESS)'); |
|||
// strCmd += blesdk.addCPCLSETMAG(2,2); |
|||
strCmd += blesdk.addCPCLText(40,150 + (y*index),'55','0',0,'无骨鸡爪 一盒(约1.5磅)'); |
|||
// strCmd += blesdk.addCPCLSETMAG(0,0); |
|||
strCmd += blesdk.addCPCLText(0,180 + (y*index),'7','2',0,'2019-08-12'); |
|||
strCmd += blesdk.addCPCLLocation(2);//循环里之后的所有的都居中了 |
|||
strCmd += blesdk.addCPCLQRCode(0,220 + (y*index),'M', 2, 6, 'qr code test'); |
|||
}) |
|||
console.log(strCmd) |
|||
strCmd += blesdk.addCPCLPrint(); |
|||
this.bufferData = strCmd; |
|||
} |
|||
} |
|||
} |
|||
</script> |
|||
|
|||
<style> |
|||
.content { |
|||
width: 100vw; |
|||
height: 100vh; |
|||
display: flex; |
|||
flex-direction: column; |
|||
align-items: center; |
|||
justify-content: center; |
|||
} |
|||
.btn-wrap{ |
|||
width:180upx; |
|||
height: 100upx; |
|||
border-radius: 16upx; |
|||
border: 2upx solid #333333; |
|||
box-sizing: border-box; |
|||
display: flex; |
|||
justify-content: center; |
|||
align-items: center; |
|||
} |
|||
</style> |
Loading…
Reference in new issue