使用ARKit和ZXing在Unity中阅读QR代码

时间:2018-11-22 13:10:50

标签: c# unity3d zxing arkit

我正在尝试使用以下库读取QR码:

  • ARKit
  • Zxing

但是,它似乎并不顺利。几个小时后,我仍然无法读取像样的QR码。 调试时,我应用纹理以查看结果。由于纹理Y,它看起来是红色的,但除此之外它显示的是QR码。 解释纹理不会返回ZXing分析的任何数据。

这是我正在使用的以下代码:

#if UNITY_IOS && !UNITY_EDITOR
    // Update is called once per frame
    // BETTER: InvokeRepeating
    void Update()
    {
        if (!done) {
            ARTextureHandles handles = arSession.GetARVideoTextureHandles();
            //ARTextureHandles handles = UnityARSessionNativeInterface.GetARSessionNativeInterface().GetARVideoTextureHandles();
            if (handles.IsNull())
            {
                return;
            }
            if (handles.TextureY != System.IntPtr.Zero) {
                ReadQRCode (handles.TextureY);
            }
        }

}
#endif

private void ReadQRCode(System.IntPtr mtlTexPtr)
{
    Debug.Log("---------------");
    Debug.Log("Scanning...");

    Resolution currentResolution = Screen.currentResolution;

    tex = (UnityEngine.Texture2D)GameObject.Find("Camera").GetComponent<UnityARVideo>().m_ClearMaterial.GetTexture("_textureCbCr");

    tex.UpdateExternalTexture(mtlTexPtr);

    try
    {
        if(barCodeReader == null) {
            Debug.Log("Could not find barcorereader");
        }
        if(tex == null) {
            Debug.Log("Could not find texture");
        }

        var data = barCodeReader.Decode(tex.GetPixels32(), currentResolution.width, currentResolution.height);
        if (data != null)
        {
            Debug.Log("QR: " + data.Text);
        }
        else
        {
            Debug.Log("NO QR: " + "No QR code detected !");
        }
    }
    catch (Exception e)
    {
        Debug.LogError("Error reading QR");
        Debug.LogError(e.Message);
    }
}

2 个答案:

答案 0 :(得分:1)

我曾经与ZXing和ARkit一起工作,但从未合作过。

试图使某些事情起作用,至少可以提示您可能发生的事情。

作为ARCameraManager的子代,添加另一个(附加到主AR相机上)带有ARVideo脚本的相机(删除音频侦听器,Flare层和guiLayer之类的东西)。此外,还添加以下脚本检索renderTexture:

[RequireComponent(typeof(Camera))]
public class WebcamFetcher : MonoBehaviour
{
private RenderTexture _vertical;
private RenderTexture _horizontal;
private Camera _camera;

// Update is called once per frame
public RenderTexture RenderTexture
{
    get
    {
        var orientation = Screen.orientation;
        if (orientation == ScreenOrientation.Landscape || orientation == ScreenOrientation.LandscapeLeft || orientation == ScreenOrientation.LandscapeRight)
        {
            return _horizontal;
        }
        else
        {
            return _vertical;
        }
    }
}

// Use this for initialization
void Start ()
{
    _camera = GetComponent<Camera>();
    _horizontal = new RenderTexture(Screen.width, Screen.height, 24);
    _vertical = new RenderTexture(Screen.height, Screen.width, 24);
}

// Update is called once per frame
void Update()
{
    var orientation = Screen.orientation;
    if (orientation == ScreenOrientation.Landscape || orientation == ScreenOrientation.LandscapeLeft || orientation == ScreenOrientation.LandscapeRight)
    {
        _camera.targetTexture = _horizontal;
    }
    else
    {
        _camera.targetTexture = _vertical;
    }
}

}

然后使用脚本,例如:

private string DecodeQR(Color32[] pixels, int width, int height)
{
    try
    {
        IBarcodeReader barcodeReader = new BarcodeReader();

        // decode the current frame
        var result = barcodeReader.Decode(pixels, width, height);

        if (result != null)
        {
            return result.Text;
        }
    }
    catch (Exception ex) { Debug.LogError(ex.Message); }
    return null;
}

[SerializeField] Text QrDisplay; // A text field to display detected QRs to

public void OnQrDetect() // a callback for a UI button
{
    var texture = new Texture2D(_webcamFetcher.RenderTexture.width, _webcamFetcher.RenderTexture.height);
    RenderTexture.active = _webcamFetcher.RenderTexture;
    texture.ReadPixels(new Rect(Vector2.zero, new Vector2(_webcamFetcher.RenderTexture.width, _webcamFetcher.RenderTexture.height)), 0, 0);

    var qrText = DecodeQR(texture.GetPixels32(), _webcamFetcher.RenderTexture.width, _webcamFetcher.RenderTexture.height);
    if (qrText != null)
    {
        QrDisplay.text = qrText;
    }
}

为我工作。纹理检索可能不是最有效的一种。 但是有效->有可能。

答案 1 :(得分:0)

进一步研究之后,我遇到了一个与OpenCV类似的示例。

事实证明,该方法可以为我带来良好的速度。

public class FrameCapturer : MonoBehaviour {

    // Script Inputs
    public bool m_shouldCaptureOnNextFrame = false;
    public Color32[] m_lastCapturedColors;

    // Privates
    Texture2D m_centerPixTex;

    void Start()
    {
        Resolution currentResolution = Screen.currentResolution;
        m_centerPixTex = new Texture2D(currentResolution.width, currentResolution.height, TextureFormat.RGBA32, false);
    }

    void OnPostRender()
    {

        if (m_shouldCaptureOnNextFrame)
        {
            Resolution res = Screen.currentResolution;
            m_lastCapturedColors = GetRenderedColors();
            m_shouldCaptureOnNextFrame = false;
        }
    }

    // Helpers
    Color32[] GetRenderedColors()
    {
        Resolution currentResolution = Screen.currentResolution;
        m_centerPixTex.ReadPixels(new Rect(0, 0, currentResolution.width, currentResolution.height), 0, 0);
        m_centerPixTex.Apply();

        return m_centerPixTex.GetPixels32();
    }
}

我将其附加到还附加了AR脚本的主摄像机上。然后,在我的二维码阅读器中,我可以简单地使用以下内容:

public class QRCodeReader : MonoBehaviour
{
    public Camera cam;
    private BarcodeReader barCodeReader;

    FrameCapturer m_pixelCapturer;

    // Use this for initialization
    void Start()
    {
        barCodeReader = new BarcodeReader();
        Resolution currentResolution = Screen.currentResolution;
        m_pixelCapturer = cam.GetComponent<FrameCapturer>();
    }

    void Update()
    {

        Resolution currentResolution = Screen.currentResolution;

        try
        {
            Color32[] framebuffer = m_pixelCapturer.m_lastCapturedColors;
            if (framebuffer.Length == 0)
            {
                return;
            }

            var data = barCodeReader.Decode(framebuffer, currentResolution.width, currentResolution.height);
            if (data != null)
            {
                // QRCode detected.
                Debug.Log(data);
                Debug.Log("QR: " + data.Text);

                //OnQrCodeRead(new QrCodeReadEventArgs() { text = data.Text });
            }

        }
        catch (Exception e)
        {
            Debug.LogError("Error reading QR");
            Debug.LogError(e.Message);
        }

        // skip 1 frame each time 
        // solves GetPixels() blocks for ReadPixels() to complete
        // https://medium.com/google-developers/real-time-image-capture-in-unity-458de1364a4c
        m_pixelCapturer.m_shouldCaptureOnNextFrame = true;

    }
}

我将答案修改为以下内容的原始来源:https://github.com/realityenhanced/ARKitExperiments/blob/master/Assets/Scripts/CaptureCenterPixel.cs