!778 新增AnsiColors(改自Spring Boot)、AnsiColorWrapper,优化QrCodeUtil

Merge pull request !778 from Tom Xin/v5-dev-qrcode-svg
This commit is contained in:
Looly
2022-09-14 14:57:18 +00:00
committed by Gitee
7 changed files with 532 additions and 95 deletions

View File

@@ -5,11 +5,8 @@ import cn.hutool.core.img.Img;
import cn.hutool.core.img.ImgUtil;
import cn.hutool.core.io.FileUtil;
import cn.hutool.core.io.IoUtil;
import cn.hutool.core.lang.ansi.Ansi8BitColor;
import cn.hutool.core.lang.ansi.AnsiElement;
import cn.hutool.core.lang.ansi.AnsiEncoder;
import cn.hutool.core.lang.ansi.*;
import cn.hutool.core.util.CharsetUtil;
import cn.hutool.core.util.NumberUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.core.util.URLUtil;
import com.google.zxing.*;
@@ -17,7 +14,9 @@ import com.google.zxing.common.BitMatrix;
import com.google.zxing.common.GlobalHistogramBinarizer;
import com.google.zxing.common.HybridBinarizer;
import java.awt.*;
import java.awt.Color;
import java.awt.Image;
import java.awt.Rectangle;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
@@ -41,12 +40,13 @@ public class QrCodeUtil {
public static final String QR_TYPE_SVG = "svg";// SVG矢量图格式
public static final String QR_TYPE_TXT = "txt";// Ascii Art字符画文本
private static final AnsiColors ansiColors= new AnsiColors(AnsiColors.BitDepth.EIGHT);
/**
* 生成代 logo 图片的 Base64 编码格式的二维码,以 String 形式表示
*
* @param content 内容
* @param qrConfig 二维码配置,包括长、宽、边距、颜色等
* @param qrConfig 二维码配置,包括宽度、高度、边距、颜色等
* @param targetType 类型(图片扩展名),见{@link #QR_TYPE_SVG}、 {@link #QR_TYPE_TXT}、{@link ImgUtil}
* @param logoBase64 logo 图片的 base64 编码
* @return 图片 Base64 编码字符串
@@ -59,7 +59,7 @@ public class QrCodeUtil {
* 生成代 logo 图片的 Base64 编码格式的二维码,以 String 形式表示
*
* @param content 内容
* @param qrConfig 二维码配置,包括长、宽、边距、颜色等
* @param qrConfig 二维码配置,包括宽度、高度、边距、颜色等
* @param targetType 类型(图片扩展名),见{@link #QR_TYPE_SVG}、 {@link #QR_TYPE_TXT}、{@link ImgUtil}
* @param logo logo 图片的byte[]
* @return 图片 Base64 编码字符串
@@ -72,7 +72,7 @@ public class QrCodeUtil {
* 生成代 logo 图片的 Base64 编码格式的二维码,以 String 形式表示
*
* @param content 内容
* @param qrConfig 二维码配置,包括长、宽、边距、颜色等
* @param qrConfig 二维码配置,包括宽度、高度、边距、颜色等
* @param targetType 类型(图片扩展名),见{@link #QR_TYPE_SVG}、 {@link #QR_TYPE_TXT}、{@link ImgUtil}
* @param logo logo 图片的byte[]
* @return 图片 Base64 编码字符串
@@ -90,7 +90,7 @@ public class QrCodeUtil {
* </p>
*
* @param content 内容
* @param qrConfig 二维码配置,包括长、宽、边距、颜色等
* @param qrConfig 二维码配置,包括宽度、高度、边距、颜色等
* @param targetType 类型(图片扩展名),见{@link #QR_TYPE_SVG}、 {@link #QR_TYPE_TXT}、{@link ImgUtil}
* @return 图片 Base64 编码字符串
*/
@@ -125,7 +125,7 @@ public class QrCodeUtil {
/**
* @param content 内容
* @param qrConfig 二维码配置,包括长、宽、边距、颜色等
* @param qrConfig 二维码配置,包括宽度、高度、边距、颜色等
* @return SVG矢量图字符串
* @since 5.8.6
*/
@@ -149,7 +149,7 @@ public class QrCodeUtil {
* 生成ASCII Art字符画形式的二维码
*
* @param content 内容
* @param qrConfig 二维码配置,仅长、宽、边距配置有效
* @param qrConfig 二维码配置,仅宽度、高度、边距配置有效
* @return ASCII Art字符画形式的二维码
* @since 5.8.6
*/
@@ -160,8 +160,8 @@ public class QrCodeUtil {
/**
* @param content 内容
* @param width 宽
* @param height
* @param width 宽度(单位:字符▄的大小)
* @param height 高度(单位:字符▄的大小)
* @return ASCII Art字符画形式的二维码
* @since 5.8.6
*/
@@ -175,8 +175,8 @@ public class QrCodeUtil {
* 生成PNG格式的二维码图片以byte[]形式表示
*
* @param content 内容
* @param width 宽度
* @param height 高度
* @param width 宽度(单位:像素)
* @param height 高度(单位:像素)
* @return 图片的byte[]
* @since 4.0.10
*/
@@ -190,7 +190,7 @@ public class QrCodeUtil {
* 生成PNG格式的二维码图片以byte[]形式表示
*
* @param content 内容
* @param config 二维码配置,包括长、宽、边距、颜色等
* @param config 二维码配置,包括宽度、高度、边距、颜色等
* @return 图片的byte[]
* @since 4.1.2
*/
@@ -204,8 +204,8 @@ public class QrCodeUtil {
* 生成二维码到文件,二维码图片格式取决于文件的扩展名
*
* @param content 文本内容
* @param width 宽度
* @param height 高度
* @param width 宽度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @param height 高度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @param targetFile 目标文件,扩展名决定输出格式
* @return 目标文件
*/
@@ -233,7 +233,7 @@ public class QrCodeUtil {
* 生成二维码到文件,二维码图片格式取决于文件的扩展名
*
* @param content 文本内容
* @param config 二维码配置,包括长、宽、边距、颜色等
* @param config 二维码配置,包括宽度、高度、边距、颜色等
* @param targetFile 目标文件,扩展名决定输出格式
* @return 目标文件
* @since 4.1.2
@@ -261,8 +261,8 @@ public class QrCodeUtil {
* 生成二维码到输出流
*
* @param content 文本内容
* @param width 宽度
* @param height 高度
* @param width 宽度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @param height 高度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @param targetType 类型(图片扩展名),见{@link #QR_TYPE_SVG}、 {@link #QR_TYPE_TXT}、{@link ImgUtil}
* @param out 目标流
*/
@@ -287,7 +287,7 @@ public class QrCodeUtil {
* 生成二维码到输出流
*
* @param content 文本内容
* @param config 二维码配置,包括长、宽、边距、颜色等
* @param config 二维码配置,包括宽度、高度、边距、颜色等
* @param targetType 类型(图片扩展名),见{@link #QR_TYPE_SVG}、 {@link #QR_TYPE_TXT}、{@link ImgUtil}
* @param out 目标流
* @since 4.1.2
@@ -313,8 +313,8 @@ public class QrCodeUtil {
* 生成二维码图片
*
* @param content 文本内容
* @param width 宽度
* @param height 高度
* @param width 宽度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @param height 高度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @return 二维码图片(黑白)
*/
public static BufferedImage generate(String content, int width, int height) {
@@ -326,8 +326,8 @@ public class QrCodeUtil {
*
* @param content 文本内容
* @param format 格式,可选二维码或者条形码
* @param width 宽度
* @param height 高度
* @param width 宽度(单位:像素)
* @param height 高度(单位:像素)
* @return 二维码图片(黑白)
*/
public static BufferedImage generate(String content, BarcodeFormat format, int width, int height) {
@@ -338,7 +338,7 @@ public class QrCodeUtil {
* 生成二维码图片
*
* @param content 文本内容
* @param config 二维码配置,包括长、宽、边距、颜色等
* @param config 二维码配置,包括宽度、高度、边距、颜色等
* @return 二维码图片(黑白)
* @since 4.1.2
*/
@@ -352,7 +352,7 @@ public class QrCodeUtil {
*
* @param content 文本内容
* @param format 格式,可选二维码、条形码等
* @param config 二维码配置,包括长、宽、边距、颜色等
* @param config 二维码配置,包括宽度、高度、边距、颜色等
* @return 二维码图片(黑白)
* @since 4.1.14
*/
@@ -390,8 +390,8 @@ public class QrCodeUtil {
* 将文本内容编码为二维码
*
* @param content 文本内容
* @param width 宽度
* @param height 高度
* @param width 宽度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @param height 高度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @return {@link BitMatrix}
*/
public static BitMatrix encode(String content, int width, int height) {
@@ -402,7 +402,7 @@ public class QrCodeUtil {
* 将文本内容编码为二维码
*
* @param content 文本内容
* @param config 二维码配置,包括长、宽、边距、颜色等
* @param config 二维码配置,包括宽度、高度、边距、颜色等
* @return {@link BitMatrix}
* @since 4.1.2
*/
@@ -415,8 +415,8 @@ public class QrCodeUtil {
*
* @param content 文本内容
* @param format 格式枚举
* @param width 宽度
* @param height 高度
* @param width 宽度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @param height 高度单位类型为一般图片或SVG时单位是像素类型为 Ascii Art 字符画时,单位是字符▄或▀的大小)
* @return {@link BitMatrix}
*/
public static BitMatrix encode(String content, BarcodeFormat format, int width, int height) {
@@ -428,7 +428,7 @@ public class QrCodeUtil {
*
* @param content 文本内容
* @param format 格式枚举
* @param config 二维码配置,包括长、宽、边距、颜色等
* @param config 二维码配置,包括宽度、高度、边距、颜色等
* @return {@link BitMatrix}
* @since 4.1.2
*/
@@ -454,11 +454,11 @@ public class QrCodeUtil {
/**
* 解码二维码或条形码图片为文本
*
* @param qrCodeInputstream 二维码输入流
* @param qrCodeInputStream 二维码输入流
* @return 解码文本
*/
public static String decode(InputStream qrCodeInputstream) {
return decode(ImgUtil.read(qrCodeInputstream));
public static String decode(InputStream qrCodeInputStream) {
return decode(ImgUtil.read(qrCodeInputStream));
}
/**
@@ -549,7 +549,7 @@ public class QrCodeUtil {
* BitMatrix转SVG(字符串)
*
* @param matrix BitMatrix
* @param qrConfig 二维码配置,包括长、宽、边距、颜色等
* @param qrConfig 二维码配置,包括宽度、高度、边距、颜色等
* @return SVG矢量图字符串
* @since 5.8.6
*/
@@ -567,7 +567,7 @@ public class QrCodeUtil {
* @return SVG矢量图字符串
* @since 5.8.6
*/
public static String toSVG(BitMatrix matrix, int foreColor, Integer backColor, Image logoImg, int ratio) {
public static String toSVG(BitMatrix matrix, Integer foreColor, Integer backColor, Image logoImg, int ratio) {
StringBuilder sb = new StringBuilder();
int qrWidth = matrix.getWidth();
int qrHeight = matrix.getHeight();
@@ -575,7 +575,7 @@ public class QrCodeUtil {
for (int y = 0; y < qrHeight; y++) {
for (int x = 0; x < qrWidth; x++) {
if (matrix.get(x, y)) {
sb.append(" M" + x + "," + y + "h1v" + moduleHeight + "h-1z");
sb.append(" M").append(x).append(",").append(y).append("h1v").append(moduleHeight).append("h-1z");
}
}
}
@@ -600,20 +600,23 @@ public class QrCodeUtil {
}
Color fore = new Color(foreColor, true);
StringBuilder result = StrUtil.builder();
result.append("<svg width=\"" + qrWidth + "\" height=\"" + qrHeight + "\" \n");
result.append("<svg width=\"").append(qrWidth).append("\" height=\"").append(qrHeight).append("\" \n");
if (backColor != null) {
Color back = new Color(backColor, true);
result.append("style=\"background-color:rgba(" + back.getRed() + "," + back.getGreen() + "," + back.getBlue() + "," + back.getAlpha() + ")\"\n");
result.append("style=\"background-color:rgba(").append(back.getRed()).append(",").append(back.getGreen()).append(",").append(back.getBlue()).append(",").append(back.getAlpha()).append(")\"\n");
}
result.append("viewBox=\"0 0 " + qrWidth + " " + qrHeight + "\" \n");
result.append("viewBox=\"0 0 ").append(qrWidth).append(" ").append(qrHeight).append("\" \n");
result.append("xmlns=\"http://www.w3.org/2000/svg\" \n");
result.append("xmlns:xlink=\"http://www.w3.org/1999/xlink\" >\n");
result.append("<path d=\"" + sb + "\" stroke=\"rgba(" + fore.getRed() + "," + fore.getGreen() + "," + fore.getBlue() + "," + fore.getAlpha() + ")\" /> \n");
result.append("<path d=\"").append(sb).append("\" ");
if (foreColor!=null){
Color fore = new Color(foreColor, true);
result.append("stroke=\"rgba(").append(fore.getRed()).append(",").append(fore.getGreen()).append(",").append(fore.getBlue()).append(",").append(fore.getAlpha()).append(")\"");
}
result.append(" /> \n");
if (StrUtil.isNotBlank(logoBase64)) {
result.append("<image xlink:href=\"" + logoBase64 + "\" height=\"" + logoHeight + "\" width=\"" + logoWidth + "\" y=\"" + logoY + "\" x=\"" + logoX + "\" />\n");
result.append("<image xlink:href=\"").append(logoBase64).append("\" height=\"").append(logoHeight).append("\" width=\"").append(logoWidth).append("\" y=\"").append(logoY).append("\" x=\"").append(logoX).append("\" />\n");
}
result.append("</svg>");
return result.toString();
@@ -622,7 +625,7 @@ public class QrCodeUtil {
/**
* BitMatrix转ASCII Art字符画形式的二维码
*
* @param bitMatrix
* @param bitMatrix BitMatrix
* @return ASCII Art字符画形式的二维码
* @since 5.8.6
*/
@@ -631,8 +634,8 @@ public class QrCodeUtil {
final int height = bitMatrix.getHeight();
final AnsiElement foreground = qrConfig.foreColor == null ? null : Ansi8BitColor.foreground(rgbToAnsi8BitValue(qrConfig.foreColor));
final AnsiElement background = qrConfig.backColor == null ? null : Ansi8BitColor.background(rgbToAnsi8BitValue(qrConfig.backColor));
final AnsiElement foreground = qrConfig.foreColor == null ? null : rgbToAnsi8BitElement(qrConfig.foreColor, ForeOrBack.FORE);
final AnsiElement background = qrConfig.backColor == null ? null : rgbToAnsi8BitElement(qrConfig.backColor, ForeOrBack.BACK);
StringBuilder builder = new StringBuilder();
for (int i = 0; i <= height; i += 2) {
@@ -655,26 +658,16 @@ public class QrCodeUtil {
return builder.toString();
}
/**
* rgb转Ansi8Bit值
/* *//**
* rgb转AnsiElement
*
* @param rgb rgb颜色值
* @return Ansi8bit颜色值
* @param foreOrBack 前景or背景
* @return AnsiElement
* @since 5.8.6
*/
private static int rgbToAnsi8BitValue(int rgb) {
final int r = (rgb >> 16) & 0xff;
final int g = (rgb >> 8) & 0xff;
final int b = (rgb) & 0xff;
final int l;
if (r == g && g == b) {
final int i = (int) (NumberUtil.div(NumberUtil.mul(r - 10.625, 23), (255 - 10.625), 0));
l = i >= 0 ? 232 + i : 0;
} else {
l = 16 + (int) (36 * NumberUtil.div(NumberUtil.mul(r, 5), 255, 0)) + (int) (6.0 * (g / 256.0 * 6.0)) + (int) (b / 256.0 * 6.0);
}
return l;
private static AnsiElement rgbToAnsi8BitElement(int rgb,ForeOrBack foreOrBack) {
return ansiColors.findClosest(new Color(rgb)).toAnsiElement(foreOrBack);
}

View File

@@ -25,9 +25,19 @@ public class QrConfig {
private static final int BLACK = 0xFF000000;
private static final int WHITE = 0xFFFFFFFF;
/** 宽 */
/**
* 宽度(单位:像素或▄)
* <div>当二维码类型为一般图片或者SVG时单位是像素<div/>
* <div>当二维码类型Ascii Art字符画时单位是字符▄或▀的大小<div/>
*/
protected int width;
/** 长 */
/**
* 高度(单位:像素或▄)
* <div>当二维码类型为一般图片或者SVG时单位是像素<div/>
* <div>当二维码类型Ascii Art字符画时单位是字符▄或▀的大小<div/>
*/
protected int height;
/** 前景色(二维码颜色) */
protected Integer foreColor = BLACK;

View File

@@ -3,11 +3,11 @@ package cn.hutool.extra.qrcode;
import cn.hutool.core.codec.Base64;
import cn.hutool.core.img.ImgUtil;
import cn.hutool.core.io.FileUtil;
import cn.hutool.core.lang.Assert;
import cn.hutool.core.lang.Console;
import com.google.zxing.BarcodeFormat;
import com.google.zxing.datamatrix.encoder.SymbolShapeHint;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;
import org.junit.Assert;
import org.junit.Ignore;
import org.junit.Test;
@@ -29,7 +29,7 @@ public class QrCodeUtilTest {
@Test
public void generateTest() {
final BufferedImage image = QrCodeUtil.generate("https://hutool.cn/", 300, 300);
Assert.assertNotNull(image);
Assert.notNull(image);
}
@Test
@@ -72,7 +72,7 @@ public class QrCodeUtilTest {
@Ignore
public void decodeTest() {
final String decode = QrCodeUtil.decode(FileUtil.file("d:/test/pic/qr.png"));
Console.log(decode);
//Console.log(decode);
}
@Test
@@ -80,13 +80,13 @@ public class QrCodeUtilTest {
public void decodeTest2() {
// 条形码
final String decode = QrCodeUtil.decode(FileUtil.file("d:/test/90.png"));
Console.log(decode);
//Console.log(decode);
}
@Test
public void generateAsBase64Test() {
final String base64 = QrCodeUtil.generateAsBase64("https://hutool.cn/", new QrConfig(400, 400), "png");
Assert.assertNotNull(base64);
Assert.notNull(base64);
}
@Test
@@ -96,27 +96,27 @@ public class QrCodeUtilTest {
new File("d:/test/qr.png"));
final String encode = Base64.encode(bytes);
final String base641 = QrCodeUtil.generateAsBase64("https://hutool.cn/", new QrConfig(400, 400), "png", encode);
Assert.assertNotNull(base641);
Assert.notNull(base641);
}
@Test
public void generateAsBase64Test3() {
final String base64 = QrCodeUtil.generateAsBase64("https://hutool.cn/", new QrConfig(400, 400), "svg");
Assert.assertNotNull(base64);
System.out.println(base64);
Assert.notNull(base64);
//Console.log(base64);
}
@Test
@Ignore
public void decodeTest3() {
final String decode = QrCodeUtil.decode(ImgUtil.read("d:/test/qr_a.png"), false, true);
Console.log(decode);
//Console.log(decode);
}
@Test
public void pdf417Test() {
final BufferedImage image = QrCodeUtil.generate("content111", BarcodeFormat.PDF_417, QrConfig.create());
Assert.assertNotNull(image);
Assert.notNull(image);
}
@Test
@@ -124,11 +124,11 @@ public class QrCodeUtilTest {
final QrConfig qrConfig = QrConfig.create();
qrConfig.setShapeHint(SymbolShapeHint.FORCE_RECTANGLE);
final BufferedImage image = QrCodeUtil.generate("content111", BarcodeFormat.DATA_MATRIX, qrConfig);
Assert.assertNotNull(image);
Assert.notNull(image);
final QrConfig config = QrConfig.create();
config.setShapeHint(SymbolShapeHint.FORCE_SQUARE);
final BufferedImage imageSquare = QrCodeUtil.generate("content111", BarcodeFormat.DATA_MATRIX, qrConfig);
Assert.assertNotNull(imageSquare);
Assert.notNull(imageSquare);
}
@Test
@@ -142,7 +142,7 @@ public class QrCodeUtilTest {
.setErrorCorrection(ErrorCorrectionLevel.M)
.setMargin(1);
final String svg = QrCodeUtil.generateAsSvg("https://hutool.cn/", qrConfig);
Assert.assertNotNull(svg);
Assert.notNull(svg);
FileUtil.writeString(svg, FileUtil.touch("d:/test/hutool_qr.svg"),StandardCharsets.UTF_8);
}
@@ -154,8 +154,8 @@ public class QrCodeUtilTest {
.setWidth(0)
.setHeight(0).setMargin(1);
final String asciiArt = QrCodeUtil.generateAsAsciiArt("https://hutool.cn/",qrConfig);
Assert.assertNotNull(asciiArt);
//Console.log(asciiArt);
Assert.notNull(asciiArt);
Console.log(asciiArt);
}
@Test
@@ -166,7 +166,7 @@ public class QrCodeUtilTest {
.setWidth(0)
.setHeight(0).setMargin(1);
final String asciiArt = QrCodeUtil.generateAsAsciiArt("https://hutool.cn/",qrConfig);
Assert.assertNotNull(asciiArt);
Assert.notNull(asciiArt);
//Console.log(asciiArt);
}
@@ -180,8 +180,8 @@ public class QrCodeUtilTest {
.setWidth(0)
.setHeight(0).setMargin(1);
final File qrFile = QrCodeUtil.generate("https://hutool.cn/", qrConfig, FileUtil.touch("d:/test/ascii_art_qr_code.txt"));
final BufferedReader reader = FileUtil.getReader(qrFile, StandardCharsets.UTF_8);
reader.lines().forEach(System.out::println);
//final BufferedReader reader = FileUtil.getReader(qrFile, StandardCharsets.UTF_8);
//reader.lines().forEach(Console::log);
}
@Test
@@ -198,21 +198,22 @@ public class QrCodeUtilTest {
}catch (final IOException e){
e.printStackTrace();
}
final BufferedReader reader = FileUtil.getReader(filepath, StandardCharsets.UTF_8);
reader.lines().forEach(System.out::println);
//final BufferedReader reader = FileUtil.getReader(filepath, StandardCharsets.UTF_8);
//reader.lines().forEach(Console::log);
}
@Test
@Ignore
public void comparePngAndAsciiArtTest() {
public void comparePngAndSvgAndAsciiArtTest() {
final QrConfig qrConfig = QrConfig.create()
.setForeColor(null)
.setBackColor(null)
.setWidth(0)
.setHeight(0).setMargin(1);
QrCodeUtil.generate("https://hutool.cn", qrConfig, FileUtil.touch("d:/test/compare/default.jpg"));
QrCodeUtil.generate("https://hutool.cn", qrConfig, FileUtil.touch("d:/test/compare/default.txt"));
QrCodeUtil.generate("https://hutool.cn", qrConfig, FileUtil.touch("d:/test/compare/default.png"));
.setBackColor(Color.WHITE)
.setWidth(200)
.setHeight(200).setMargin(1);
QrCodeUtil.generate("https://hutool.cn", qrConfig, FileUtil.touch("d:/test/compare/config_null_color.jpg"));
QrCodeUtil.generate("https://hutool.cn", qrConfig, FileUtil.touch("d:/test/compare/config_null_color.txt"));
QrCodeUtil.generate("https://hutool.cn", qrConfig, FileUtil.touch("d:/test/compare/config_null_color.png"));
QrCodeUtil.generate("https://hutool.cn", qrConfig, FileUtil.touch("d:/test/compare/config_null_color.svg"));
}
}