2013-07-20 4 views
1

Я создал приложение, которое может сканировать QR-код. Он отлично работает со всеми устройствами Android, за исключением Samsung Galaxy s4.
Приложение не сканирует QR-код при использовании устройства Galaxy s4.
Теперь, когда эта Galaxy s4 имеет версию Android 4.2.2, я также проверил свое приложение на других устройствах, имеющих такую ​​же версию Android (4.2.2), что и Nexus-4, и с ней все отлично работает.
Какое-либо другое оборудование, используемое для сканирования QR-кода в Galaxy s4?
Нужна помощь в решении этой странной проблемы!

Ниже приведен код, который я использовал в своем приложении.

CameraManager.javaQR-сканирование Проблема в Samsung Galaxy s4

/** 
* This object wraps the Camera service object and expects to be the only one talking to it. The 
* implementation encapsulates the steps needed to take preview-sized images, which are used for 
* both preview and decoding. 
* 
* @author [email protected] (Daniel Switkin) 
*/ 
public final class CameraManager { 

    private static final String TAG = CameraManager.class.getSimpleName(); 

    private static final int MIN_FRAME_WIDTH = 240; 
    private static final int MIN_FRAME_HEIGHT = 240; 
    private static final int MAX_FRAME_WIDTH = 480; 
    private static final int MAX_FRAME_HEIGHT = 360; 

    private static CameraManager cameraManager; 

    static final int SDK_INT; // Later we can use Build.VERSION.SDK_INT 
    static { 
    int sdkInt; 
    try { 
     sdkInt = Integer.parseInt(Build.VERSION.SDK); 
    } catch (NumberFormatException nfe) { 
     // Just to be safe 
     sdkInt = 10000; 
    } 
    SDK_INT = sdkInt; 
    } 

    private final Context context; 
    private final CameraConfigurationManager configManager; 
    private Camera camera; 
    private Rect framingRect; 
    private Rect framingRectInPreview; 
    private boolean initialized; 
    private boolean previewing; 
    private boolean reverseImage; 
    private final boolean useOneShotPreviewCallback; 
    /** 
    * Preview frames are delivered here, which we pass on to the registered handler. Make sure to 
    * clear the handler so it will only receive one message. 
    */ 
    private final PreviewCallback previewCallback; 

    /** Autofocus callbacks arrive here, and are dispatched to the Handler which requested them. */ 
    private final AutoFocusCallback autoFocusCallback; 

    /** 
    * Initializes this static object with the Context of the calling Activity. 
    * 
    * @param context The Activity which wants to use the camera. 
    */ 
    public static void init(Context context) { 
    if (cameraManager == null) { 
     cameraManager = new CameraManager(context); 
    } 
    } 

    /** 
    * Gets the CameraManager singleton instance. 
    * 
    * @return A reference to the CameraManager singleton. 
    */ 
    public static CameraManager get() { 
    return cameraManager; 
    } 

    private CameraManager(Context context) { 

    this.context = context; 
    this.configManager = new CameraConfigurationManager(context); 

    // Camera.setOneShotPreviewCallback() has a race condition in Cupcake, so we use the older 
    // Camera.setPreviewCallback() on 1.5 and earlier. For Donut and later, we need to use 
    // the more efficient one shot callback, as the older one can swamp the system and cause it 
    // to run out of memory. We can't use SDK_INT because it was introduced in the Donut SDK. 
    useOneShotPreviewCallback = Integer.parseInt(Build.VERSION.SDK) > 3; // 3 = Cupcake 

    previewCallback = new PreviewCallback(configManager, useOneShotPreviewCallback); 
    autoFocusCallback = new AutoFocusCallback(); 
    } 

    /** 
    * Opens the camera driver and initializes the hardware parameters. 
    * 
    * @param holder The surface object which the camera will draw preview frames into. 
    * @throws IOException Indicates the camera driver failed to open. 
    */ 
    public void openDriver(SurfaceHolder holder) throws IOException { 
    if (camera == null) { 
     camera = Camera.open(); 
     if (camera == null) { 
     throw new IOException(); 
     } 
    } 
    camera.setPreviewDisplay(holder); 
    if (!initialized) { 
     initialized = true; 
     configManager.initFromCameraParameters(camera); 
    } 
    configManager.setDesiredCameraParameters(camera); 

    SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context); 
    reverseImage = prefs.getBoolean(PreferencesActivity.KEY_REVERSE_IMAGE, false); 
    if (prefs.getBoolean(PreferencesActivity.KEY_FRONT_LIGHT, false)) { 
     FlashlightManager.enableFlashlight(); 
    } 
    } 

    /** 
    * Closes the camera driver if still in use. 
    */ 
    public void closeDriver() { 
    if (camera != null) { 
     FlashlightManager.disableFlashlight(); 
     camera.release(); 
     camera = null; 

     // Make sure to clear these each time we close the camera, so that any scanning rect 
     // requested by intent is forgotten. 
     framingRect = null; 
     framingRectInPreview = null; 
    } 
    } 

    /** 
    * Asks the camera hardware to begin drawing preview frames to the screen. 
    */ 
    public void startPreview() { 
    if (camera != null && !previewing) { 
     camera.startPreview(); 
     previewing = true; 
    } 
    } 

    /** 
    * Tells the camera to stop drawing preview frames. 
    */ 
    public void stopPreview() { 
    if (camera != null && previewing) { 
     if (!useOneShotPreviewCallback) { 
     camera.setPreviewCallback(null); 
     } 
     camera.stopPreview(); 
     previewCallback.setHandler(null, 0); 
     autoFocusCallback.setHandler(null, 0); 
     previewing = false; 
    } 
    } 

    /** 
    * A single preview frame will be returned to the handler supplied. The data will arrive as byte[] 
    * in the message.obj field, with width and height encoded as message.arg1 and message.arg2, 
    * respectively. 
    * 
    * @param handler The handler to send the message to. 
    * @param message The what field of the message to be sent. 
    */ 
    public void requestPreviewFrame(Handler handler, int message) { 
    if (camera != null && previewing) { 
     previewCallback.setHandler(handler, message); 
     if (useOneShotPreviewCallback) { 
     camera.setOneShotPreviewCallback(previewCallback); 
     } else { 
     camera.setPreviewCallback(previewCallback); 
     } 
    } 
    } 

    /** 
    * Asks the camera hardware to perform an autofocus. 
    * 
    * @param handler The Handler to notify when the autofocus completes. 
    * @param message The message to deliver. 
    */ 
    public void requestAutoFocus(Handler handler, int message) { 
    if (camera != null && previewing) { 
     autoFocusCallback.setHandler(handler, message); 
     //Log.d(TAG, "Requesting auto-focus callback"); 
     camera.autoFocus(autoFocusCallback); 
    } 
    } 

    /** 
    * Calculates the framing rect which the UI should draw to show the user where to place the 
    * barcode. This target helps with alignment as well as forces the user to hold the device 
    * far enough away to ensure the image will be in focus. 
    * 
    * @return The rectangle to draw on screen in window coordinates. 
    */ 
    public Rect getFramingRect() { 
    if (framingRect == null) { 
     if (camera == null) { 
     return null; 
     } 
     Point screenResolution = configManager.getScreenResolution(); 
     int width = screenResolution.x * 3/4; 
     if (width < MIN_FRAME_WIDTH) { 
     width = MIN_FRAME_WIDTH; 
     } else if (width > MAX_FRAME_WIDTH) { 
     width = MAX_FRAME_WIDTH; 
     } 
     int height = screenResolution.y * 3/4; 
     if (height < MIN_FRAME_HEIGHT) { 
     height = MIN_FRAME_HEIGHT; 
     } else if (height > MAX_FRAME_HEIGHT) { 
     height = MAX_FRAME_HEIGHT; 
     } 
     int leftOffset = (screenResolution.x - width)/2; 
     int topOffset = (screenResolution.y - height)/2; 
     framingRect = new Rect(leftOffset, topOffset, leftOffset + width, topOffset + height); 
     Log.d(TAG, "Calculated framing rect: " + framingRect); 
    } 
    return framingRect; 
    } 

    /** 
    * Like {@link #getFramingRect} but coordinates are in terms of the preview frame, 
    * not UI/screen. 
    */ 
    public Rect getFramingRectInPreview() { 
    if (framingRectInPreview == null) { 
     Rect rect = new Rect(getFramingRect()); 
     Point cameraResolution = configManager.getCameraResolution(); 
     Point screenResolution = configManager.getScreenResolution(); 

     /* updated to allow for portrait instead of landscape 
     rect.left = rect.left * cameraResolution.y/screenResolution.x; 
     rect.right = rect.right * cameraResolution.y/screenResolution.x; 
     rect.top = rect.top * cameraResolution.x/screenResolution.y; 
     rect.bottom = rect.bottom * cameraResolution.x/screenResolution.y; */ 

     rect.left = rect.left * cameraResolution.x/screenResolution.x; 
     rect.right = rect.right * cameraResolution.x/screenResolution.x; 
     rect.top = rect.top * cameraResolution.y/screenResolution.y; 
     rect.bottom = rect.bottom * cameraResolution.y/screenResolution.y; 
     framingRectInPreview = rect; 
    } 
    return framingRectInPreview; 
    } 

    /** 
    * Allows third party apps to specify the scanning rectangle dimensions, rather than determine 
    * them automatically based on screen resolution. 
    * 
    * @param width The width in pixels to scan. 
    * @param height The height in pixels to scan. 
    */ 
    public void setManualFramingRect(int width, int height) { 
    Point screenResolution = configManager.getScreenResolution(); 
    if (width > screenResolution.x) { 
     width = screenResolution.x; 
    } 
    if (height > screenResolution.y) { 
     height = screenResolution.y; 
    } 
    int leftOffset = (screenResolution.x - width)/2; 
    int topOffset = (screenResolution.y - height)/2; 
    framingRect = new Rect(leftOffset, topOffset, leftOffset + width, topOffset + height); 
    Log.d(TAG, "Calculated manual framing rect: " + framingRect); 
    framingRectInPreview = null; 
    } 

    /** 
    * A factory method to build the appropriate LuminanceSource object based on the format 
    * of the preview buffers, as described by Camera.Parameters. 
    * 
    * @param data A preview frame. 
    * @param width The width of the image. 
    * @param height The height of the image. 
    * @return A PlanarYUVLuminanceSource instance. 
    */ 
    public PlanarYUVLuminanceSource buildLuminanceSource(byte[] data, int width, int height) { 
    Rect rect = getFramingRectInPreview(); 
    int previewFormat = configManager.getPreviewFormat(); 
    String previewFormatString = configManager.getPreviewFormatString(); 

    switch (previewFormat) { 
     // This is the standard Android format which all devices are REQUIRED to support. 
     // In theory, it's the only one we should ever care about. 
     case PixelFormat.YCbCr_420_SP: 
     // This format has never been seen in the wild, but is compatible as we only care 
     // about the Y channel, so allow it. 
     case PixelFormat.YCbCr_422_SP: 
     return new PlanarYUVLuminanceSource(data, width, height, rect.left, rect.top, 
      rect.width(), rect.height(), reverseImage); 
     default: 
     // The Samsung Moment incorrectly uses this variant instead of the 'sp' version. 
     // Fortunately, it too has all the Y data up front, so we can read it. 
     if ("yuv420p".equals(previewFormatString)) { 
      return new PlanarYUVLuminanceSource(data, width, height, rect.left, rect.top, 
       rect.width(), rect.height(), reverseImage); 
     } 
    } 
    throw new IllegalArgumentException("Unsupported picture format: " + 
     previewFormat + '/' + previewFormatString); 
    } 

} 



PreviewCallback.java

final class PreviewCallback implements Camera.PreviewCallback { 

    private static final String TAG = PreviewCallback.class.getSimpleName(); 

    private final CameraConfigurationManager configManager; 
    private final boolean useOneShotPreviewCallback; 
    private Handler previewHandler; 
    private int previewMessage; 

    PreviewCallback(CameraConfigurationManager configManager, boolean useOneShotPreviewCallback) { 
    this.configManager = configManager; 
    this.useOneShotPreviewCallback = useOneShotPreviewCallback; 
    } 

    void setHandler(Handler previewHandler, int previewMessage) { 
    this.previewHandler = previewHandler; 
    this.previewMessage = previewMessage; 
    } 

    public void onPreviewFrame(byte[] data, Camera camera) { 
    Point cameraResolution = configManager.getCameraResolution(); 
    if (!useOneShotPreviewCallback) { 
     camera.setPreviewCallback(null); 
    } 
    if (previewHandler != null) { 
     Message message = previewHandler.obtainMessage(previewMessage, cameraResolution.x, 
      cameraResolution.y, data); 
     message.sendToTarget(); 
     previewHandler = null; 
    } else { 
     Log.d(TAG, "Got preview callback, but no handler for it"); 
    } 
    } 

} 
+0

Если он работает на «другом устройстве», он может быть «слишком локализован». Я мог бы предложить заглянуть в форумы Galaxy. – ChiefTwoPencils

+0

@BobbyDigital .. я уже прошел через форумы и другие сайты Galaxy, но не нашел никакой помощи по этой проблеме. –

ответ

3

Вы должны установить формат просмотра камеры, прежде чем перейти в режим предварительного просмотра. Да, в документации на Android говорится, что формат по умолчанию должен быть PixelFormat.YCbCr_420_SP = 17, но я видел некоторые устройства, которые не следуют этому правилу. И некоторые используютyuyv упакованный формат 422, который не имеет блока яркости для копирования, как это делает ваш код, но требует перестановки байтов. Было бы также целесообразно установить ширину и высоту предварительного просмотра. Скорее всего, вам не нужно больше, чем разрешение VGA для сканирования QR-кодов.

Обновление: на моем устройстве, рамка предварительного просмотра по умолчанию - 1920x1080 пикселей. Обратите внимание, что это соотношение сторон 16: 9, а не 4: 3. Я вижу в вашем коде некоторую ссылку на hardcoded 4: 3, но если это предположение используется для интерпретации QR-кода, это само по себе может быть причиной неудачи.

+0

да, вы правы, в моем коде соотношение сторон 4: 3 и Galaxy s4 требует 16: 9 ... это может быть причиной неудачи. Итак, как мы можем установить его динамически? –

+2

Открытая камера, getParameters, setPreviewSize 640x480, setParameters, startPreview, прибыль! –

+1

Thnx alex возникла проблема из-за такого жесткого разрешения, я изменил его и установил разрешение по умолчанию для устройства, так что теперь он работает .. и теперь поймайте вашу прибыль :) ... но все же мне пришлось удерживать телефон подальше от QR-код для сканирования ... Возможно ли здесь управлять фокусирующей системой камеры? –

Смежные вопросы