(水印)html转图片

简介: html转图片
<dependency>
    <groupId>gui.ava</groupId>
    <artifactId>html2image</artifactId>
    <version>2.0.1</version>
</dependency>
<dependency>
    <groupId>xml-apis</groupId>
    <artifactId>xml-apis</artifactId>
    <version>1.4.01</version>
</dependency

html


String htmlTemplate = "<div>" +
        "<span>IP地址:127.0.0.1</span><span style=\"margin-left: 60px\">国家:AAA</span><br>\n" +
        "<span>省份:AAAA</span><span style=\"margin-left: 88px\">位置:XXXXXXX</span></div>";
HtmlParser htmlParser = new HtmlParserImpl();
htmlParser.loadHtml(htmlTemplate);
ImageRenderer imageRenderer = new ImageRendererImpl(htmlParser);
imageRenderer.saveImage("D:\\hello-world.png");

图片添加水印


/**
 * 复制图片文件,并添加水印
 * @param srcImgPath  原图片路径
 * @param outImgPath  生成的新图片路径
 * @param waterMarkContent  水印内容
 * @param markContentColor  水印颜色
 * @param font  水印字体
 */
public static void waterPress(String srcImgPath, String outImgPath, String waterMarkContent, Color markContentColor, Font font) {
   
    try {
   
        // 读取原图片信息
        File srcImgFile = new File(srcImgPath);
        Image srcImg = null;
        if (srcImgFile.exists() && srcImgFile.isFile() && srcImgFile.canRead()) {
   
            srcImg = ImageIO.read(srcImgFile);
        }
        // 宽、高
        int srcImgWidth = srcImg.getWidth(null);
        int srcImgHeight = srcImg.getHeight(null);
        // 加水印
        BufferedImage bufImg = new BufferedImage(srcImgWidth, srcImgHeight, BufferedImage.TYPE_INT_RGB);
        Graphics2D g = bufImg.createGraphics();
        g.drawImage(srcImg, 0, 0, srcImgWidth, srcImgHeight, null);

        //设置水印颜色
        g.setColor(markContentColor);
        g.setFont(font);
        // 抗锯齿
        g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
        int fontLength = getWatermarkLength(waterMarkContent, g);
        // 实际生成的水印文字,实际文字行数
        Double textLineCount = Math.ceil(Integer.valueOf(fontLength).doubleValue() / Integer.valueOf(srcImgWidth).doubleValue());
        int fontSize = font.getSize();
        // 实际所有的水印文字的高度
        int textHeight = textLineCount.intValue() * fontSize;
        // 相对与X的起始的位置
        int originX = 0;
        // 相对与Y的起始的位置
        int originY = 0;
        // 实际文字大于1行,则x则为默认起始0,
        if (1 == textLineCount.intValue()) {
   
            // 实际文字行数是1,1/2个图片高度,减去1/2个字符高度
            originY = srcImgHeight / 2 - fontSize / 2;
            // 实际文字行数是1,计算x的居中的起始位置
            originX = (srcImgWidth - fontLength) / 2;
        } else {
   
            // 实际文字行数大于1,1/2个图片高度减去文字行数所需的高度
            originY = (srcImgHeight - textHeight) / 10;
        }
        System.out.println("水印文字总长度:" + fontLength + ",图片宽度:" + srcImgWidth + ",字符个数:" + waterMarkContent.length());
        //文字叠加,自动换行叠加
        int tempX = originX;
        int tempY = originY;
        int tempCharLen = 0;//单字符长度
        int tempLineLen = 0;//单行字符总长度临时计算
        StringBuffer stringBuffer = new StringBuffer();
        for (int i = 0; i < waterMarkContent.length(); i++) {
   
            char tempChar = waterMarkContent.charAt(i);
            tempCharLen = getCharLen(tempChar, g);
            if (tempLineLen >= srcImgWidth) {
   
                // 绘制前一行
                g.drawString(stringBuffer.toString(), tempX, tempY);
                //清空内容,重新追加
                stringBuffer.delete(0, stringBuffer.length());
                //文字长度已经满一行,Y的位置加1字符高度
                tempY = tempY + fontSize;
                tempLineLen = 0;
            }
            //追加字符
            stringBuffer.append(tempChar);
            tempLineLen += tempCharLen;
        }
        //最后叠加余下的文字
        g.drawString(stringBuffer.toString(), tempX, tempY);
        g.dispose();
        // 输出图片
        FileOutputStream outImgStream = new FileOutputStream(outImgPath);
        ImageIO.write(bufImg, "png", outImgStream);
        outImgStream.flush();
        outImgStream.close();
    } catch (Exception e) {
   
        e.printStackTrace();
    }
}

public static int getCharLen(char c, Graphics2D g) {
   
    return g.getFontMetrics(g.getFont()).charWidth(c);
}
/**
 * 获取水印文字总长度
 *
 * @paramwaterMarkContent水印的文字
 * @paramg
 * @return水印文字总长度
 */
public static int getWatermarkLength(String waterMarkContent, Graphics2D g) {
   
    return g.getFontMetrics(g.getFont()).charsWidth(waterMarkContent.toCharArray(), 0, waterMarkContent.length());
}

//测试复制图片并添加水印
public static void testWaterPressFile(){
   
    // 原图位置, 输出图片位置, 水印字体,水印文字样式,水印文字颜色, 水印文字大小,水印文字内容
    Font font = new Font("微软雅黑",Font.BOLD+ Font.ITALIC, 18);   //水印字体
    Color color = Color.cyan;
    String content = "图片来源:Gaoxs";
    waterPress("D:\\hello-world.png",
            "D:\\hello-world22.png",
            content, color, font);
}
public static void main(String[] args) {
   
   testWaterPressFile();
}
目录
相关文章
|
Go iOS开发 MacOS
手把手教你在Mac上从零搭建Go语言开发环境
手把手教你在Mac上从零搭建Go语言开发环境
5648 0
|
存储 关系型数据库 MySQL
达梦(DM)数据迁移工具
讲述达梦数据库迁移工具使用
|
JavaScript
自行封装的tabs组件配合echarts而出现的bug以及解决
自行封装的tabs组件配合echarts而出现的bug以及解决
584 0
|
Java Maven Windows
添加阿里云maven镜像
添加阿里云maven镜像
|
11月前
|
Python
同步和异步的区别
用Python讲解同异步
|
Java Apache Scala
【阿里云镜像】配置阿里云Maven 镜像
【阿里云镜像】配置阿里云Maven 镜像
25879 1
【阿里云镜像】配置阿里云Maven 镜像
|
机器学习/深度学习 存储
数据结构(九)---并查集
数据结构(九)---并查集
603 5
|
自然语言处理 监控 安全
2025年阿里云短信验证码价格多少钱?计费模式与场景选型指南
随着企业数字化转型,短信验证码作为用户身份验证的重要工具,其成本与效率的平衡至关重要。阿里云短信服务以高可靠性、灵活计费和多场景适配著称。按量付费模式适合需求波动大的场景,而短信套餐包则为长期稳定需求提供了成本优势。针对不同业务场景,如高频验证、跨境业务及中小型企业轻量级需求,阿里云提供了定制化的选型策略。此外,通过阶梯定价、防盗刷监控等措施实现成本优化与风险规避,并不断进行技术升级以确保服务的安全性和稳定性。根据2025年最新数据,企业可根据自身需求选择最适合的阿里云短信验证码服务方案。
GLM-4模型微调报内核版本不匹配的错误
GLM-4模型微调报内核版本不匹配的错误
|
存储 监控 算法
(六)JVM成神路之GC基础篇:对象存活判定算法、GC算法、STW、GC种类详解
经过前面五个章节的分析后,对于JVM的大部分子系统都已阐述完毕,在本文中则开始对JVM的GC子系统进行全面阐述,GC机制也是JVM的重中之重,调优、监控、面试都逃不开的JVM话题。
1214 9