├── .editorconfig ├── .gitignore ├── .travis.yml ├── LICENSE ├── README.md ├── pom.xml └── src ├── main └── java │ └── com │ └── github │ └── binarywang │ └── utils │ └── qrcode │ ├── BufferedImageLuminanceSource.java │ ├── MatrixToImageWriter.java │ ├── MatrixToLogoImageConfig.java │ └── QrcodeUtils.java └── test ├── java └── com │ └── github │ └── binarywang │ └── utils │ └── qrcode │ └── QrcodeUtilsTest.java └── resources └── logo.png /.editorconfig: -------------------------------------------------------------------------------- 1 | # EditorConfig: http://editorconfig.org/ 2 | 3 | root = true 4 | 5 | [*] 6 | indent_style = space 7 | indent_size = 2 8 | end_of_line = lf 9 | charset = utf-8 10 | trim_trailing_whitespace = true 11 | insert_final_newline = true 12 | 13 | [*.md] 14 | trim_trailing_whitespace = false 15 | 16 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | *.class 2 | 3 | # Mobile Tools for Java (J2ME) 4 | .mtj.tmp/ 5 | 6 | # Package Files # 7 | *.jar 8 | *.war 9 | *.ear 10 | 11 | # eclipse 12 | .settings/ 13 | .classpath 14 | .project 15 | .dbeaver/ 16 | 17 | # virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml 18 | hs_err_pid* 19 | /.idea/ 20 | *.iml 21 | /target/ 22 | pom.xml.versionsBackup 23 | -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | language: java 2 | jdk: 3 | - openjdk8 4 | 5 | script: "mvn clean package -Dmaven.test.skip=true" 6 | 7 | branches: 8 | only: 9 | - main 10 | 11 | notifications: 12 | email: 13 | - a@binarywang.com 14 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Apache License 2 | Version 2.0, January 2004 3 | http://www.apache.org/licenses/ 4 | 5 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 6 | 7 | 1. Definitions. 8 | 9 | "License" shall mean the terms and conditions for use, reproduction, 10 | and distribution as defined by Sections 1 through 9 of this document. 11 | 12 | "Licensor" shall mean the copyright owner or entity authorized by 13 | the copyright owner that is granting the License. 14 | 15 | "Legal Entity" shall mean the union of the acting entity and all 16 | other entities that control, are controlled by, or are under common 17 | control with that entity. For the purposes of this definition, 18 | "control" means (i) the power, direct or indirect, to cause the 19 | direction or management of such entity, whether by contract or 20 | otherwise, or (ii) ownership of fifty percent (50%) or more of the 21 | outstanding shares, or (iii) beneficial ownership of such entity. 22 | 23 | "You" (or "Your") shall mean an individual or Legal Entity 24 | exercising permissions granted by this License. 25 | 26 | "Source" form shall mean the preferred form for making modifications, 27 | including but not limited to software source code, documentation 28 | source, and configuration files. 29 | 30 | "Object" form shall mean any form resulting from mechanical 31 | transformation or translation of a Source form, including but 32 | not limited to compiled object code, generated documentation, 33 | and conversions to other media types. 34 | 35 | "Work" shall mean the work of authorship, whether in Source or 36 | Object form, made available under the License, as indicated by a 37 | copyright notice that is included in or attached to the work 38 | (an example is provided in the Appendix below). 39 | 40 | "Derivative Works" shall mean any work, whether in Source or Object 41 | form, that is based on (or derived from) the Work and for which the 42 | editorial revisions, annotations, elaborations, or other modifications 43 | represent, as a whole, an original work of authorship. For the purposes 44 | of this License, Derivative Works shall not include works that remain 45 | separable from, or merely link (or bind by name) to the interfaces of, 46 | the Work and Derivative Works thereof. 47 | 48 | "Contribution" shall mean any work of authorship, including 49 | the original version of the Work and any modifications or additions 50 | to that Work or Derivative Works thereof, that is intentionally 51 | submitted to Licensor for inclusion in the Work by the copyright owner 52 | or by an individual or Legal Entity authorized to submit on behalf of 53 | the copyright owner. For the purposes of this definition, "submitted" 54 | means any form of electronic, verbal, or written communication sent 55 | to the Licensor or its representatives, including but not limited to 56 | communication on electronic mailing lists, source code control systems, 57 | and issue tracking systems that are managed by, or on behalf of, the 58 | Licensor for the purpose of discussing and improving the Work, but 59 | excluding communication that is conspicuously marked or otherwise 60 | designated in writing by the copyright owner as "Not a Contribution." 61 | 62 | "Contributor" shall mean Licensor and any individual or Legal Entity 63 | on behalf of whom a Contribution has been received by Licensor and 64 | subsequently incorporated within the Work. 65 | 66 | 2. Grant of Copyright License. Subject to the terms and conditions of 67 | this License, each Contributor hereby grants to You a perpetual, 68 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 69 | copyright license to reproduce, prepare Derivative Works of, 70 | publicly display, publicly perform, sublicense, and distribute the 71 | Work and such Derivative Works in Source or Object form. 72 | 73 | 3. Grant of Patent License. Subject to the terms and conditions of 74 | this License, each Contributor hereby grants to You a perpetual, 75 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 76 | (except as stated in this section) patent license to make, have made, 77 | use, offer to sell, sell, import, and otherwise transfer the Work, 78 | where such license applies only to those patent claims licensable 79 | by such Contributor that are necessarily infringed by their 80 | Contribution(s) alone or by combination of their Contribution(s) 81 | with the Work to which such Contribution(s) was submitted. If You 82 | institute patent litigation against any entity (including a 83 | cross-claim or counterclaim in a lawsuit) alleging that the Work 84 | or a Contribution incorporated within the Work constitutes direct 85 | or contributory patent infringement, then any patent licenses 86 | granted to You under this License for that Work shall terminate 87 | as of the date such litigation is filed. 88 | 89 | 4. Redistribution. You may reproduce and distribute copies of the 90 | Work or Derivative Works thereof in any medium, with or without 91 | modifications, and in Source or Object form, provided that You 92 | meet the following conditions: 93 | 94 | (a) You must give any other recipients of the Work or 95 | Derivative Works a copy of this License; and 96 | 97 | (b) You must cause any modified files to carry prominent notices 98 | stating that You changed the files; and 99 | 100 | (c) You must retain, in the Source form of any Derivative Works 101 | that You distribute, all copyright, patent, trademark, and 102 | attribution notices from the Source form of the Work, 103 | excluding those notices that do not pertain to any part of 104 | the Derivative Works; and 105 | 106 | (d) If the Work includes a "NOTICE" text file as part of its 107 | distribution, then any Derivative Works that You distribute must 108 | include a readable copy of the attribution notices contained 109 | within such NOTICE file, excluding those notices that do not 110 | pertain to any part of the Derivative Works, in at least one 111 | of the following places: within a NOTICE text file distributed 112 | as part of the Derivative Works; within the Source form or 113 | documentation, if provided along with the Derivative Works; or, 114 | within a display generated by the Derivative Works, if and 115 | wherever such third-party notices normally appear. The contents 116 | of the NOTICE file are for informational purposes only and 117 | do not modify the License. You may add Your own attribution 118 | notices within Derivative Works that You distribute, alongside 119 | or as an addendum to the NOTICE text from the Work, provided 120 | that such additional attribution notices cannot be construed 121 | as modifying the License. 122 | 123 | You may add Your own copyright statement to Your modifications and 124 | may provide additional or different license terms and conditions 125 | for use, reproduction, or distribution of Your modifications, or 126 | for any such Derivative Works as a whole, provided Your use, 127 | reproduction, and distribution of the Work otherwise complies with 128 | the conditions stated in this License. 129 | 130 | 5. Submission of Contributions. Unless You explicitly state otherwise, 131 | any Contribution intentionally submitted for inclusion in the Work 132 | by You to the Licensor shall be under the terms and conditions of 133 | this License, without any additional terms or conditions. 134 | Notwithstanding the above, nothing herein shall supersede or modify 135 | the terms of any separate license agreement you may have executed 136 | with Licensor regarding such Contributions. 137 | 138 | 6. Trademarks. This License does not grant permission to use the trade 139 | names, trademarks, service marks, or product names of the Licensor, 140 | except as required for reasonable and customary use in describing the 141 | origin of the Work and reproducing the content of the NOTICE file. 142 | 143 | 7. Disclaimer of Warranty. Unless required by applicable law or 144 | agreed to in writing, Licensor provides the Work (and each 145 | Contributor provides its Contributions) on an "AS IS" BASIS, 146 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 147 | implied, including, without limitation, any warranties or conditions 148 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A 149 | PARTICULAR PURPOSE. You are solely responsible for determining the 150 | appropriateness of using or redistributing the Work and assume any 151 | risks associated with Your exercise of permissions under this License. 152 | 153 | 8. Limitation of Liability. In no event and under no legal theory, 154 | whether in tort (including negligence), contract, or otherwise, 155 | unless required by applicable law (such as deliberate and grossly 156 | negligent acts) or agreed to in writing, shall any Contributor be 157 | liable to You for damages, including any direct, indirect, special, 158 | incidental, or consequential damages of any character arising as a 159 | result of this License or out of the use or inability to use the 160 | Work (including but not limited to damages for loss of goodwill, 161 | work stoppage, computer failure or malfunction, or any and all 162 | other commercial damages or losses), even if such Contributor 163 | has been advised of the possibility of such damages. 164 | 165 | 9. Accepting Warranty or Additional Liability. While redistributing 166 | the Work or Derivative Works thereof, You may choose to offer, 167 | and charge a fee for, acceptance of support, warranty, indemnity, 168 | or other liability obligations and/or rights consistent with this 169 | License. However, in accepting such obligations, You may act only 170 | on Your own behalf and on Your sole responsibility, not on behalf 171 | of any other Contributor, and only if You agree to indemnify, 172 | defend, and hold each Contributor harmless for any liability 173 | incurred by, or claims asserted against, such Contributor by reason 174 | of your accepting any such warranty or additional liability. 175 | 176 | END OF TERMS AND CONDITIONS 177 | 178 | APPENDIX: How to apply the Apache License to your work. 179 | 180 | To apply the Apache License to your work, attach the following 181 | boilerplate notice, with the fields enclosed by brackets "{}" 182 | replaced with your own identifying information. (Don't include 183 | the brackets!) The text should be enclosed in the appropriate 184 | comment syntax for the file format. We also recommend that a 185 | file or class name and description of purpose be included on the 186 | same "printed page" as the copyright notice for easier 187 | identification within third-party archives. 188 | 189 | Copyright {yyyy} {name of copyright owner} 190 | 191 | Licensed under the Apache License, Version 2.0 (the "License"); 192 | you may not use this file except in compliance with the License. 193 | You may obtain a copy of the License at 194 | 195 | http://www.apache.org/licenses/LICENSE-2.0 196 | 197 | Unless required by applicable law or agreed to in writing, software 198 | distributed under the License is distributed on an "AS IS" BASIS, 199 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 200 | See the License for the specific language governing permissions and 201 | limitations under the License. 202 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # qrcode-utils 2 | 二维码生成工具 3 | --------------------------------- 4 | [![Maven Central](https://maven-badges.herokuapp.com/maven-central/com.github.binarywang/qrcode-utils/badge.svg)](https://maven-badges.herokuapp.com/maven-central/com.github.binarywang/qrcode-utils) 5 | [![Build Status](https://app.travis-ci.com/binarywang/qrcode-utils.svg?branch=main)](https://app.travis-ci.com/github/binarywang/qrcode-utils) 6 | 7 | #### Maven: 8 | 9 | 10 | 11 | com.github.binarywang 12 | qrcode-utils 13 | 1.3 14 | 15 | -------------------------------------------------------------------------------- /pom.xml: -------------------------------------------------------------------------------- 1 | 3 | 4.0.0 4 | com.github.binarywang 5 | qrcode-utils 6 | jar 7 | 1.3 8 | qrcode-utils 9 | https://github.com/binarywang/qrcode-utils 10 | qrcode utils 11 | 12 | 13 | The Apache License, Version 2.0 14 | http://www.apache.org/licenses/LICENSE-2.0.txt 15 | 16 | 17 | 18 | 19 | scm:git:https://github.com/binarywang/qrcode-utils.git 20 | scm:git:git@github.com:binarywang/qrcode-utils.git 21 | https://github.com/binarywang/qrcode-utils 22 | 23 | 24 | 25 | 26 | Binary Wang 27 | binarywang@gmail.com 28 | https://github.com/binarywang 29 | 30 | 31 | 32 | 33 | 1.7 34 | 1.7 35 | 36 | UTF-8 37 | 38 | 39 | 40 | org.testng 41 | testng 42 | 6.9.10 43 | test 44 | 45 | 46 | 47 | com.google.zxing 48 | core 49 | 3.5.0 50 | 51 | 52 | commons-io 53 | commons-io 54 | 2.14.0 55 | 56 | 57 | com.google.guava 58 | guava 59 | 33.2.0-jre 60 | 61 | 62 | org.slf4j 63 | slf4j-api 64 | 1.7.22 65 | 66 | 67 | org.slf4j 68 | slf4j-log4j12 69 | 1.7.36 70 | test 71 | 72 | 73 | org.projectlombok 74 | lombok 75 | 1.18.24 76 | provided 77 | 78 | 79 | 80 | 81 | 82 | ossrh 83 | https://oss.sonatype.org/content/repositories/snapshots 84 | 85 | 86 | ossrh 87 | https://oss.sonatype.org/service/local/staging/deploy/maven2/ 88 | 89 | 90 | 91 | 92 | 93 | doclint-java8-disable 94 | 95 | [1.8,) 96 | 97 | 98 | -Xdoclint:none 99 | 100 | 101 | 102 | 103 | release 104 | 105 | 106 | 107 | org.apache.maven.plugins 108 | maven-source-plugin 109 | 2.2.1 110 | 111 | 112 | attach-sources 113 | 114 | jar-no-fork 115 | 116 | 117 | 118 | 119 | 120 | org.apache.maven.plugins 121 | maven-javadoc-plugin 122 | 2.9.1 123 | 124 | 125 | attach-javadocs 126 | 127 | jar 128 | 129 | 130 | 131 | 132 | ${javadoc.opts} 133 | UTF-8 134 | zh_CN 135 | 136 | 137 | 138 | org.apache.maven.plugins 139 | maven-gpg-plugin 140 | 1.5 141 | 142 | 143 | sign-artifacts 144 | verify 145 | 146 | sign 147 | 148 | 149 | 150 | 151 | 152 | 153 | 154 | 155 | 156 | 157 | 158 | 159 | 160 | org.apache.maven.plugins 161 | maven-surefire-plugin 162 | 2.17 163 | 164 | 165 | 166 | 167 | 168 | 169 | org.sonatype.plugins 170 | nexus-staging-maven-plugin 171 | 1.6.3 172 | true 173 | 174 | ossrh 175 | https://oss.sonatype.org/ 176 | true 177 | 178 | 179 | 180 | org.apache.maven.plugins 181 | maven-release-plugin 182 | 2.5.1 183 | 184 | true 185 | false 186 | release 187 | deploy 188 | 189 | 190 | 191 | org.apache.maven.plugins 192 | maven-compiler-plugin 193 | 3.6.0 194 | 195 | UTF-8 196 | 197 | 198 | 199 | 200 | 201 | -------------------------------------------------------------------------------- /src/main/java/com/github/binarywang/utils/qrcode/BufferedImageLuminanceSource.java: -------------------------------------------------------------------------------- 1 | package com.github.binarywang.utils.qrcode; 2 | 3 | import com.google.zxing.LuminanceSource; 4 | 5 | import java.awt.*; 6 | import java.awt.geom.AffineTransform; 7 | import java.awt.image.BufferedImage; 8 | 9 | /** 10 | *
 11 |  * Created by Binary Wang on 2017-01-05.
 12 |  * @author binarywang(Binary Wang)
 13 |  * 
14 | */ 15 | public final class BufferedImageLuminanceSource extends LuminanceSource { 16 | 17 | private final BufferedImage image; 18 | private final int left; 19 | private final int top; 20 | 21 | public BufferedImageLuminanceSource(BufferedImage image) { 22 | this(image, 0, 0, image.getWidth(), image.getHeight()); 23 | } 24 | 25 | public BufferedImageLuminanceSource(BufferedImage image, int left, int top, 26 | int width, int height) { 27 | super(width, height); 28 | 29 | int sourceWidth = image.getWidth(); 30 | int sourceHeight = image.getHeight(); 31 | if (left + width > sourceWidth || top + height > sourceHeight) { 32 | throw new IllegalArgumentException("Crop rectangle does not fit within image data."); 33 | } 34 | 35 | for (int y = top; y < top + height; y++) { 36 | for (int x = left; x < left + width; x++) { 37 | if ((image.getRGB(x, y) & 0xFF000000) == 0) { 38 | image.setRGB(x, y, 0xFFFFFFFF);// = white 39 | } 40 | } 41 | } 42 | 43 | this.image = new BufferedImage(sourceWidth, sourceHeight, 44 | BufferedImage.TYPE_BYTE_GRAY); 45 | this.image.getGraphics().drawImage(image, 0, 0, null); 46 | this.left = left; 47 | this.top = top; 48 | } 49 | 50 | @Override 51 | public byte[] getRow(int y, byte[] row) { 52 | if (y < 0 || y >= getHeight()) { 53 | throw new IllegalArgumentException( 54 | "Requested row is outside the image: " + y); 55 | } 56 | int width = getWidth(); 57 | if (row == null || row.length < width) { 58 | row = new byte[width]; 59 | } 60 | this.image.getRaster().getDataElements(this.left, this.top + y, width, 61 | 1, row); 62 | return row; 63 | } 64 | 65 | @Override 66 | public byte[] getMatrix() { 67 | int width = getWidth(); 68 | int height = getHeight(); 69 | int area = width * height; 70 | byte[] matrix = new byte[area]; 71 | this.image.getRaster().getDataElements(this.left, this.top, width, 72 | height, matrix); 73 | return matrix; 74 | } 75 | 76 | @Override 77 | public boolean isCropSupported() { 78 | return true; 79 | } 80 | 81 | @Override 82 | public LuminanceSource crop(int left, int top, int width, int height) { 83 | return new BufferedImageLuminanceSource(this.image, this.left + left, 84 | this.top + top, width, height); 85 | } 86 | 87 | @Override 88 | public boolean isRotateSupported() { 89 | return true; 90 | } 91 | 92 | @Override 93 | public LuminanceSource rotateCounterClockwise() { 94 | 95 | int sourceWidth = this.image.getWidth(); 96 | int sourceHeight = this.image.getHeight(); 97 | 98 | AffineTransform transform = new AffineTransform(0.0, -1.0, 1.0, 0.0, 99 | 0.0, sourceWidth); 100 | 101 | BufferedImage rotatedImage = new BufferedImage(sourceHeight, 102 | sourceWidth, BufferedImage.TYPE_BYTE_GRAY); 103 | 104 | Graphics2D g = rotatedImage.createGraphics(); 105 | g.drawImage(this.image, transform, null); 106 | g.dispose(); 107 | 108 | int width = getWidth(); 109 | return new BufferedImageLuminanceSource(rotatedImage, this.top, 110 | sourceWidth - (this.left + width), getHeight(), width); 111 | } 112 | 113 | } 114 | -------------------------------------------------------------------------------- /src/main/java/com/github/binarywang/utils/qrcode/MatrixToImageWriter.java: -------------------------------------------------------------------------------- 1 | package com.github.binarywang.utils.qrcode; 2 | 3 | import com.google.zxing.common.BitMatrix; 4 | 5 | import javax.imageio.ImageIO; 6 | import java.awt.image.BufferedImage; 7 | import java.io.File; 8 | import java.io.IOException; 9 | import java.io.OutputStream; 10 | 11 | /** 12 | *
13 |  * Created by Binary Wang on 2017-01-05.
14 |  * @author binarywang(Binary Wang)
15 |  * 
16 | */ 17 | public final class MatrixToImageWriter { 18 | 19 | private static final int BLACK = 0xFF000000; 20 | private static final int WHITE = 0xFFFFFFFF; 21 | 22 | private MatrixToImageWriter() { 23 | } 24 | 25 | public static BufferedImage toBufferedImage(BitMatrix matrix) { 26 | int width = matrix.getWidth(); 27 | int height = matrix.getHeight(); 28 | BufferedImage image = new BufferedImage(width, height, 29 | BufferedImage.TYPE_INT_RGB); 30 | for (int x = 0; x < width; x++) { 31 | for (int y = 0; y < height; y++) { 32 | image.setRGB(x, y, matrix.get(x, y) ? BLACK : WHITE); 33 | } 34 | } 35 | 36 | return image; 37 | } 38 | 39 | public static void writeToFile(BitMatrix matrix, String format, File file) 40 | throws IOException { 41 | BufferedImage image = toBufferedImage(matrix); 42 | if (!ImageIO.write(image, format, file)) { 43 | throw new IOException("Could not write an image of format " + format + " to " + file); 44 | } 45 | } 46 | 47 | public static void writeToStream(BitMatrix matrix, String format, OutputStream stream) 48 | throws IOException { 49 | BufferedImage image = toBufferedImage(matrix); 50 | if (!ImageIO.write(image, format, stream)) { 51 | throw new IOException("Could not write an image of format " + format); 52 | } 53 | } 54 | 55 | } 56 | -------------------------------------------------------------------------------- /src/main/java/com/github/binarywang/utils/qrcode/MatrixToLogoImageConfig.java: -------------------------------------------------------------------------------- 1 | package com.github.binarywang.utils.qrcode; 2 | 3 | import java.awt.*; 4 | 5 | /** 6 | *
 7 |  * Created by Binary Wang on 2017-01-05.
 8 |  * @author binarywang(Binary Wang)
 9 |  * 
10 | */ 11 | public class MatrixToLogoImageConfig { 12 | /** 13 | * logo默认边框颜色 14 | */ 15 | public static final Color DEFAULT_BORDERCOLOR = Color.WHITE; 16 | /** 17 | * logo默认边框宽度 18 | */ 19 | public static final int DEFAULT_BORDER = 5; 20 | /** 21 | * logo大小默认为照片的1/5 22 | */ 23 | public static final int DEFAULT_LOGOPART = 5; 24 | 25 | private final int border; 26 | private final Color borderColor; 27 | private final int logoPart; 28 | 29 | public MatrixToLogoImageConfig() { 30 | this(DEFAULT_BORDERCOLOR, DEFAULT_LOGOPART); 31 | } 32 | 33 | public MatrixToLogoImageConfig(Color borderColor, int logoPart) { 34 | this(borderColor, logoPart, DEFAULT_BORDER); 35 | } 36 | 37 | public MatrixToLogoImageConfig(Color borderColor, int logoPart, int border) { 38 | this.borderColor = borderColor; 39 | this.logoPart = logoPart; 40 | this.border = border; 41 | } 42 | 43 | 44 | public Color getBorderColor() { 45 | return this.borderColor; 46 | } 47 | 48 | public int getBorder() { 49 | return this.border; 50 | } 51 | 52 | public int getLogoPart() { 53 | return this.logoPart; 54 | } 55 | 56 | } 57 | -------------------------------------------------------------------------------- /src/main/java/com/github/binarywang/utils/qrcode/QrcodeUtils.java: -------------------------------------------------------------------------------- 1 | package com.github.binarywang.utils.qrcode; 2 | 3 | import com.google.common.collect.Maps; 4 | import com.google.zxing.*; 5 | import com.google.zxing.common.BitMatrix; 6 | import com.google.zxing.common.HybridBinarizer; 7 | import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel; 8 | import lombok.experimental.UtilityClass; 9 | import lombok.extern.slf4j.Slf4j; 10 | 11 | import javax.imageio.ImageIO; 12 | import java.awt.*; 13 | import java.awt.geom.RoundRectangle2D; 14 | import java.awt.image.BufferedImage; 15 | import java.io.*; 16 | import java.nio.charset.StandardCharsets; 17 | import java.nio.file.Files; 18 | import java.util.Map; 19 | 20 | /** 21 | *
 22 |  * Created by Binary Wang on 2017-01-05.
 23 |  * @author binarywang(Binary Wang)
 24 |  * 
25 | */ 26 | @Slf4j 27 | @UtilityClass 28 | public class QrcodeUtils { 29 | /** 30 | * 生成二维码的默认边长,因为是正方形的,所以高度和宽度一致 31 | */ 32 | private static final int DEFAULT_LENGTH = 400; 33 | /** 34 | * 生成二维码的格式 35 | */ 36 | private static final String FORMAT = "jpg"; 37 | 38 | /** 39 | * 根据内容生成二维码数据 40 | * 41 | * @param content 二维码文字内容[为了信息安全性,一般都要先进行数据加密] 42 | * @param length 二维码图片宽度和高度 43 | */ 44 | public static BitMatrix createQrcodeMatrix(String content, int length) { 45 | return createQrcodeMatrix(content, length, ErrorCorrectionLevel.H); 46 | } 47 | 48 | public static BitMatrix createQrcodeMatrix(String content, int length, ErrorCorrectionLevel level) { 49 | Map hints = Maps.newEnumMap(EncodeHintType.class); 50 | // 设置字符编码 51 | hints.put(EncodeHintType.CHARACTER_SET, StandardCharsets.UTF_8.name()); 52 | // 指定纠错等级 53 | hints.put(EncodeHintType.ERROR_CORRECTION, level); 54 | 55 | try { 56 | return new MultiFormatWriter().encode(content, BarcodeFormat.QR_CODE, length, length, hints); 57 | } catch (WriterException e) { 58 | throw new RuntimeException("内容为:【" + content + "】的二维码生成失败!", e); 59 | } 60 | } 61 | 62 | /** 63 | * 根据指定边长创建生成的二维码,允许配置logo属性 64 | * 65 | * @param content 二维码内容 66 | * @param length 二维码的高度和宽度 67 | * @param logoFile logo 文件对象,可以为空 68 | * @param logoConfig logo配置,可设置logo展示长宽,边框颜色 69 | * @return 二维码图片的字节数组 70 | */ 71 | public static byte[] createQrcode(String content, int length, File logoFile, MatrixToLogoImageConfig logoConfig) 72 | throws Exception { 73 | if (logoFile != null && !logoFile.exists()) { 74 | throw new IllegalArgumentException("请提供正确的logo文件!"); 75 | } 76 | 77 | try (InputStream logo = logoFile == null ? null : new FileInputStream(logoFile)) { 78 | return createQrcode(content, length, logo, logoConfig); 79 | } 80 | } 81 | 82 | public static byte[] createQrcode(String content, int length, InputStream logo, MatrixToLogoImageConfig logoConfig) throws Exception { 83 | BufferedImage img = generateQRCodeImage(content, length, logo, logoConfig); 84 | ByteArrayOutputStream baos = new ByteArrayOutputStream(); 85 | ImageIO.write(img, FORMAT, baos); 86 | return baos.toByteArray(); 87 | } 88 | 89 | /** 90 | * 根据指定边长创建生成的二维码 91 | * 92 | * @param content 二维码内容 93 | * @param length 二维码的高度和宽度 94 | * @param logoFile logo 文件对象,可以为空 95 | * @return 二维码图片的字节数组 96 | */ 97 | public static byte[] createQrcode(String content, int length, File logoFile) throws Exception { 98 | return createQrcode(content, length, logoFile, new MatrixToLogoImageConfig()); 99 | } 100 | 101 | /** 102 | * 创建生成默认高度(400)的二维码图片 103 | * 可以指定是否贷logo 104 | * 105 | * @param content 二维码内容 106 | * @param logoFile logo 文件对象,可以为空 107 | * @return 二维码图片的字节数组 108 | */ 109 | public static byte[] createQrcode(String content, File logoFile) throws Exception { 110 | return createQrcode(content, DEFAULT_LENGTH, logoFile); 111 | } 112 | 113 | public static BufferedImage generateQRCodeImage(String content, int length, InputStream logo, MatrixToLogoImageConfig logoConfig) throws Exception { 114 | // 生成二维码图像 115 | BitMatrix qrCodeMatrix = createQrcodeMatrix(content, length); 116 | BufferedImage img = MatrixToImageWriter.toBufferedImage(qrCodeMatrix); 117 | try { 118 | if (logo != null) { 119 | overlapImage(img, FORMAT, logo, logoConfig); 120 | } 121 | } catch (Exception e) { 122 | throw new RuntimeException("为二维码添加LOGO时失败!", e); 123 | } 124 | return img; 125 | } 126 | 127 | public static BufferedImage generateQRCodeImage(String content, int length, File logoFile, MatrixToLogoImageConfig logoConfig) throws Exception { 128 | if (logoFile != null && !logoFile.exists()) { 129 | throw new IllegalArgumentException("请提供正确的logo文件!"); 130 | } 131 | try (InputStream logo = Files.newInputStream(logoFile.toPath())) { 132 | return generateQRCodeImage(content, length, logo, logoConfig); 133 | } 134 | } 135 | 136 | public static BufferedImage generateQRCodeImage(String content, int length, InputStream logo) throws Exception { 137 | return generateQRCodeImage(content, length, logo, new MatrixToLogoImageConfig()); 138 | } 139 | 140 | public static BufferedImage generateQRCodeImage(String content, int length, File logoFile) throws Exception { 141 | return generateQRCodeImage(content, length, logoFile, new MatrixToLogoImageConfig()); 142 | } 143 | 144 | public static BufferedImage generateQRCodeImage(String content, InputStream logo) throws Exception { 145 | return generateQRCodeImage(content, DEFAULT_LENGTH, logo); 146 | } 147 | 148 | public static BufferedImage generateQRCodeImage(String content, File logoFile) throws Exception { 149 | return generateQRCodeImage(content, DEFAULT_LENGTH, logoFile); 150 | } 151 | 152 | /** 153 | * 将logo添加到二维码中间 154 | * 155 | * @param image 生成的二维码图片对象 156 | * @param logo logo文件对象 157 | * @param ignoredFormat 图片格式 158 | */ 159 | private static void overlapImage(final BufferedImage image, String ignoredFormat, final InputStream logo, 160 | MatrixToLogoImageConfig logoConfig) throws IOException { 161 | BufferedImage logoImg = ImageIO.read(logo); 162 | logoImg = clipRound(logoImg); 163 | Graphics2D g = logoImg.createGraphics(); 164 | // 考虑到logo图片贴到二维码中,建议大小不要超过二维码的1/5; 165 | int width = image.getWidth() / logoConfig.getLogoPart(); 166 | int height = image.getHeight() / logoConfig.getLogoPart(); 167 | int radius = width / 10; 168 | // logo起始位置,此目的是为logo居中显示 169 | int x = (image.getWidth() - width) / 2; 170 | int y = (image.getHeight() - height) / 2; 171 | 172 | // 创建一个支持有透明度的图像缓冲区 173 | BufferedImage buffer = g.getDeviceConfiguration().createCompatibleImage(image.getWidth(), image.getHeight(), Transparency.TRANSLUCENT); 174 | g.dispose(); 175 | 176 | // 绘制阴影 177 | g = buffer.createGraphics(); 178 | g.setComposite(AlphaComposite.getInstance(AlphaComposite.XOR, 0.1f)); 179 | g.setColor(Color.BLACK); 180 | g.fillRoundRect(x + 10, y + 10, width - 20, height, radius, radius); 181 | g.dispose(); 182 | 183 | // 绘制LOGO到缓冲区 184 | g = buffer.createGraphics(); 185 | g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC, 1)); 186 | g.drawImage(logoImg, x, y, width, height, null); 187 | 188 | // 给logo画边框 189 | // 构造一个具有指定线条宽度以及 cap 和 join 风格的默认值的实心 BasicStroke 190 | g.setStroke(new BasicStroke(logoConfig.getBorder())); 191 | g.setColor(logoConfig.getBorderColor()); 192 | g.drawRoundRect(x, y, width, height, radius, radius); 193 | g.setStroke(new BasicStroke(1)); 194 | g.setColor(Color.GRAY); 195 | g.drawRoundRect(x + logoConfig.getBorder() / 2, y + logoConfig.getBorder() / 2, width - logoConfig.getBorder(), 196 | height - logoConfig.getBorder(), radius, radius); 197 | g.dispose(); 198 | 199 | // 将带阴影的图像绘制到二维码上 200 | g = image.createGraphics(); 201 | g.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_ATOP, 1.0f)); 202 | g.drawImage(buffer, 0, 0, image.getWidth(), image.getHeight(), null); 203 | g.dispose(); 204 | } 205 | 206 | /** 207 | * 为LOGO剪出圆角 208 | * 209 | * @param srcImage LOGO图像 210 | */ 211 | private static BufferedImage clipRound(BufferedImage srcImage) { 212 | int width = srcImage.getWidth(); 213 | int height = srcImage.getHeight(); 214 | int radius = width / 10; 215 | 216 | BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB); 217 | Graphics2D g = image.createGraphics(); 218 | g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON); 219 | g.setClip(new RoundRectangle2D.Double(0, 0, width, height, radius, radius)); 220 | g.drawImage(srcImage, 0, 0, null); 221 | g.dispose(); 222 | return image; 223 | } 224 | 225 | /** 226 | * 解析二维码 227 | * 228 | * @param file 二维码文件内容 229 | * @return 二维码的内容 230 | */ 231 | public static String decodeQrcode(File file) throws IOException, NotFoundException { 232 | BufferedImage image = ImageIO.read(file); 233 | LuminanceSource source = new BufferedImageLuminanceSource(image); 234 | Binarizer binarizer = new HybridBinarizer(source); 235 | BinaryBitmap binaryBitmap = new BinaryBitmap(binarizer); 236 | Map hints = Maps.newEnumMap(DecodeHintType.class); 237 | hints.put(DecodeHintType.CHARACTER_SET, StandardCharsets.UTF_8.name()); 238 | return new MultiFormatReader().decode(binaryBitmap, hints).getText(); 239 | } 240 | } 241 | -------------------------------------------------------------------------------- /src/test/java/com/github/binarywang/utils/qrcode/QrcodeUtilsTest.java: -------------------------------------------------------------------------------- 1 | package com.github.binarywang.utils.qrcode; 2 | 3 | import com.beust.jcommander.internal.Lists; 4 | import lombok.extern.slf4j.Slf4j; 5 | import org.apache.commons.io.FileUtils; 6 | import org.apache.log4j.BasicConfigurator; 7 | import org.testng.Assert; 8 | import org.testng.annotations.BeforeTest; 9 | import org.testng.annotations.Test; 10 | 11 | import javax.imageio.ImageIO; 12 | import java.awt.*; 13 | import java.awt.image.BufferedImage; 14 | import java.io.File; 15 | import java.io.InputStream; 16 | import java.nio.file.Files; 17 | import java.nio.file.Path; 18 | import java.util.List; 19 | 20 | /** 21 | *
22 |  * Created by Binary Wang on 2017-01-05.
23 |  * @author binarywang(Binary Wang)
24 |  * 
25 | */ 26 | @Slf4j 27 | public class QrcodeUtilsTest { 28 | 29 | private String content = "abc"; 30 | private List generatedQrcodePaths = Lists.newArrayList(); 31 | 32 | @BeforeTest 33 | public void setup() { 34 | BasicConfigurator.configure(); 35 | } 36 | 37 | @Test 38 | public void testCreateQrcode() throws Exception { 39 | byte[] bytes = QrcodeUtils.createQrcode(content, 800, null); 40 | Path path = Files.createTempFile("qrcode_800_", ".jpg"); 41 | generatedQrcodePaths.add(path); 42 | log.info("{}", path.toAbsolutePath()); 43 | Files.write(path, bytes); 44 | 45 | bytes = QrcodeUtils.createQrcode(content, null); 46 | path = Files.createTempFile("qrcode_400_", ".jpg"); 47 | generatedQrcodePaths.add(path); 48 | log.info("{}", path.toAbsolutePath()); 49 | Files.write(path, bytes); 50 | } 51 | 52 | @Test 53 | public void testCreateQrcodeWithLogo() throws Exception { 54 | try (InputStream inputStream = ClassLoader.getSystemResourceAsStream("logo.png")) { 55 | File logoFile = Files.createTempFile("logo_", ".jpg").toFile(); 56 | FileUtils.copyInputStreamToFile(inputStream, logoFile); 57 | log.info("{}", logoFile); 58 | byte[] bytes = QrcodeUtils.createQrcode(content, 800, logoFile); 59 | Path path = Files.createTempFile("qrcode_with_logo_", ".jpg"); 60 | generatedQrcodePaths.add(path); 61 | log.info("{}", path.toAbsolutePath()); 62 | Files.write(path, bytes); 63 | } 64 | } 65 | 66 | @Test(dependsOnMethods = {"testCreateQrcode", "testCreateQrcodeWithLogo"}) 67 | public void testDecodeQrcode() throws Exception { 68 | for (Path path : generatedQrcodePaths) { 69 | Assert.assertEquals(QrcodeUtils.decodeQrcode(path.toFile()), content); 70 | } 71 | } 72 | 73 | 74 | @Test 75 | public void testPersonCode() throws Exception { 76 | BufferedImage img = QrcodeUtils.generateQRCodeImage(content, 400, ClassLoader.getSystemResourceAsStream("logo.png")); 77 | File outFile = Files.createTempFile("qrcode_with_logo_", ".png").toFile(); 78 | log.info("{}", outFile.getAbsolutePath()); 79 | ImageIO.write(img, "png", outFile); 80 | Desktop.getDesktop().open(outFile); 81 | } 82 | 83 | } 84 | -------------------------------------------------------------------------------- /src/test/resources/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/binarywang/qrcode-utils/16b6c6f894cff4d27e07649013fdb1bfb2a4a441/src/test/resources/logo.png --------------------------------------------------------------------------------