需求:


是在做证书的时候碰到的这个问题。
当时需求是能够进行在线打印证书,第一次进行的操作是直接打印html,并且已经排好版(用jqprint插件)进行打印。在打印时碰到了兼容的问题,另外因为背景图片载入较文字载入的慢。则出现先载入文字后出现图片的现象,则显的非常不专业。遂採用了将信息转化为图片。然后直接打印图片,不论怎么样进行打印,都不会改变,并且能够导出图片,对图片进行更为精细的操作。

代码编写:


先不多说 先上代码:
package com.jzba.utils;

import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.RenderingHints;
import java.awt.Transparency;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Map;
import java.util.Random; import javax.imageio.ImageIO;
import javax.swing.ImageIcon; import com.sun.image.codec.jpeg.ImageFormatException;
import com.sun.image.codec.jpeg.JPEGCodec;
import com.sun.image.codec.jpeg.JPEGImageEncoder;
public class ImgUtils { private static int newWidth=0,newHeight=0,newX=0,newY=0; public static void main(String[] args) throws Exception{--1、本地測试
String over="D:/zz/1/over.png";
Color color = new Color(100,100,100);
ImageUtil iu=new ImageUtil();
Map map=iu.getImageSizeByBufferedImage("D:/zz/ceshi.png");--备注:见代码下方
int width=(Integer)map.get("width");
int height=(Integer)map.get("height");
float bili=width/height;
if (width>height) {
newWidth=660;
newHeight=height*660/width;
newX=600;
newY=1160+330-newHeight/2;
}else{
newHeight=660;
newWidth=width*660/height;
newY=1160;
newX=600+330-newWidth/2;
}
//String temp1=createCard("D:/zz/1/background.png","D:/zz/1/ceshi.png","D:/zz/1/code.bmp","D:/zz/1/temp1.png",newX,newY,600,1210,newWidth,newHeight,250,250);
String temp1=createCard("D:/zz/1/background.png","D:/zz/1/ceshi2.jpg","D:/zz/1/code.bmp","D:/zz/1/temp1.png",100,100,600,1210,newWidth,newHeight,250,250);
ImageIcon imgIcon = new ImageIcon(temp1);
Image theImg = imgIcon.getImage();
int w = theImg.getWidth(null);
int h = theImg.getHeight(null);
BufferedImage bufImg = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);
// 获取Graphics2D
Graphics2D g = bufImg.createGraphics();
g.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
// 绘图
bufImg = g.getDeviceConfiguration().createCompatibleImage(w, h, Transparency.TRANSLUCENT);
g.dispose();
g = bufImg.createGraphics();
g.setStroke(new BasicStroke(1));
g.drawImage(theImg, 0, 0, w, h, null);//背景图起始位置
g.setColor(color);
Font nf = loadFont("D:/zz/simhei.ttf",60);
Font nf2 = loadFont("D:/zz/simhei.ttf",50);
g.setFont(nf);
g.drawString("秘 密",2500,1040);//
g.setFont(nf2);
g.drawString("秘 密",1920,1190);//
g.drawString("秘 密", 1920, 1290);//
g.drawString("秘 密", 1920, 1390);//
g.drawString("秘 密", 1920, 1490);//
g.drawString("秘 密 ", 1920, 1590);//
g.drawString("秘 密", 1920, 1700);//
g.drawString("秘 密", 1920, 1800);//
//释放对象
File tofile=new File(over);
ImageIO.write(bufImg, "png", tofile);
g.dispose();
}
/**
*
* 将所要加入的图片整合在一起进行加入
* @param backsPath 背景图片的路径
* @param titleImgPath 缩略图图片的路径
* ...
*
* @return toPath 合成后图片的路径
*
* */ public static String createCard(String backsPath,String titleImgPath,String codeImgPath,String toPath,int x,int y,int newx,int newy,int widthTitle,int heightTitle,int codeWidth,int codeHeight){
try {
InputStream imageBack=new FileInputStream(backsPath);//背景图
InputStream imageTitle=new FileInputStream(titleImgPath);//缩略图图片
InputStream imageCode=new FileInputStream(codeImgPath);//二维码图片
BufferedImage backImg=ImageIO.read(imageBack);
BufferedImage titleImg=ImageIO.read(imageTitle);
BufferedImage codeImg=ImageIO.read(imageCode);
System.out.println("*****Graphics****"+backsPath);
Graphics g=backImg.getGraphics();
System.out.println("*****Graphics****1");
g.drawImage(titleImg,x,y,widthTitle,heightTitle,null);
System.out.println("*****Graphics****2");
g.drawImage(codeImg,newx,newy,codeWidth,codeHeight,null);
System.out.println("*****Graphics****3");
OutputStream outImage=new FileOutputStream(toPath);
System.out.println("*****Graphics****4");
// JPEGImageEncoder enc=JPEGCodec.createJPEGEncoder(outImage);--这里存在问题
// System.out.println("*****Graphics****5");--在window下能够。在linux下无法进行
// enc.encode(backImg);
File tofile=new File(toPath);
ImageIO.write(backImg, "png", tofile);
// System.out.println("*****"+backsPath);
imageBack.close();//关闭输出端口
imageTitle.close();
imageCode.close();
outImage.close();
} catch (Exception e) {
e.printStackTrace();
}
return toPath;
}
/**
* 将所要加入的文本信息整合在一起进行加入
*
*
* **/
public static String addCardWord(String filePath,String toPath,int fontsize, Color markContentColor,String fontPath,String code,String type,String artist,
String artname,String description,String material,String artisttype,String recordescription,String releaseDate,int codex,int codey,int typex,int typey,
int artistx,int artisty,int artnamex,int artnamey,int descriptionx,int descriptiony,int materialx,int materialy,int artisttypex,int artisttypey,
int recordescriptionx,int recordescriptiony)throws Exception{
ImageIcon imgIcon = new ImageIcon(filePath);
Image theImg = imgIcon.getImage();
int w = theImg.getWidth(null);
int h = theImg.getHeight(null);
BufferedImage bufImg = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);
// 获取Graphics2D
Graphics2D g = bufImg.createGraphics();
g.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
// 绘图
bufImg = g.getDeviceConfiguration().createCompatibleImage(w, h, Transparency.TRANSLUCENT);
g.dispose();
g = bufImg.createGraphics();
g.setStroke(new BasicStroke(1));
g.drawImage(theImg, 0, 0, w, h, null);//背景图起始位置
g.setColor(markContentColor);
Font nf = loadFont(fontPath,fontsize);
g.setFont(nf);
g.drawString(code, codex, codey);
g.drawString(type, typex, typey);
g.drawString(artist, artistx, artisty);
g.drawString(artname, artnamex, artnamey);
g.drawString(description, descriptionx, descriptiony);
g.drawString(material, materialx, materialy);
g.drawString(artisttype, artisttypex, artisttypey);
g.drawString(recordescription, recordescriptionx, recordescriptiony);
//释放对象
File tofile=new File(toPath);
ImageIO.write(bufImg, "png", tofile);
g.dispose();
return toPath;
} /**
* 向图片里面加入图片
* @param backsPath 背景图片路径
* @param upPath 加入图片路径
* @param toPath 生成图片路径
* @param x 所加入图片在背景图片上的横向位移,以左上角为标准
* @param y 所加入图片在背景图片上的竖向位移,以左上角为标准
* @param width 所加入图片的宽度
* @param height 所加入图片的高度
* @return toPath String 返回生成图片的路径
*
* **/
public static String addImg2Img(String backsPath,String upPath,String toPath,int x,int y,int width,int height){
try {
InputStream imagein=new FileInputStream(backsPath);
InputStream imagein2=new FileInputStream(upPath);
BufferedImage image=ImageIO.read(imagein);//读取背景图片
BufferedImage image2=ImageIO.read(imagein2);//读取要加入的图片
Graphics g=image.getGraphics();//将背景图片作为绘图面板
g.drawImage(image2,x,y,width,height,null);//在背景图片的基础上进行绘图,将待加入图片加入
OutputStream outImage=new FileOutputStream(toPath);//打开导出图片路径
JPEGImageEncoder enc=JPEGCodec.createJPEGEncoder(outImage);
enc.encode(image);//将生成的图片输出
imagein.close();//关闭打开的导入、导出通道
imagein2.close();
outImage.close();
} catch (Exception e) {
e.printStackTrace();
}
return toPath;
}
/**
* 向图片里面加入文字
* @param filePath 背景图片路径
* @param toPath 生成图片路径
* @param markContent 加入的文本内容
* @param fontsize 加入文本的字体大小
* @param markContentColor 加入文本的颜色
* @param fontPath 加入文本的字体
* @param x 所加入图片在背景图片上的横向位移。以左上角为标准
* @param y 所加入图片在背景图片上的竖向位移,以左上角为标准
* @return toPath String 返回生成图片的路径
*
* **/
public static String addWord2Img(String filePath,String toPath, String markContent,int fontsize, Color markContentColor,String fontPath,int x,int y) throws Exception {
ImageIcon imgIcon = new ImageIcon(filePath);
Image theImg = imgIcon.getImage();
int w = theImg.getWidth(null);
int h = theImg.getHeight(null);
BufferedImage bufImg = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);
// 获取Graphics2D
Graphics2D g = bufImg.createGraphics();
g.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
// 绘图
bufImg = g.getDeviceConfiguration().createCompatibleImage(w, h, Transparency.TRANSLUCENT);
g.dispose();
g = bufImg.createGraphics();
g.setStroke(new BasicStroke(1));
g.drawImage(theImg, 0, 0, w, h, null);//背景图起始位置
g.setColor(markContentColor);//给加入的文本进行颜色设置
Font nf = loadFont(fontPath,fontsize);
g.setFont(nf);//对加入的文本进行字体和字体大小设置
g.drawString(markContent, x, y);//将文本加入到图片指定的位置
//释放对象
File tofile=new File(toPath);//打开导出通道
ImageIO.write(bufImg, "png", tofile);//进行绘图
g.dispose();
return toPath;
}
/**
* 对加入文本的字体和字体大小进行设置
* @param fontFileName 外部字体名
* @param fontsize 加入文本的字体大小
* @param markContentColor 加入文本的颜色
* @return dynamicFontPt Font 对文本的对应设置
*
* **/
public static Font loadFont(String fontFileName, float fontSize) throws Exception //第一个參数是外部字体名,第二个是字体大小
{
File file = new File(fontFileName);
FileInputStream aixing = new FileInputStream(file);
Font dynamicFont = Font.createFont(Font.TRUETYPE_FONT, aixing);
Font dynamicFontPt = dynamicFont.deriveFont(fontSize);
aixing.close();
return dynamicFontPt;
} }

备注:在代码中调用到了上一篇文章中获取图片尺寸的对应类。用在这里的目的是。针对不同的作品,可能是横向图片。也可能是竖条图片,为了保证图片不会变形失真。则将图片显示在固定的一个区域内,像这里:


默认了一段660*660的区域,首先进行判定要加入图片是横条还是竖条,假设是横条,则宽度是大于高度的,则设置大的宽度为660,对应的依据比例:660/宽度=要求的高度/实际高度。则能够得到对应的高度值。并且此高度值肯定是小于660的,由于是横条。所以宽度的位移就是指定区域的对应位移,这里为600px,而高度位移,则应该是加入图片的左上角的位移,为背景图片的高度一半减去图片当前高度的一半;同理假设是竖条,则高度是大于宽度的,则设置大的高度为660。以此类推。得到对应的位移和尺寸。从而实现的效果是什么样子呢,我们来简单看下:
(这是高度大于宽度)
(这是宽度大于高度)

积跬步,聚小流------java信息生成图片的更多相关文章

  1. 积跬步,聚小流------java获取图片的尺寸

    在一篇文章中获取到通过例如以下两种方式进行获取: 1.使用ImageReader进行获取: 2.使用BufferedImage进行获取: 而且经过验证ImageReader进行操作的耗时远远低于Buf ...

  2. 积跬步,聚小流------关于UML类图

    UML的存在 类图是使用频率比較高的UML图,它用于描写叙述系统中所含的类以及它们之间的相互关系,帮助人们简化对系统的理解,也是系统分析和设计阶段的重要产物,也是系统编码和測试的重要类型根据. UML ...

  3. 积跬步,聚小流------Bootstrap学习记录(3)

    响应式作为Bootstrap的一大特色.栅格系统可谓是功不可没,既然如此,那我们就来看一下栅格系统是怎样帮助bootstrap实现响应式布局的呢? 1.什么是栅格系统 我们能够从Bootstrap的官 ...

  4. 积跬步,聚小流------Bootstrap学习记录(2)

    现阶段开启每一次新的征程,已然离不开"Hello World"的习惯仪式.这次自然也不例外.先来看下给出的官网给出的演示样例: 1.bootstrap官网提供的html基本模板代码 ...

  5. 积跬步,聚小流-------js实现placeholder的效果

    前几天在"技术问答"上问了问题,然后有回复一句话就给概括了:placeholder的效果,不得不说.了解的多了才干说起来言简意赅,用最简单的语言描写叙述最清晰的表达. 可是plac ...

  6. 积跬步,聚小流------ps有用小技巧,改变png图标颜色

    *  实现效果: 原图:  改动后: *  实现目的: 满足为实现不同界面色彩搭配改动png图标的颜色 *  实现方法: 1.打开Photoshop工具,导入须要进行改动的png图标: 2.对导入的图 ...

  7. 积跬步,聚小流------界面经常使用的jeecms标签

    * JEECMS初印象 第一次接触JEECMS,突然脑海就浮现了一句话"20元建站,立等可取",原来这都是真的... * JEECMS的界面经常使用标签 临时忽略掉环境搭建.栏目配 ...

  8. 积跬步,聚小流------Bootstrap学习记录(1)

    watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/ ...

  9. qingshow “不积跬步无以至千里,不积小流无以成江海”。--荀子《劝学篇》 用tomcat+花生壳搭建自己的web服务器+域名(参考)

    链接地址:http://www.blogjava.net/qingshow/archive/2010/01/17/309846.html 用tomcat搭建web服务器 目标:免费拥有自己的网站及域名 ...

随机推荐

  1. javascript设计模式-掺元类

    有一种重用代码的方法不需要用到严格的继承.如果想把一个函数用到多个类中,可以通过扩充的方式让这些类共享该函数.其实际做法大大体为:先创建一个包含各种通用方法的类,然后再用它扩充其他的类.这种方式就叫做 ...

  2. Mysql外键的变种 三种关系

    一.介绍 因为有foreign key的约束,使得两张表形成了三种了关系: 多对一 多对多 一对一 二.重点理解如果找出两张表之间的关系 分析步骤: #1.先站在左表的角度去找 是否左表的多条记录可以 ...

  3. (转)(C++)关于抽象基类和纯虚函数

    ★抽象类:一个类可以抽象出不同的对象来表达一个抽象的概念和通用的接口,这个类不能实例化(创造)对象. ★纯虚函数(pure virtual):在本类里不能有实现(描述功能),实现需要在子类中实现.例: ...

  4. 第十课: - 读取/写入Excel/Json格式数据

    第 10 课 从DataFrame到Excel 从Excel到DataFrame 从DataFrame到JSON 从JSON到DataFrame In [1]: import pandas as pd ...

  5. iOS中容易混淆的知识点(持续更新中)

    1.成员变量和属性的区别 @interface Person : NSObject { NSString *_sex; } @property (nonatomic, copy) NSString * ...

  6. C#中跨线程访问控件

    net 原则上禁止跨线程访问控件,因为这样可能造成错误的发生,推荐的解决方法是采用代理,用代理方法来间接操作不是同一线程创建的控件. 第二种方法是禁止编译器对跨线程访问作检查,可以实现访问,但是出不出 ...

  7. 将电脑特定文件夹保存在U盘中

    为什么 各种网盘,借着国家扫黄的阶梯,纷纷取消自己的网盘的服务.但自己有一些不是很大,但又很重要的东西,比如说代码(虽然学的渣) 怎么做 再网上百度,有一些将U盘的文件偷偷拷到电脑的脚本,改一下复制文 ...

  8. Lazy Stored Properties--无括号时为匿名函数

    第一次使用的时候进行计算和初始化,后面的引用不在进行计算. A lazy stored property is a property whose initial value is not calcul ...

  9. C#的split函数分割

    C#的split函数分割 string str = textBox1.Text; string[] strlist = str.Split("\r\n".ToCharArray() ...

  10. 优动漫PAINT(clip studio paint)怎么画一幅水墨竹子图

    今天小编分享使用优动漫PAINT绘制一个水墨竹子教程,绘画的过程中我只用到了两个笔刷,即钢笔模式下的“美术字”和“效果线专用”,并且全程鼠标绘制哦,所以生疏的笔触效果大家见谅,没有数位板的小伙伴不妨试 ...