如何在Gluon中使用图像叠加生成二维码?

问题描述 投票:1回答:2

我想生成QR CODE,必须将徽标放在中间。我已经检查了zxing库,并通过阅读此代码(How to generate QR code with logo inside it?)使用Java应用程序进行了此操作。但据我所知,我们不能在android / ios中使用swing。那么如何处理呢?

java gluon gluon-mobile
2个回答
3
投票

基于您所指的link,可以接受的答案指向此post,您可以在其中看到生成QR的技巧,该QR可以隐藏其中心部分而不影响QR扫描仪的可读性。通过增加错误level来完成:

30%(H)的错误校正是级别为H的错误校正,即使被遮盖了30%,QRCode仍然有效

作为此question的后续,您可以在编码文本时仅包含提示:

public Image generateQR(int width, int height) {
    File root = Services.get(StorageService.class)
            .flatMap(StorageService::getPrivateStorage)
            .orElseThrow(() -> new RuntimeException("Storage Service not found"));

    String uuid = Services.get(DeviceService.class)
            .map(DeviceService::getUuid)
            .orElse("123456789");

    MultiFormatWriter codeWriter = new MultiFormatWriter();

    // Add maximum correction
    Map<EncodeHintType, ErrorCorrectionLevel> hints = new HashMap<>();
    hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);

    try {
        BarcodeFormat format = BarcodeFormat.QR_CODE;
        BitMatrix bitMatrix = codeWriter.encode(uuid, format, 
               width, height, hints);  // <--- add hints
        ...
        return writableImage;
    }
}

您将获得一个QR码图像,没有覆盖层。

下一步是不使用Swing将该图像与徽标图像合并。但是实际上我们不需要这样做:我们可以只使用两个ImageView节点!

ImageView imageView = new ImageView();
imageView.setFitWidth(256);
imageView.setFitHeight(256);
imageView.setImage(service.generateQR(256, 256));

ImageView overlay = new ImageView(
     new Image(getClass().getResourceAsStream("/icon.png")));
overlay.setFitWidth(80);
overlay.setFitHeight(80);

StackPane combinedQR = new StackPane(imageView, overlay);
...

生成的代码仍然可读。

您还可以处理结果,为图像添加混合效果,例如:

overlay.setBlendMode(BlendMode.ADD);

但这取决于您的徽标图像与QR融合的方式。

最后,如果仍然需要单个组合图像,则可以创建堆栈窗格的快照:

WritableImage snapshot = combinedQR.snapshot(new SnapshotParameters(), null);

0
投票

受José答案的启发,我在Kotlin中为Android设备编写了自己的解决方案。首先,将ZXing添加到您的项目中:

implementation "com.google.zxing:core:3.4.0"

创建一个虚拟的可绘制对象并将其放在您的res/drawable文件夹中:

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
    android:shape="oval">
    <solid android:color="@color/colorAccent" />
    <size
        android:width="48dp"
        android:height="48dp" />
</shape>

然后将以下扩展名添加到您的项目中:

String.encodeAsQrCodeBitmap将字符串转换为QR码位图。它还接受overlayBitmap,如果为null,则仅将String转换为QR码,并且您可以自定义QR码的颜色。

Bitmap.addOverlayToCenter将两个位图合并为一个,并将叠加位图放置在中心。

[Int.dpToPx()将DP转换为像素。

@Throws(WriterException::class)
fun String.encodeAsQrCodeBitmap(
    dimension: Int,
    overlayBitmap: Bitmap? = null,
    @ColorInt color1: Int = Color.BLACK,
    @ColorInt color2: Int = Color.WHITE
): Bitmap? {

    val result: BitMatrix
    try {
        result = MultiFormatWriter().encode(
            this,
            BarcodeFormat.QR_CODE,
            dimension,
            dimension,
            hashMapOf(EncodeHintType.ERROR_CORRECTION to ErrorCorrectionLevel.H)
        )
    } catch (e: IllegalArgumentException) {
        // Unsupported format
        return null
    }

    val w = result.width
    val h = result.height
    val pixels = IntArray(w * h)
    for (y in 0 until h) {
        val offset = y * w
        for (x in 0 until w) {
            pixels[offset + x] = if (result.get(x, y)) color1 else color2
        }
    }
    val bitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888)
    bitmap.setPixels(pixels, 0, dimension, 0, 0, w, h)

    return if (overlayBitmap != null) {
        bitmap.addOverlayToCenter(overlayBitmap)
    } else {
        bitmap
    }
}

fun Bitmap.addOverlayToCenter(overlayBitmap: Bitmap): Bitmap {

    val bitmap2Width = overlayBitmap.width
    val bitmap2Height = overlayBitmap.height
    val marginLeft = (this.width * 0.5 - bitmap2Width * 0.5).toFloat()
    val marginTop = (this.height * 0.5 - bitmap2Height * 0.5).toFloat()
    val canvas = Canvas(this)
    canvas.drawBitmap(this, Matrix(), null)
    canvas.drawBitmap(overlayBitmap, marginLeft, marginTop, null)
    return this
}

fun Int.dpToPx(): Int {
    return (this * Resources.getSystem().displayMetrics.density).toInt()
}

然后在Fragment / Activity中执行以下操作:从资源中获取叠加层,将String转换为QR码并在ImageView中显示它:

try {
     val displayMetrics = DisplayMetrics()
     activity?.windowManager?.defaultDisplay?.getMetrics(displayMetrics)

     val size = displayMetrics.widthPixels.coerceAtMost(displayMetrics.heightPixels)

     val overlay = ContextCompat.getDrawable(requireContext(), R.drawable.dummy_round)
                ?.toBitmap(72.dpToPx(), 72.dpToPx())

     val bitmap = "https://www.example.com".encodeAsQrCodeBitmap(size, overlay)

     imageView.setImageBitmap(bitmap)
} catch (e: Exception) {
      // handle Errors here
}

结果:

enter image description here

© www.soinside.com 2019 - 2024. All rights reserved.