package com.avcon.util;
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import javax.imageio.ImageIO;
import com.swetake.util.Qrcode;
public class RQCodeUtils {
public static int max_data_size_small = 84;
public static int max_data_size_large = 500;
/**
*
* @param srcValue
* @param qrcodePicfilePath
* @return
*/
public static boolean encode(String srcValue, String qrcodePicfilePath) {
return encode_84(srcValue, qrcodePicfilePath);
}
/**
* Encoding the information to a QRCode, size of the information must be less than 84 byte.
*
* @param srcValue
* @param qrcodePicfilePath
* @return
*/
public static boolean encode_84(String srcValue, String qrcodePicfilePath) {
int MAX_DATA_LENGTH = max_data_size_small; // 限制生成二维码的数据最大大小
byte[] d = srcValue.getBytes();
int dataLength = d.length;
int imageWidth = 113; /* 113是预先计算出来的. 注意:图像宽度必须比生成的二维码图像宽度大,至少相等,否则,二维码识别不出来 */
int imageHeight = imageWidth;
BufferedImage bi = new BufferedImage(imageWidth, imageHeight, BufferedImage.TYPE_INT_RGB);
Graphics2D g = bi.createGraphics();
g.setBackground(Color.WHITE);
g.clearRect(0, 0, imageWidth, imageHeight);
g.setColor(Color.BLACK);
if (dataLength > 0 && dataLength <= MAX_DATA_LENGTH) {
/* 生成二维码 */
Qrcode qrcode = new Qrcode();
qrcode.setQrcodeErrorCorrect('M'); // L, Q, H, 默认
qrcode.setQrcodeEncodeMode('B'); // A, N, 默认
qrcode.setQrcodeVersion(5); // 37字节, (37-1)*3+2+3-1+1 = 113
boolean[][] b = qrcode.calQrcode(d);
int qrcodeDataLen = b.length;
for (int i = 0; i < qrcodeDataLen; i++) {
for (int j = 0; j < qrcodeDataLen; j++) {
if (b[j][i]) {
g.fillRect(j * 3 + 2, i * 3 + 2, 3, 3); /*
* 画二维码图形, 画出的图形宽度是 ((qrcodeDataLen-1)*3+2) + 3 -1 ; 生成的image的宽度大小必须>=该值,外围的1个像素用来标识此块区域为二维码
*/
/*
* fillRect(int x, int y, int width, int height) 函数作用: 填充指定的矩形。该矩形左边和右边位于 x 和 x + width - 1。顶边和底边位于 y 和 y + height - 1。 得到的矩形覆盖的区域宽度为 width 像素,高度为 height 像素。 使用图形上下文的当前颜色填充该矩形。 参数: x - 要填充矩形的 x 坐标。 y - 要填充矩形的 y 坐标。 width - 要填充矩形的宽度。 height - 要填充矩形的高度。
*
* 参考:http://bk.chinaar.com/index.php?doc-view-2999
*/
}
}
}
System.out.println("二维码数据长度(字节):" + qrcodeDataLen);
} else {
System.out.println("Generate QRCode image error! Data size is " + dataLength + ", it is lager than 84 bytes.");
return false;
}
g.dispose();
bi.flush();
/* generate image */
File f = new File(qrcodePicfilePath);
if (f.exists()) {
return true;
}
System.out.println(qrcodePicfilePath);
String suffix = f.getName().substring(f.getName().indexOf(".") + 1, f.getName().length());
try {
ImageIO.write(bi, suffix, f); // "png"
} catch (IOException ioe) {
System.out.println("Generate QRCode image error!" + ioe.getMessage());
return false;
}
return true;
}
/**
* Encoding the information to a QRCode, size of the information must be less tah 500 byte.
*
* @param srcValue
* @param qrcodePicfilePath
* @return
*/
public static boolean encode_500(String srcValue, String qrcodePicfilePath) {
int MAX_DATA_LENGTH = max_data_size_large; // 限制生成二维码的数据最大大小. 500字节的原始数据, 生成二维码时, 是89宽度
byte[] d = srcValue.getBytes();
int dataLength = d.length;
int imageWidth = 269; /* 269是预先计算出来的. 注意:图像宽度必须比生成的二维码图像宽度大,至少相等,否则,二维码识别不出来 */
int imageHeight = imageWidth;
BufferedImage bi = new BufferedImage(imageWidth, imageHeight, BufferedImage.TYPE_INT_RGB);
Graphics2D g = bi.createGraphics();
g.setBackground(Color.WHITE);
g.clearRect(0, 0, imageWidth, imageHeight);
g.setColor(Color.BLACK);
if (dataLength > 0 && dataLength <= MAX_DATA_LENGTH) {
/* 生成二维码 */
Qrcode qrcode = new Qrcode();
qrcode.setQrcodeErrorCorrect('M'); // L, Q, H, 默认
qrcode.setQrcodeEncodeMode('B'); // A, N, 默认
qrcode.setQrcodeVersion(18); // 0<= version <=40; 89字节,
// (89-1)*3+2+3-1+1 = 269
boolean[][] b = qrcode.calQrcode(d);
int qrcodeDataLen = b.length;
for (int i = 0; i < qrcodeDataLen; i++) {
for (int j = 0; j < qrcodeDataLen; j++) {
if (b[j][i]) {
g.fillRect(j * 3 + 2, i * 3 + 2, 3, 3); /*
* 画二维码图形, 画出的图形宽度是 ((qrcodeDataLen-1)*3+2) + 3 -1 = 136; 生成的image的宽度大小必须>=(136+1),外围的1个像素用来标识此块区域为二维码
*/
/*
* fillRect(int x, int y, int width, int height) 函数作用: 填充指定的矩形。该矩形左边和右边位于 x 和 x + width - 1。顶边和底边位于 y 和 y + height - 1。 得到的矩形覆盖的区域宽度为 width 像素,高度为 height 像素。 使用图形上下文的当前颜色填充该矩形。 参数: x - 要填充矩形的 x 坐标。 y - 要填充矩形的 y 坐标。 width - 要填充矩形的宽度。 height - 要填充矩形的高度。
*
* 参考:http://bk.chinaar.com/index.php?doc-view-2999
*/
}
}
}
System.out.println("二维码数据长度(字节):" + qrcodeDataLen);
} else {
return false;
}
g.dispose();
bi.flush();
/* generate image */
File f = new File(qrcodePicfilePath);
String suffix = f.getName().substring(f.getName().indexOf(".") + 1, f.getName().length());
System.out.println(suffix);
try {
ImageIO.write(bi, suffix, f); // "png"
} catch (IOException ioe) {
System.out.println("Generate QRCode image error!" + ioe.getMessage());
return false;
}
return true;
}
}
package com.avcon.util;
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.geom.AffineTransform;
import java.awt.image.AffineTransformOp;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import javax.imageio.ImageIO;
import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType;
import com.google.zxing.MultiFormatWriter;
import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;
public class RQCodeImageUtils {
// 图片宽度的一般
private static final int IMAGE_WIDTH = 50;
private static final int IMAGE_HEIGHT = 50;
private static final int IMAGE_HALF_WIDTH = IMAGE_WIDTH / 2;
private static final int FRAME_WIDTH = 2;
// 二维码写码器
private static MultiFormatWriter mutiWriter = new MultiFormatWriter();
public static void encode(String content, int width, int height, String srcImagePath, String destImagePath) {
try {
ImageIO.write(genBarcode(content, width, height, srcImagePath), "jpg", new File(destImagePath));
} catch (IOException e) {
e.printStackTrace();
} catch (WriterException e) {
e.printStackTrace();
}
}
private static BufferedImage genBarcode(String content, int width, int height, String srcImagePath) throws WriterException, IOException {
// 读取源图像
BufferedImage scaleImage = scale(srcImagePath, IMAGE_WIDTH, IMAGE_HEIGHT, true);
int[][] srcPixels = new int[IMAGE_WIDTH][IMAGE_HEIGHT];
for (int i = 0; i < scaleImage.getWidth(); i++) {
for (int j = 0; j < scaleImage.getHeight(); j++) {
srcPixels[i][j] = scaleImage.getRGB(i, j);
}
}
Map<EncodeHintType, Object> hint = new HashMap<EncodeHintType, Object>();
hint.put(EncodeHintType.CHARACTER_SET, "utf-8");
hint.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);
// 生成二维码
BitMatrix matrix = mutiWriter.encode(content, BarcodeFormat.QR_CODE, width, height, hint);
// 二维矩阵转为一维像素数组
int halfW = matrix.getWidth() / 2;
int halfH = matrix.getHeight() / 2;
int[] pixels = new int[width * height];
for (int y = 0; y < matrix.getHeight(); y++) {
for (int x = 0; x < matrix.getWidth(); x++) {
// 读取图片
if (x > halfW - IMAGE_HALF_WIDTH && x < halfW + IMAGE_HALF_WIDTH && y > halfH - IMAGE_HALF_WIDTH && y < halfH + IMAGE_HALF_WIDTH) {
pixels[y * width + x] = srcPixels[x - halfW + IMAGE_HALF_WIDTH][y - halfH + IMAGE_HALF_WIDTH];
}
// 在图片四周形成边框
else if ((x > halfW - IMAGE_HALF_WIDTH - FRAME_WIDTH && x < halfW - IMAGE_HALF_WIDTH + FRAME_WIDTH && y > halfH - IMAGE_HALF_WIDTH - FRAME_WIDTH && y < halfH + IMAGE_HALF_WIDTH + FRAME_WIDTH) || (x > halfW + IMAGE_HALF_WIDTH - FRAME_WIDTH && x < halfW + IMAGE_HALF_WIDTH + FRAME_WIDTH && y > halfH - IMAGE_HALF_WIDTH - FRAME_WIDTH && y < halfH + IMAGE_HALF_WIDTH + FRAME_WIDTH) || (x > halfW - IMAGE_HALF_WIDTH - FRAME_WIDTH && x < halfW + IMAGE_HALF_WIDTH + FRAME_WIDTH && y > halfH - IMAGE_HALF_WIDTH - FRAME_WIDTH && y < halfH - IMAGE_HALF_WIDTH + FRAME_WIDTH) || (x > halfW - IMAGE_HALF_WIDTH - FRAME_WIDTH && x < halfW + IMAGE_HALF_WIDTH + FRAME_WIDTH && y > halfH + IMAGE_HALF_WIDTH - FRAME_WIDTH && y < halfH + IMAGE_HALF_WIDTH + FRAME_WIDTH)) {
pixels[y * width + x] = 0xfffffff;
} else {
// 此处可以修改二维码的颜色,可以分别制定二维码和背景的颜色;
pixels[y * width + x] = matrix.get(x, y) ? 0xff000000 : 0xfffffff;
}
}
}
BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
image.getRaster().setDataElements(0, 0, width, height, pixels);
return image;
}
/**
* 把传入的原始图像按高度和宽度进行缩放,生成符合要求的图标
*
* @param srcImageFile
* 源文件地址
* @param height
* 目标高度
* @param width
* 目标宽度
* @param hasFiller
* 比例不对时是否需要补白:true为补白; false为不补白;
* @throws IOException
*/
private static BufferedImage scale(String srcImageFile, int height, int width, boolean hasFiller) throws IOException {
double ratio = 0.0; // 缩放比例
File file = new File(srcImageFile);
BufferedImage srcImage = ImageIO.read(file);
Image destImage = srcImage.getScaledInstance(width, height, BufferedImage.SCALE_SMOOTH);
// 计算比例
if ((srcImage.getHeight() > height) || (srcImage.getWidth() > width)) {
if (srcImage.getHeight() > srcImage.getWidth()) {
ratio = (new Integer(height)).doubleValue() / srcImage.getHeight();
} else {
ratio = (new Integer(width)).doubleValue() / srcImage.getWidth();
}
AffineTransformOp op = new AffineTransformOp(AffineTransform.getScaleInstance(ratio, ratio), null);
destImage = op.filter(srcImage, null);
}
if (hasFiller) {// 补白
BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
Graphics2D graphic = image.createGraphics();
graphic.setColor(Color.white);
graphic.fillRect(0, 0, width, height);
if (width == destImage.getWidth(null))
graphic.drawImage(destImage, 0, (height - destImage.getHeight(null)) / 2, destImage.getWidth(null), destImage.getHeight(null), Color.white, null);
else
graphic.drawImage(destImage, (width - destImage.getWidth(null)) / 2, 0, destImage.getWidth(null), destImage.getHeight(null), Color.white, null);
graphic.dispose();
destImage = image;
}
return (BufferedImage) destImage;
}
}