package com.google.zxing.client.android;
+import com.google.zxing.BarcodeFormat;
+import com.google.zxing.BinaryBitmap;
+import com.google.zxing.DecodeHintType;
+import com.google.zxing.MultiFormatReader;
+import com.google.zxing.ReaderException;
+import com.google.zxing.Result;
+import com.google.zxing.ResultPointCallback;
+import com.google.zxing.common.GlobalHistogramBinarizer;
+
import android.content.SharedPreferences;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.preference.PreferenceManager;
-import com.google.zxing.BarcodeFormat;
-import com.google.zxing.DecodeHintType;
-import com.google.zxing.MultiFormatReader;
-import com.google.zxing.ReaderException;
-import com.google.zxing.Result;
+import android.util.Log;
-import java.util.Date;
import java.util.Hashtable;
import java.util.Vector;
/**
* This thread does all the heavy lifting of decoding the images.
+ *
+ * @author dswitkin@google.com (Daniel Switkin)
*/
final class DecodeThread extends Thread {
-
public static final String BARCODE_BITMAP = "barcode_bitmap";
+ private static final String TAG = "DecodeThread";
- public Handler mHandler;
- private final CaptureActivity mActivity;
- private final MultiFormatReader mMultiFormatReader;
+ private Handler handler;
+ private final CaptureActivity activity;
+ private final MultiFormatReader multiFormatReader;
+ private final ResultPointCallback resultPointCallback;
- DecodeThread(CaptureActivity activity, String mode) {
- mActivity = activity;
- mMultiFormatReader = new MultiFormatReader();
+ DecodeThread(CaptureActivity activity, String mode, ResultPointCallback resultPointCallback) {
+ this.activity = activity;
+ multiFormatReader = new MultiFormatReader();
+ this.resultPointCallback = resultPointCallback;
// The prefs can't change while the thread is running, so pick them up once here.
if (mode == null || mode.length() == 0) {
}
}
+ Handler getHandler() {
+ return handler;
+ }
+
@Override
public void run() {
Looper.prepare();
- mHandler = new Handler() {
+ handler = new Handler() {
+ @Override
public void handleMessage(Message message) {
switch (message.what) {
case R.id.decode:
}
private void setDecodeProductMode() {
- Hashtable<DecodeHintType, Object> hints = new Hashtable<DecodeHintType, Object>(3);
- Vector<BarcodeFormat> vector = new Vector<BarcodeFormat>();
- vector.addElement(BarcodeFormat.UPC_A);
- vector.addElement(BarcodeFormat.UPC_E);
- vector.addElement(BarcodeFormat.EAN_13);
- vector.addElement(BarcodeFormat.EAN_8);
- hints.put(DecodeHintType.POSSIBLE_FORMATS, vector);
- mMultiFormatReader.setHints(hints);
+ doSetDecodeMode(BarcodeFormat.UPC_A,
+ BarcodeFormat.UPC_E,
+ BarcodeFormat.EAN_13,
+ BarcodeFormat.EAN_8);
}
- // TODO: This is fragile in case we add new formats. It would be better to have a new enum
- // value which represented all 1D formats.
+ /**
+ * Select the 1D formats we want this client to decode by hand.
+ */
private void setDecode1DMode() {
- Hashtable<DecodeHintType, Object> hints = new Hashtable<DecodeHintType, Object>(3);
- Vector<BarcodeFormat> vector = new Vector<BarcodeFormat>();
- vector.addElement(BarcodeFormat.UPC_A);
- vector.addElement(BarcodeFormat.UPC_E);
- vector.addElement(BarcodeFormat.EAN_13);
- vector.addElement(BarcodeFormat.EAN_8);
- vector.addElement(BarcodeFormat.CODE_39);
- vector.addElement(BarcodeFormat.CODE_128);
- hints.put(DecodeHintType.POSSIBLE_FORMATS, vector);
- mMultiFormatReader.setHints(hints);
+ doSetDecodeMode(BarcodeFormat.UPC_A,
+ BarcodeFormat.UPC_E,
+ BarcodeFormat.EAN_13,
+ BarcodeFormat.EAN_8,
+ BarcodeFormat.CODE_39,
+ BarcodeFormat.CODE_128,
+ BarcodeFormat.ITF);
}
private void setDecodeQRMode() {
- Hashtable<DecodeHintType, Object> hints = new Hashtable<DecodeHintType, Object>(3);
- Vector<BarcodeFormat> vector = new Vector<BarcodeFormat>();
- vector.addElement(BarcodeFormat.QR_CODE);
- hints.put(DecodeHintType.POSSIBLE_FORMATS, vector);
- mMultiFormatReader.setHints(hints);
+ doSetDecodeMode(BarcodeFormat.QR_CODE);
}
+ /**
+ * Instead of calling setHints(null), which would allow new formats to sneak in, we
+ * explicitly set which formats are available.
+ */
private void setDecodeAllMode() {
- mMultiFormatReader.setHints(null);
+ doSetDecodeMode(BarcodeFormat.UPC_A,
+ BarcodeFormat.UPC_E,
+ BarcodeFormat.EAN_13,
+ BarcodeFormat.EAN_8,
+ BarcodeFormat.CODE_39,
+ BarcodeFormat.CODE_128,
+ BarcodeFormat.ITF,
+ BarcodeFormat.QR_CODE);
+ }
+
+ private void doSetDecodeMode(BarcodeFormat... formats) {
+ Hashtable<DecodeHintType, Object> hints = new Hashtable<DecodeHintType, Object>(3);
+ Vector<BarcodeFormat> vector = new Vector<BarcodeFormat>(formats.length);
+ for (BarcodeFormat format : formats) {
+ vector.addElement(format);
+ }
+ hints.put(DecodeHintType.POSSIBLE_FORMATS, vector);
+ hints.put(DecodeHintType.NEED_RESULT_POINT_CALLBACK, resultPointCallback);
+ multiFormatReader.setHints(hints);
}
/**
* @param height The height of the preview frame.
*/
private void decode(byte[] data, int width, int height) {
- Date startDate = new Date();
- boolean success;
+ long start = System.currentTimeMillis();
Result rawResult = null;
- YUVMonochromeBitmapSource source = new YUVMonochromeBitmapSource(data, width, height,
- CameraManager.get().getFramingRect());
+ PlanarYUVLuminanceSource source = CameraManager.get().buildLuminanceSource(data, width, height);
+ BinaryBitmap bitmap = new BinaryBitmap(new GlobalHistogramBinarizer(source));
try {
- rawResult = mMultiFormatReader.decodeWithState(source);
- success = true;
- } catch (ReaderException e) {
- success = false;
+ rawResult = multiFormatReader.decodeWithState(bitmap);
+ } catch (ReaderException re) {
+ // continue
}
- Date endDate = new Date();
- if (success) {
- Message message = Message.obtain(mActivity.mHandler, R.id.decode_succeeded, rawResult);
- message.arg1 = (int) (endDate.getTime() - startDate.getTime());
+ if (rawResult != null) {
+ long end = System.currentTimeMillis();
+ Log.v(TAG, "Found barcode (" + (end - start) + " ms):\n" + rawResult.toString());
+ Message message = Message.obtain(activity.getHandler(), R.id.decode_succeeded, rawResult);
Bundle bundle = new Bundle();
- bundle.putParcelable(BARCODE_BITMAP, source.renderToBitmap());
+ bundle.putParcelable(BARCODE_BITMAP, source.renderCroppedGreyscaleBitmap());
message.setData(bundle);
message.sendToTarget();
} else {
- Message message = Message.obtain(mActivity.mHandler, R.id.decode_failed);
- message.arg1 = (int) (endDate.getTime() - startDate.getTime());
+ Message message = Message.obtain(activity.getHandler(), R.id.decode_failed);
message.sendToTarget();
}
}
-
}