You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

454 lines
12 KiB

<template>
<div>
<el-dialog
v-model="details.popUpShow.visible"
destroy-on-close
align-center
:title="props.title"
:width="props.width"
:fullscreen="props.isFullscreen && details.isFullscreen"
:show-close="!props.isFullscreen"
>
<!-- 标题 -- 头部控件区 -->
<template v-if="props.isFullscreen" #header="{ close, titleId, titleClass }">
<div class="my-header flex-c-sb">
<div class="fwb" :id="titleId" :class="titleClass">{{ props.title }}</div>
<div class="flex-c-c">
<!-- 全屏显示按钮 -->
<el-button type="text" @click="handleFullScrean()" v-if="!details.isFullscreen">
<el-icon class=""><FullScreen /></el-icon>
</el-button>
<el-button type="text" @click="handleFullScrean()" v-else>
<el-icon class=""><CopyDocument /></el-icon>
</el-button>
<!-- 弹窗关闭按钮 -->
<el-button type="text">
<el-icon class="" @click="close"><Close /></el-icon>
</el-button>
</div>
</div>
</template>
<div class="printCode" ref="printNodeRef">
<template v-if="details.html">
<div v-html="details.html"></div>
</template>
<div>
<slot></slot>
</div>
</div>
<div class="flex-c-c mt10">
<el-button @click="handleClose" icon="CircleClose"> </el-button>
<el-button type="primary" v-if="props.isShowExport" @click="handleExport" icon="Download">
导 出
</el-button>
<el-button
type="primary"
@click="printTemplate"
icon="Printer"
:disabled="details.printLoading"
:loading="details.printLoading"
>
打 印
</el-button>
<el-button
type="primary"
@click="handleShowEdit"
icon="Edit"
:disabled="details.printLoading"
:loading="details.printLoading"
>
编辑打印机
</el-button>
<!-- <el-select
v-model="activeIndex"
placeholder="打印机"
style="width: 200px; margin-left: 10px"
>
<el-option
v-for="item in $store.state.print.printList"
:label="item.label"
:value="item.value"
:key="item.label"
/>
</el-select> -->
</div>
</el-dialog>
<!-- 确认打印机 -->
<el-dialog v-model="isShow" title="确认打印机" align-center width="fit-content">
<div style="margin-bottom: 40px" class="flex-c-c">
<div class="flex-none">编辑打印机:</div>
<el-select
v-model="activeIndex"
placeholder="打印机"
style="width: 200px; margin-left: 10px"
>
<el-option
v-for="item in $store.state.print.printList"
:label="item.label"
:value="item.value"
:key="item.label"
/>
</el-select>
</div>
</el-dialog>
<el-dialog
v-model="details.popUpShow.setNum"
title="确认打印标签数"
align-center
width="50%"
destroy-on-close
>
<template v-for="item in details.waybillInfo.detailList">
<div class="fwb">品类名:{{ item.productName }}</div>
<div class="flex mt10 mb20">
<div class="flex1 align-center mr20">
<div class="mr10">开始索引</div>
<el-input-number
class="flex1"
v-model="item.start"
:min="1"
:max="item.num"
></el-input-number>
</div>
<div class="flex1 align-center">
<div class="mr10">结束索引</div>
<el-input-number
class="flex1"
v-model="item.end"
:min="isNumber(item.start > 1) ? item.start : 0"
:max="item.num"
></el-input-number>
</div>
</div>
</template>
<div class="flex-c-sb">
<div></div>
<div>
<el-button @click="details.popUpShow.setNum = false">取消</el-button>
<el-button type="primary" @click="handleShowPrint" icon="Printer">确认</el-button>
</div>
</div>
</el-dialog>
</div>
</template>
<script setup lang="ts">
import { defineProps, computed, ref, nextTick, reactive, watch } from 'vue';
import print from '@/utils/print';
import { ElMessage, ElMessageBox } from 'element-plus';
import { getObjType } from '@/utils/util';
import * as XLSX from 'xlsx';
import { exportExcelByDom } from '@/utils/export';
import { dateNow } from '@/utils/date';
import { postFindZeroWaybillLabelInfo } from '@/api/waybill/WaybillOrderList.js';
import { useStore } from 'vuex';
import { isNumber } from '@antfu/utils';
const $store = useStore();
const props = defineProps({
/** 弹窗宽度 */
width: {
type: String,
default: '780px',
},
/** 标题 */
title: {
type: String,
default: '二维码',
},
/** 是否显示导出按钮 */
isShowExport: {
type: Boolean,
default: true,
},
/** 打印函数 */
printFn: {
type: Function,
default: null,
},
/** 打印类型 */
type: {
type: String,
default: 'titlePrint',
},
/** 导出名称 */
exportName: {
type: String,
default: '导出',
},
/** 是否开启全屏按钮 */
isFullscreen: {
type: Boolean,
default: false,
},
});
const $emit = defineEmits(['update:modelValue']);
const isShow = ref(false);
const activeIndex = computed({
get() {
return $store.state.print[props.type + 'ActiveIndex'];
},
set(activeIndex) {
// 设置打印机
$store.commit('SET_PRINT_ACTIVE', { activeIndex, type: props.type });
},
});
const details = reactive({
/** 打印loading */
printLoading: false,
/** 是否全屏 */
isFullscreen: false,
/** 运单信息 */
waybillInfo: {} as any,
popUpShow: {
/** 打印 */
visible: false,
setNum: false,
},
html: '',
waybillNo: '',
});
const handleClose = () => {
details.popUpShow.visible = false;
};
const printNodeRef = ref();
/** 生成Excel */
const createExcel = () => {
// const table = this.$refs.tableRef;
/* 将表格转换为 Workbook 对象 */
const wb = XLSX.utils.table_to_book(printNodeRef.value);
console.log('wb :>> ', wb);
wb.Sheets.Sheet1;
//单元格外侧框线
const borderAll = {
top: {
style: 'thin',
},
bottom: {
style: 'thin',
},
left: {
style: 'thin',
},
right: {
style: 'thin',
},
};
for (const key in wb.Sheets.Sheet1) {
const item = wb.Sheets.Sheet1[key];
const type = getObjType(item);
console.log('type :>> ', type);
if (type !== 'object') continue;
item.s = {
border: borderAll,
font: {
// name: "微软雅黑",
// sz: 16,
color: { rgb: '000000' },
bold: true,
italic: false,
underline: false,
},
fill: {
fgColor: { rgb: 'C5D9F1' },
},
alignment: {
horizontal: 'center',
vertical: 'center',
wrapText: false, // 自动换行
},
};
item.t = 's';
}
/* 导出 Workbook 到 Excel 文件 */
XLSX.writeFile(wb, 'data.xlsx');
};
/** 导出 */
const handleExport = () => {
if (!printNodeRef.value) return ElMessage.warning('数据暂未请求完成, 请稍等');
ElMessageBox.confirm('确认导出?', '', {
confirmButtonText: '确认',
cancelButtonText: '关闭',
type: 'warning',
}).then(() => {
console.log('printNodeRef.value :>> ', printNodeRef.value);
props.type === 'titlePrint'
? createExcel()
: exportExcelByDom(printNodeRef.value, props.exportName + ' - ' + dateNow() + '.xlsx');
});
};
/** 打印 */
const printTemplate = () => {
if (props.printFn) return props.printFn();
details.printLoading = true;
const nodeArr = document.querySelectorAll('.printSmallText');
for (let i = 0; i < nodeArr.length; i++) {
const val = nodeArr[i] as any;
val.style.fontSize = '0.5rem';
val.style.lineHeight = '0.7rem';
}
const timer = setTimeout(() => {
const printNode = document.querySelectorAll('.printCode > div > div');
console.log('printNode :>> ', printNode);
print(printNode, props.type);
handleClose();
details.printLoading = false;
clearTimeout(timer);
}, 500);
};
/**
* 是否开启床车明细全屏
* @params(_type) 开启或关闭
*/
const handleFullScrean = () => {
details.isFullscreen = !details.isFullscreen;
};
const handleGetWaybillInfo = async (waybillNo, loading, loadingKey = 'pageLoading') => {
try {
loading[loadingKey] = true;
details.waybillNo = waybillNo;
const res = await postFindZeroWaybillLabelInfo({ waybillNo });
const { code, data } = res.data;
if (code !== 200) return;
if (getObjType(data) !== 'object') return;
details.waybillInfo = data;
if (getObjType(details.waybillInfo.detailList) !== 'array') details.waybillInfo.detailList = [];
for (let i = 0; i < details.waybillInfo.detailList.length; i++) {
const value = details.waybillInfo.detailList[i];
value.start = value.num > 1 ? 1 : 0;
value.end = value.num >= 1 ? value.num : 0;
}
details.popUpShow.setNum = true;
} catch (error) {
console.log('error :>> ', error);
} finally {
loading[loadingKey] = false;
}
};
const handleShowPrint = () => {
let flag = true;
let html = '';
let waybillProductName = '';
for (let i = 0; i < details.waybillInfo.detailList.length; i++) {
const value = details.waybillInfo.detailList[i];
waybillProductName += value.productName + ' - ' + value.num + '<br />';
}
for (let i = 0; i < details.waybillInfo.detailList.length; i++) {
const value = details.waybillInfo.detailList[i];
if (value.end <= 0) continue;
flag = false;
for (let index = value.start; index <= value.end; index++) {
let template = details.waybillInfo.template;
template = template.replace('发站仓信息', details.waybillInfo.destinationWarehouseName);
template = template.replace('目的仓信息', details.waybillInfo.departureWarehouseName);
template = template.replace('发站信息', details.waybillInfo.destination);
template = template.replace('到站信息', details.waybillInfo.departure);
template = template.replace('运单号', details.waybillNo);
template = template.replace('订单号', details.waybillInfo.orderNo);
template = template.replace('运单地址', details.waybillInfo.waybillNoCode);
template = template.replace('总件数值', details.waybillInfo.totalCount);
template = template.replace('总重量值', details.waybillInfo.totalWeight);
template = template.replace('总方数值', details.waybillInfo.total_volume);
template = template.replace('货物单项信息', value.productName + ' - ' + index);
template = template.replace(
'寄件人信息',
details.waybillInfo.shipperName +
details.waybillInfo.shipperMobile +
details.waybillInfo.shipperAddress
);
template = template.replace('货物信息', waybillProductName);
template = template.replace(
'收件人信息',
details.waybillInfo.consigneeName +
details.waybillInfo.consigneeMobile +
details.waybillInfo.consigneeAddress
);
template = template.replace('送货方式值', details.waybillInfo.delivery_way);
template = template.replace('送货时间值', details.waybillInfo.createDate.split(' ')[0]);
html += template;
}
}
if (flag) return ElMessage.warning('请输入需要打印的范围');
details.html = html;
details.popUpShow.visible = true;
};
const handleShowEdit = () => {
$store.commit('GET_PRINT_LIST');
isShow.value = true;
};
defineExpose({ handleGetWaybillInfo });
</script>
<style scoped lang="scss">
.printCode {
max-height: 80vh;
overflow-y: scroll;
}
.my-header {
:deep(.el-icon) {
font-size: 20px;
}
}
</style>