生成微信二維碼

package com.common.utils;

import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.net.URLEncoder;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Arrays;
import java.util.Date;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Map;

import javax.imageio.ImageIO;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;

import org.apache.commons.codec.binary.Base64;
import org.apache.commons.lang3.StringUtils;
import org.junit.Assert;

import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType;
import com.google.zxing.MultiFormatWriter;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;


/**
 * @author EX-CHENKEFENG001
 * 
 *         微信驗證、二維碼生成工具類
 * 
 */
public class WechatValidateUtils {
	private static final boolean NEEDCOMPRESS = false;
	
	private static String LOGO_URL;
    // LOGO寬度
    private static final int WIDTH = 60;
    // LOGO高度
    private static final int HEIGHT = 60;
    
    static{
    	 LOGO_URL = PropertiesUtils.getProperty("context-ilife_core.properties", "pemobi.dmz.logo.url");
    }
    
	/**
	 * 生成二維碼的工具類
	 * 
	 * @param data
	 *            二維碼中攜帶的數據
	 * @param width
	 *            二維碼的寬度
	 * @param height
	 *            二維碼的高度
	 * @return 返回base64格式的圖片() <br>
	 *         eg: <code>
	 * &ltimg alt="" src=""&gt
	 * </code>
	 */
	public static String generateQRCode(String data, int width, int height,String isPicture) {

		Assert.assertTrue("param data cannot empty.", data != null
				&& data.trim().length() > 0);

		Assert.assertTrue("param width and height must gt 0.", width > 0
				&& height > 0);

		ByteArrayOutputStream bos = null;

		MultiFormatWriter formatWriter = new MultiFormatWriter();

		Hashtable<EncodeHintType, Object> param = new Hashtable<EncodeHintType, Object>();
		param.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.Q);
		param.put(EncodeHintType.CHARACTER_SET, "utf-8");
		param.put(EncodeHintType.MARGIN, 0);

		try {
			BitMatrix bitMatrix = formatWriter.encode(data,
					BarcodeFormat.QR_CODE, width, height, param);
			//1.1去白邊
			int[] rec = bitMatrix.getEnclosingRectangle();  
			int resWidth = rec[2] + 1;  
			int resHeight = rec[3] + 1;  
			BitMatrix resMatrix = new BitMatrix(resWidth, resHeight);  
			resMatrix.clear();  
			for (int i = 0; i < resWidth; i++) {  
			    for (int j = 0; j < resHeight; j++) {  
			        if (bitMatrix.get(i + rec[0], j + rec[1])) { 
			             resMatrix.set(i, j); 
			        } 
			    }  
			}  
			int width1 = resMatrix.getWidth();
			int height1 = resMatrix.getHeight();
			BufferedImage qrcode = new BufferedImage(width1, height1,
					BufferedImage.TYPE_INT_RGB);
			for (int x = 0; x < width1; x++) {
			    for (int y = 0; y < height1; y++) {
			    	qrcode.setRGB(x, y, resMatrix.get(x, y) == true ? 
			        Color.BLACK.getRGB():Color.WHITE.getRGB());
			    }
			}
				if(CommonUtil.isNotNull(LOGO_URL) && CommonUtil.isNotNull(isPicture)){
					insertImage(width1,qrcode,LOGO_URL,NEEDCOMPRESS);
				}
				Logger.INFO("LOGO_URL------------------------"+LOGO_URL);
			bos = new ByteArrayOutputStream();
			ImageIO.write(qrcode, "png", bos);
			
			String img = StringUtils.deleteWhitespace(Base64
					.encodeBase64String(bos.toByteArray()));
			return String.format("data:image/png;base64,%s", img);
		} catch (Exception e) {
			throw new RuntimeException(e);
		} finally {
			if (bos != null) {
				try {
					bos.close();
				} catch (IOException e) {
				}
			}
		}

	}
    private static void insertImage(int QRCODE_SIZE,BufferedImage source, String imgPath,
            boolean needCompress) throws Exception {
        File file = new File(imgPath);
        if (!file.exists()) {
            System.err.println(""+imgPath+"   該文件不存在!");
            return;
        }
        Image src = ImageIO.read(new File(imgPath));
        int width = src.getWidth(null);
        int height = src.getHeight(null);
        if (needCompress) { // 壓縮LOGO
            if (width > WIDTH) {
                width = WIDTH;
            }
            if (height > HEIGHT) {
                height = HEIGHT;
            }
            Image image = src.getScaledInstance(width, height,
                    Image.SCALE_SMOOTH);
            BufferedImage tag = new BufferedImage(width, height,
                    BufferedImage.TYPE_INT_RGB);
            Graphics g = tag.getGraphics();
            g.drawImage(image, 0, 0, null); // 繪製縮小後的圖
            g.dispose();
            src = image;
        }
        // 插入LOGO
        Graphics2D graph = source.createGraphics();
        int x = (QRCODE_SIZE - width) / 2;
        int y = (QRCODE_SIZE - height) / 2;
        graph.drawImage(src, x, y, width, height, null);
       // Shape shape = new RoundRectangle2D.Float(x, y, width, width, 6, 6);
        graph.setStroke(new BasicStroke(3f));
        //graph.draw(shape); 去除圖片白邊
        graph.dispose();
    }

   
	/**
	 * @param data
	 *            二維碼攜帶的數據
	 * @param size
	 *            二維碼的寬、高
	 * @return
	 * @see WechatValidateUtils#generateQRCode(String, int, int)
	 */
	public static String generateQRCode(String data, int size,String isPicture) {
		return generateQRCode(data, size, size,isPicture);
	}

	/**
	 * 校驗姓名是否合法
	 * 
	 * @param name
	 * @return 驗證經過true,不然false
	 */
	public static boolean validateName(String name) {

		return StringUtils.isNotBlank(name);
	}

	/**
	 * 身份證號碼校驗
	 * 
	 * @param idNo
	 * @return 驗證經過 true,不然 false
	 */
	public static boolean validateIdNo(String idNo) {
		if (StringUtils.isBlank(idNo)) {
			return false;
		}

		String reg = "^(?:\\d{8}(?:0[1-9]|1[0-2])[0123]\\d{4}|\\d{6}(?:18|19|20)\\d{2}(?:0[1-9]|1[0-2])[0123]\\d{4}[0-9X])?$";

		return idNo.matches(reg);
	}

	/**
	 * 校驗手機號碼是否合法
	 * 
	 * @param mobile
	 * @return 校驗經過true,不然false
	 */
	public static boolean validateMobile(String mobile) {
		if (StringUtils.isBlank(mobile)) {
			return false;
		}
		String reg = "^(13|14|15|17|18)[0-9]{9}$";

		return mobile.matches(reg);
	}

	/**
	 * 根據身份證號碼獲取 性別,年齡
	 * 
	 * @param idNo
	 *            身份證號碼
	 * @return
	 * 
	 *         <code>
	 * sex: 'M|F',
	 * birth: Date,
	 * birthStr: String(yyyy-MM-dd),
	 * age: int
	 * </code>
	 * 
	 */
	public static Map<String, Object> analyzeIdNo(String idNo, String format) {
		if (StringUtils.isBlank(idNo)) {
			return null;
		}
		Map<String, Object> result = new HashMap<String, Object>();
		if (StringUtils.isBlank(format)) {
			format = "yyyy-MM-dd";
		}
		DateFormat df = new SimpleDateFormat(format);
		DateFormat parser = new SimpleDateFormat("yyyyMMdd");

		String birthStr;
		String sex;

		switch (idNo.length()) {
		case 18:
			birthStr = idNo.substring(6, 14);
			sex = (Integer.valueOf(idNo.substring(16, 17)) % 2) == 0 ? "F"
					: "M";
			break;
		case 15:
			birthStr = String.format("19%s", idNo.substring(6, 12));
			sex = (Integer.valueOf(idNo.substring(14, 15)) % 2) == 0 ? "F"
					: "M";
			break;
		default:
			return null;
		}
		try {
			Date birth = parser.parse(birthStr);
			result.put("sex", sex);
			result.put("birth", birth);
			result.put("birthStr", df.format(birth));
			result.put("age", DateUtil.calculateAgeInBirth(birth, new Date()));

			return result;
		} catch (Exception e) {
			throw new RuntimeException(e);
		}
	}

	public static String getSHA1(String openid, String timestamp, String nonce,
			String key) {
		String[] arr = new String[] { key, timestamp, nonce, openid };
		Arrays.sort(arr);
		StringBuilder content = new StringBuilder();
		for (int i = 0; i < arr.length; i++) {
			content.append(arr[i]);
		}
		MessageDigest md = null;
		String tmpStr = null;

		try {
			md = MessageDigest.getInstance("SHA-1");
			// 將三個參數字符串拼接成一個字符串進行sha1加密
			byte[] digest = md.digest(content.toString().getBytes());
			tmpStr = byteToStr(digest);
		} catch (NoSuchAlgorithmException e) {
			e.printStackTrace();
		}

		content = null;
		// 將sha1加密後的字符串可與signature對比,標識該請求來源於微信
		if(null == tmpStr) {
			return null;
		}
		return tmpStr.toLowerCase();

	}

	/**
	 * 將字節數組轉換爲十六進制字符串
	 * 
	 * @param byteArray
	 * @return
	 */
	private static String byteToStr(byte[] byteArray) {
		String strDigest = "";
		for (int i = 0; i < byteArray.length; i++) {
			strDigest += byteToHexStr(byteArray[i]);
		}
		return strDigest;
	}

	/**
	 * 將字節轉換爲十六進制字符串
	 * 
	 * @param mByte
	 * @return
	 */
	private static String byteToHexStr(byte mByte) {
		char[] Digit = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A',
				'B', 'C', 'D', 'E', 'F' };
		char[] tempArr = new char[2];
		tempArr[0] = Digit[(mByte >>> 4) & 0X0F];
		tempArr[1] = Digit[mByte & 0X0F];

		String s = new String(tempArr);
		return s;
	}

	/**
	 * 校驗openid的合法性
	 * 
	 * @param openid
	 *            受權返回的用戶openid
	 * @param signature
	 *            受權後返回的加密串
	 * @param nonce
	 *            受權返回的隨機數
	 * @param timestamp
	 *            受權返回的時間戳
	 * @return
	 */
	public static boolean validateOpenid(String openid, String signature,
			String nonce, String timestamp) {

		if (StringUtils.isBlank(openid)) {
			return false;
		}

		String key = Configuration.getValue("wechat.paztcf.secret.key", "");		
		System.out.println("wechat.paztcf.secret.key--------" + key );
		String signatureTemp = getSHA1(openid, timestamp, nonce, key);		
		if (StringUtils.isBlank(signature)) {
			signature = "";
		}

		return signature.equalsIgnoreCase(signatureTemp);

	}

	/**
	 * 校驗openid的合法性, 若是openid合法,則返回openid,不然返回null
	 * 
	 * @param request
	 * @return
	 */
	public static String validateOpenid(HttpServletRequest request) {
		HttpSession session = request.getSession();
		String openid = (String) session.getAttribute("openid");

		if (StringUtils.isNotBlank(openid)) {
			return openid;
		}

		openid = request.getParameter("openid"); // 我的微信號
		String signature = request.getParameter("signature"); // 簽名串
		String nonce = request.getParameter("nonce"); // 隨機數
		String timestamp = request.getParameter("timestamp"); // 時間戳

		if (validateOpenid(openid, signature, nonce, timestamp)) {
			session.setAttribute("openid", openid);
			return openid;
		}

		return null;
	}

	/**
	 * 校驗fid 與 sfid是否配對
	 * 
	 * @param fid
	 *            分享人的openid
	 * @param sfid
	 *            分享人的加密後的openid
	 * @return
	 */
	public static boolean isLegalOpenId(String fid, String sfid) {
		if (StringUtils.isBlank(fid) || StringUtils.isBlank(sfid)) {
			return false;
		}

		String signature = secretOpenid(fid);

		return sfid.equalsIgnoreCase(signature);
	}

	/**
	 * 對openid進行加密
	 * 
	 * @param openid
	 * @return
	 */
	public static String secretOpenid(String openid) {

		String key = Configuration.getValue("wechat.paztcf.secret.key", "");

		return getSHA1(openid, "100", "100", key);
	}

	/**
	 * 微信環境不精準檢測
	 * 
	 * @param userAgent
	 * @return
	 */
	public static boolean isComeFromWechat(String userAgent) {
		userAgent = userAgent.toLowerCase();

		return userAgent.contains("micromessenger");
	}

	/**
	 * 微信環境不精準檢測
	 * 
	 * @param userAgent
	 * @return
	 */
	public static boolean isComeFromWechat(HttpServletRequest request) {
		String userAgent = request.getHeader("User-Agent").toLowerCase();
		;

		return userAgent.contains("micromessenger");
	}

	/**
	 * 是否來自qq瀏覽器
	 * 
	 * @param request
	 * @return
	 */
	public static boolean isComeFromQQ(HttpServletRequest request) {
		String userAgent = request.getHeader("User-Agent").toLowerCase();
		;

		return userAgent.contains("qq");
	}

	/**
	 * 獲取微信受權的url地址
	 * 
	 * @param backUrl
	 * @return
	 */
	public static String getAuthorizeUrl(String backUrl) {
		// 受權相關
		String appid = Configuration.getValue("ucp.life.appid"); // 公衆帳號appId
		String weappNo = Configuration.getValue("ucp.life.weappNo"); // 公衆帳號
		String redirectServer = Configuration.getValue("ucp.ztkf.redirect.url"); // ucp回調服務器
		String authorizeUrl = "https://open.weixin.qq.com/connect/oauth2/authorize?"
				+ "appid=%s&redirect_uri=%s&"
				+ "response_type=code&scope=snsapi_base&state=sig#wechat_redirect"; // 微信受權的地址

		try {
			backUrl = URLEncoder.encode(backUrl, "utf-8"); // 對backUrl作編碼
		} catch (Exception e) {
			e.printStackTrace();
		}

		String redirectUri = String.format("%s?weappNo=%s&backUrl=%s",
				redirectServer, weappNo, backUrl); // ucp回調調轉地址
		try {
			redirectUri = URLEncoder.encode(redirectUri, "utf-8"); // 對redirectUri作編碼
		} catch (Exception e) {
			// nothing todo.
		}

		return String.format(authorizeUrl, appid, redirectUri);
	}


	public static void main(String[] args) throws Exception {
		// 生成二維碼
		String url = "http://www.pingan.com/pa18shoplife/mobile/product/warminglife/quote.jsp?WT.mc_id=mobi07-pab-s102-ewm&eid=normal&appColumn=2A7010AC328C32F8E054002128BFD47A";
		String src = generateQRCode(url,200,"1");
		System.out.println(url.length());
		String html = "<img src=\"" + src + "\">";
		FileOutputStream ops = new FileOutputStream("D:\\QRCodeImage\\cecece.html");
		org.apache.commons.io.IOUtils.write(html, ops, "utf-8");
		ops.close();
	}

}
相關文章
相關標籤/搜索