我正在尝试使用以下库读取QR码:
但是,它似乎并不顺利。几个小时后,我仍然无法读取像样的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);
}
}
答案 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