今天领导给了个需求,需要配合其他项目组给一个公共组件的npm进行,公共组件打印,打印操作由这个npm包来操作。(经过开会商讨,最后决定配置一个path路径,来展示页面,然后我负责在公共这里打印相应页面内容。)在这之初都是好着的,首先进行配置,这里只需要一个input框来给后端传值就可以了,这一步没什么好讲的,
好了 正头戏开始了,通过请求在mounted中发送请求,获取到配置好的 path路径,开始我想着用路由进行编写后来发现过于麻烦,选择了iframe标签来进行编写,(这里再说一下留下来的需求,需要弹窗然后展示其他组要展示的内容)这里选择了iframe ,然而iframe的src值如果不是本项目页面的话 是会报跨域的,so(幸好我们是同一个域名同一个端口号) 这里我对src 路径的最前面的地址进行处理,编写一个 获取域名以及端口号等的一个过滤方法,方法如下:
// 获取http域名
urlhttp() {
console.log(window.location.href, "window.location.href");
// 截取域名前面的部分
const url = window.location.href.split("://")[1];
const urlhttp = window.location.href.split("://")[0];
const urlArr = url.split("/");
const urlStr = urlArr[0];
const urlStrArr = urlStr.split(".");
const urlStrArrLen = urlStrArr.length;
const urlStrArrLenLast = urlStrArrLen - 1;
const urlStrArrLenLastStr = urlStrArr[urlStrArrLenLast];
const urlStrArrLenLastStrLen = urlStrArrLenLastStr.length;
const urlStrArrLenLastStrLenLast = urlStrArrLenLastStrLen - 1;
const urlStrArrLenLastStrLenLastStr = urlStrArrLenLastStr[urlStrArrLenLastStrLenLast];
const urlStrArrLenLastStrLenLastStrNum = Number(urlStrArrLenLastStrLenLastStr);
if (urlStrArrLenLastStrLenLastStrNum) {
this.urlStr = urlStrArr[urlStrArrLenLastStrLenLastStrNum];
} else {
this.urlStr = urlStrArr[urlStrArrLenLastStrLenLastStr];
}
console.log(urlhttp + urlStr, "urlStr");
// 截取域名带http+域名
this.urlhttpStr = urlhttp + "://" + urlStr;
console.log(this.urlhttpStr, "urlhttpStr");
},
通过次方法,对页面域名进行格式化,其中window.location.href 获取的是当前地址框的所有地址,然后通过.split(" : / / " )[1];等进行数据的获取,最终得到:http://localhost:8080 这种地址,
从而拼接起来我配置的 iframe的src地址拼接后,弹窗展示了相应的页面,
然后在弹窗上给一个打印按钮(重头戏来了) 我先对iframe标签进行ref的设置,从而获取该节点,然后我将
const contentWindow = iframe.contentWindow.document;
节点进行赋值,拿到后直接打印该节点,
调用 contentWindow.print(); 方法,发现打印信息不全的情况,然后我通过翻阅资料方得知 会有样式排版等一些问题导致至此,所以次方法pass,过,后又发现可以通过创建新的页面进行展示打印,然后我调用
const printWindow = window.open("", "_blank");
printWindow.document.write(`
`);
去创建新页面,在新页面中执行打印方法
printWindow.document.close();
printWindow.print();
然后发现打印后,我iframe中的打印的东西消失了,然后我便复制一个子节点然后再用子节点打印
const childElementcopy = childElement.cloneNode(true); // 复制元素
发现奏效了,但是没有css样式,我开始便利子节点的样式
Array.from(sourceComputedStyle).forEach(style => {
const value = sourceComputedStyle.getPropertyValue(style);
printElement.style.setProperty(style, value);
});
但是次方法未奏效,尝试了好多办法后,我查看元素时候发现 style标签我可以直接copy旗下所有元素啊 然后便复制了一份,一起加入打印的新页面,至此完成次需求,下面是全部代码
async submitPrint() {
const iframe = this.$refs.myIframe;
console.log(this.$refs.myIframe, "this.$refs.myIframe");
const contentWindow = iframe.contentWindow.document;
const head = contentWindow.querySelector("head")
// 访问iframe的子元素
const childElement = contentWindow.getElementById(this.printLocation) || contentWindow.getElementsByClassName(this.printLocation)[0]
//this.printLocation是我后配置的要打印模块的id
console.log(childElement);
const childElementcopy = childElement.cloneNode(true); // 复制html元素
const headcopy = head.cloneNode(true); // 复制css 样式元素元素
const printWindow = window.open("", "_blank");
// 在前确保先监听
printWindow.document.write(`
`);
const printBody = printWindow.document.querySelector("body");
const printhead = printWindow.document.querySelector("html");
printhead.appendChild(headcopy);
printBody.appendChild(childElementcopy);
printWindow.onafterprint = function() {
// 在打印对话框关闭后执行的逻辑
console.log("打印对话框已关闭");
// 判断用户是否点击了取消按钮
var printResult = window.matchMedia("print");
if (printResult.matches) {
console.log("用户点击了取消按钮");
// 执行取消打印的相关逻辑
printWindow.close(); // 关闭新页面
} else {
console.log("用户点击了打印按钮");
// 执行打印操作的相关逻辑
printWindow.close(); // 关闭新页面
}
};
setTimeout(() => {
printWindow.document.close();
printWindow.print();
}, 500);
// const printBody = printWindow.document.querySelector('body');
// const printhead = printWindow.document.querySelector('html');
// printhead.appendChild(headcopy);
// printBody.appendChild(childElementcopy);
// printWindow.document.close();
// printWindow.print();
// 监听新窗口的onload事件,在加载完成后执行打印操作
},