当前位置: 首页 > wzjs >正文

百度做网站好吗目前引流最好的app

百度做网站好吗,目前引流最好的app,专业东莞网站制作公司,网站建设 台州使用PDFBox可以渲染生成pdf文档,并且自定义程度高,只是比较麻烦,pdf的内容位置都需要手动设置x(横向)和y(纵向)绝对位置,但是每个企业的单据都是不一样的,一般来说都会设…

使用PDFBox可以渲染生成pdf文档,并且自定义程度高,只是比较麻烦,pdf的内容位置都需要手动设置x(横向)和y(纵向)绝对位置,但是每个企业的单据都是不一样的,一般来说都会设置一个模板,然后内容再填充到适当位置,所以这个功能还是有用的

1. 实际效果

在这里插入图片描述
填充数据后效果
在这里插入图片描述
实现代码:
以下代码基于PDFBox依赖版本-2.0.23

public class Demo01 {public static void main(String[] args) throws Exception{// 设定中文字体File fontFile = new File("C:\\Windows\\Fonts\\simHei.ttf");try (PDDocument document = new PDDocument()) {PDType0Font load = PDType0Font.load(document, fontFile);PDPage page;for (int i = 0; i < 1; i++) {page = new PDPage();document.addPage(page);// 对具体PDPage设定内容try(PDPageContentStream contentStream = new PDPageContentStream(document, page)) {contentStream.setFont(load, 25);contentStream.beginText();// newLineAtOffset方法contentStream.newLineAtOffset(220, 750);contentStream.showText("借用出库打印单");contentStream.setFont(load, 12);contentStream.endText();// 仓库和会员渲染位置contentStream.beginText();contentStream.newLineAtOffset(80, 700); // 80,700contentStream.showText("仓库:");contentStream.newLineAtOffset(300, 0); //380,700contentStream.showText("会员:");contentStream.endText();// 销售员和操作人渲染位置contentStream.beginText();contentStream.newLineAtOffset(80, 675); // 80,675contentStream.showText("销售员:");contentStream.newLineAtOffset(300, 0); //380,675contentStream.showText("操作人:");contentStream.endText();// 操作时间位置contentStream.beginText();contentStream.newLineAtOffset(80, 650); // 80,650contentStream.showText("操作时间:");contentStream.endText();// ----------------实际内容-----------------------// 表头contentStream.beginText();contentStream.newLineAtOffset(80, 625); //80,625contentStream.showText("序号");contentStream.newLineAtOffset(40, 0); //120,625contentStream.showText("商品编号");contentStream.newLineAtOffset(80, 0); //200,625contentStream.showText("商品名称");contentStream.newLineAtOffset(70, 0); //270,625contentStream.showText("单位");contentStream.newLineAtOffset(40, 0); //310,625contentStream.showText("借出数量");contentStream.newLineAtOffset(70, 0); //380,625contentStream.showText("备注");contentStream.newLineAtOffset(100, 0); //480,625contentStream.showText("零售价");contentStream.endText();Map<String, String> contentMap = new HashMap<>();contentMap.put("序号", "1");contentMap.put("商品编号", "000212130023");contentMap.put("商品名称", "洗地机124123");contentMap.put("单位", "个");contentMap.put("借出数量", "13");contentMap.put("备注", "我是备注我是备注");contentMap.put("零售价", "1123300.34");fillContent(contentStream, contentMap, load);// 结尾结构渲染// 合计位置contentStream.beginText();contentStream.newLineAtOffset(80, 150); // 80,150contentStream.showText("合计");contentStream.endText();// 出库数量和总金额位置contentStream.beginText();contentStream.newLineAtOffset(110, 125); // 110,125contentStream.showText("出库数量:");contentStream.newLineAtOffset(270, 0); // 380,125contentStream.showText("总金额:");contentStream.endText();// 签名位置contentStream.beginText();contentStream.newLineAtOffset(80, 50); // 110,125contentStream.showText("签名:_______");contentStream.endText();// 模拟填充模板Map<String, String> map = new HashMap<>();map.put("仓库", "上海仓");map.put("会员", "小明");map.put("销售员", "销售员01");map.put("操作人", "系统管理员");map.put("操作时间", "2025年4月1日23点07分");map.put("出库数量", "1455");map.put("总金额", "285743835.45");fillTemplate(contentStream, map);}}document.save("demo01.pdf");System.out.println("PDF created successfully!");} catch (IOException e) {throw new RuntimeException(e);}}// 填充固定模板方法 该方法不填充中间详细内容public static void fillTemplate(PDPageContentStream contentStream, Map<String, String> map) {try {contentStream.beginText();contentStream.newLineAtOffset(130, 700);contentStream.showText(map.get("仓库"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(430, 700);contentStream.showText(map.get("会员"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(130, 675);contentStream.showText(map.get("销售员"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(430, 675);contentStream.showText(map.get("操作人"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(150, 650);contentStream.showText(map.get("操作时间"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(180, 125);contentStream.showText(map.get("出库数量"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(430, 125);contentStream.showText(map.get("总金额"));contentStream.endText();} catch (IOException e) {throw new RuntimeException(e);}}public static void fillContent(PDPageContentStream contentStream, Map<String, String> map, PDType0Font font) {try {contentStream.setFont(font, 10);contentStream.beginText();contentStream.newLineAtOffset(80, 600);contentStream.showText(map.get("序号"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(120, 600);contentStream.showText(map.get("商品编号"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(200, 600);contentStream.showText(map.get("商品名称"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(270, 600);contentStream.showText(map.get("单位"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(310, 600);contentStream.showText(map.get("借出数量"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(380, 600);contentStream.showText(map.get("备注"));contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(480, 600);contentStream.showText(map.get("零售价"));contentStream.endText();contentStream.setFont(font, 12);} catch (IOException e) {throw new RuntimeException(e);}}
}

上述代码看着确实是挺繁琐,每个内容的位置都需要设置x和y值,但是没办法
PDF文件本质是「坐标画布」‌:

  1. PDF的渲染模型基于‌绝对坐标系‌(原点在页面左下角),所有元素(文字、图形)必须明确指定位置(x,y)。
  2. 无布局引擎‌:PDF规范未定义“自动换行”或“文档流”等高级排版概念,开发者需自行计算坐标。

但是这样设计的好处就是自定义程度高,你可以任意设计一个PDF文档的模板应该是什么样子,内容该如何填充全部由你自由设定,就像低代码平台一样,市面上成熟开源的低代码平台有许多,但是逻辑都是一开始就定好的,如果你想加上许多符合自己公司需求的功能但是平台没有那么都得自行开发,并且自行开发的代码融合进已有的系统不是一件容易的事情,甚至比自行开发一套系统都麻烦。
所以如果你有这样的需求可以看下上述代码实现,上述代码只是一个简单的demo,我只是进行记录方便自己以后用到。

tips:关于一些方法的解释

                    contentStream.beginText();contentStream.newLineAtOffset(80, 700); // 80,700  --绝对定位contentStream.showText("仓库:");contentStream.newLineAtOffset(300, 0); //380,700  --相对定位(以'仓库:'的位置为准)contentStream.showText("会员:");contentStream.endText();contentStream.beginText();contentStream.newLineAtOffset(80, 675); // 80,675  --绝对定位contentStream.showText("销售员:");contentStream.newLineAtOffset(300, 0); //380,675  --相对定位(以'销售员'的位置为准)contentStream.showText("操作人:");contentStream.endText();

上述代码可以看到在渲染内容时是被包裹在beginText()和endText()方法中间的,这样当你调用newLineAtOffset(x, y)方法时参数中的x和y才从坐标系的绝对位置(绝对位置为画布的左下角0,0)进行定位。如果你在定位时没有重新开启beginText()和endText()时,调用newLineAtOffset(x, y)方法则是参照上一个文本的位置进行相对定位的,相对定位对于需要在同一行的不同位置渲染内容会比较方便。
newLineAtOffset(x, y)方法的官方注释有问题,官方说法是移动到下一行的开头,从当前行的开头进行偏移 (x, y),实测不对,并不会移动到下一行的开头,并且在相对定位时参考的位置也是你上一次的位置的起始点。
如果你需要像写文章那样一段一段的文字进行渲染,那么可以考虑使用另外一个方法

     				contentStream.beginText();contentStream.newLineAtOffset(80, 500); // 设定绝对位置的起点contentStream.setLeading(20); // 文本行距contentStream.showText("XXXXX"); //渲染内容contentStream.newLine(); //开启新行contentStream.showText("XXXXX"); //渲染内容contentStream.newLine(); //开启新行contentStream.showText("XXXXX"); //渲染内容contentStream.newLine(); //开启新行contentStream.endText();

这个方法更适合大段连贯的文字渲染,你只要设定好固定行距之后就可以直接开启新行,新行的位置会成功进入到下一行的开头并且行距就是你设定的值,这样你就不用每次都自行定位了,效果如下
在这里插入图片描述

http://www.dtcms.com/wzjs/575577.html

相关文章:

  • 淘宝客做网站crm管理系统app
  • 广州市做企业网站网站建设狼盾网络
  • 优秀企业网站案例哔哩哔哩网页版和客户端哪个好
  • 自己怎么开网站申请网站的域名
  • 昆明市建设厅网站两学一做网站是多少
  • 手机网站规格贵阳市做网站电话
  • 校园网站建设与管理海市科技网站建设
  • 电商网站开发教程汕头网络推广平台
  • 南通网站推广公司wordpress 自定义文章类型
  • 南通长城建设集团有限公司网站本机安装wordpress
  • 在线制作图片网站自己在线制作logo免费 生成器
  • 做网站 传视频 用什么笔记本好在线免费看影视网站
  • 女頻做的最好的网站东莞十大保安公司
  • php 网站发布php c2c网站开发的 书
  • 上海免费模板建站免费ppt模板下载包图网
  • 嘉兴营销型网站建设网页设计代码如何写实训报告过程
  • 东莞网站建设优化推广广州白云网站建设
  • 网站被host重定向如何开发一个app软件
  • 丹阳市建设局网站网上接效果图平台
  • 上海网站设计与制作公司都是自己制作网站
  • 企业网站硬件设计免费行情软件app网站下载大全
  • 网站建设 软件开发的公司排名多语言网站制作
  • 毕设源码网站艺术设计方案
  • 工作室 网站备案手机做网站服务器
  • 建设科技网络网站的意义和目的网站建设合同细节
  • 滨州哪里有做网站的seo81
  • 网站建设新手教程视频北京做网站电话
  • 河南省建设教育协会网站首页网站建设的前端用什么编程
  • html手机网站怎么做广州 网站建设模板
  • 产品做推广都有那些网站wordpress边栏个性化