Сканирование штрих-кода с изображения в галерее Android
Я создаю проект Android, основная функция сканирования штрих-кода.
Я пытался интегрировать с библиотекой Zxing в мой проект, и он работает нормально.
Тем не менее, кажется, не поддерживает сканирование штрих-кода из доступного изображения в галерее устройств Android.
Как я могу это сделать? или с другой библиотекой штрих-кодов?
Пожалуйста, помогите мне.
Спасибо за все ответы.
2 ответа
Вы можете использовать этот класс MultiFormatReader из библиотеки ZXing.
Вы должны получить изображение галереи в BitMap и преобразовать его так:
Bitmap bMap = [...];
String contents = null;
int[] intArray = new int[bMap.getWidth()*bMap.getHeight()];
//copy pixel data from the Bitmap into the 'intArray' array
bMap.getPixels(intArray, 0, bMap.getWidth(), 0, 0, bMap.getWidth(), bMap.getHeight());
LuminanceSource source = new RGBLuminanceSource(bMap.getWidth(), bMap.getHeight(), intArray);
BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(source));
Reader reader = new MultiFormatReader();
Result result = reader.decode(bitmap);
contents = result.getText();
Update1
Чтобы манипулировать большим изображением, пожалуйста, посмотрите на:
https://developer.android.com/training/articles/memory.html
https://developer.android.com/training/displaying-bitmaps/manage-memory.html
Вы можете использовать это свойство android:largeHeap для увеличения размера кучи.
У меня есть рабочий пример того, как это реализовать, если вы читаете в 2016 году, вот как я это сделал:
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
//initialize variables to make them global
private ImageButton Scan;
private static final int SELECT_PHOTO = 100;
//for easy manipulation of the result
public String barcode;
//call oncreate method
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//cast neccesary variables to their views
Scan = (ImageButton)findViewById(R.id.ScanBut);
//set a new custom listener
Scan.setOnClickListener(this);
//launch gallery via intent
Intent photoPic = new Intent(Intent.ACTION_PICK);
photoPic.setType("image/*");
startActivityForResult(photoPic, SELECT_PHOTO);
}
//do necessary coding for each ID
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.ScanBut:
//launch gallery via intent
Intent photoPic = new Intent(Intent.ACTION_PICK);
photoPic.setType("image/*");
startActivityForResult(photoPic, SELECT_PHOTO);
break;
}
}
//call the onactivity result method
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent imageReturnedIntent) {
super.onActivityResult(requestCode, resultCode, imageReturnedIntent);
switch (requestCode) {
case SELECT_PHOTO:
if (resultCode == RESULT_OK) {
//doing some uri parsing
Uri selectedImage = imageReturnedIntent.getData();
InputStream imageStream = null;
try {
//getting the image
imageStream = getContentResolver().openInputStream(selectedImage);
} catch (FileNotFoundException e) {
Toast.makeText(getApplicationContext(), "File not found", Toast.LENGTH_SHORT).show();
e.printStackTrace();
}
//decoding bitmap
Bitmap bMap = BitmapFactory.decodeStream(imageStream);
Scan.setImageURI(selectedImage);// To display selected image in image view
int[] intArray = new int[bMap.getWidth() * bMap.getHeight()];
// copy pixel data from the Bitmap into the 'intArray' array
bMap.getPixels(intArray, 0, bMap.getWidth(), 0, 0, bMap.getWidth(),
bMap.getHeight());
LuminanceSource source = new RGBLuminanceSource(bMap.getWidth(),
bMap.getHeight(), intArray);
BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(source));
Reader reader = new MultiFormatReader();// use this otherwise
// ChecksumException
try {
Hashtable<DecodeHintType, Object> decodeHints = new Hashtable<DecodeHintType, Object>();
decodeHints.put(DecodeHintType.TRY_HARDER, Boolean.TRUE);
decodeHints.put(DecodeHintType.PURE_BARCODE, Boolean.TRUE);
Result result = reader.decode(bitmap, decodeHints);
//*I have created a global string variable by the name of barcode to easily manipulate data across the application*//
barcode = result.getText().toString();
//do something with the results for demo i created a popup dialog
if(barcode!=null){
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("Scan Result");
builder.setIcon(R.mipmap.ic_launcher);
builder.setMessage("" + barcode);
AlertDialog alert1 = builder.create();
alert1.setButton(DialogInterface.BUTTON_POSITIVE, "Done", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Intent i = new Intent (getBaseContext(),MainActivity.class);
startActivity(i);
}
});
alert1.setCanceledOnTouchOutside(false);
alert1.show();}
else
{
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("Scan Result");
builder.setIcon(R.mipmap.ic_launcher);
builder.setMessage("Nothing found try a different image or try again");
AlertDialog alert1 = builder.create();
alert1.setButton(DialogInterface.BUTTON_POSITIVE, "Done", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Intent i = new Intent (getBaseContext(),MainActivity.class);
startActivity(i);
}
});
alert1.setCanceledOnTouchOutside(false);
alert1.show();
}
//the end of do something with the button statement.
} catch (NotFoundException e) {
Toast.makeText(getApplicationContext(), "Nothing Found", Toast.LENGTH_SHORT).show();
e.printStackTrace();
} catch (ChecksumException e) {
Toast.makeText(getApplicationContext(), "Something weird happen, i was probably tired to solve this issue", Toast.LENGTH_SHORT).show();
e.printStackTrace();
} catch (FormatException e) {
Toast.makeText(getApplicationContext(), "Wrong Barcode/QR format", Toast.LENGTH_SHORT).show();
e.printStackTrace();
} catch (NullPointerException e) {
Toast.makeText(getApplicationContext(), "Something weird happen, i was probably tired to solve this issue", Toast.LENGTH_SHORT).show();
e.printStackTrace();
}
}
}
}
}
Сначала, конечно, прочтите изображение из галереи (это может быть в вашей деятельности): Help by
Intent pickIntent = new Intent(Intent.ACTION_PICK);
pickIntent.setDataAndType( android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
startActivityForResult(pickIntent, 111);
После этого просто получите uri изображения в результате активности, и ZXing сотворит чудеса:
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
switch (requestCode) {
//the case is because you might be handling multiple request codes here
case 111:
if(data == null || data.getData()==null) {
Log.e("TAG", "The uri is null, probably the user cancelled the image selection process using the back button.");
return;
}
Uri uri = data.getData();
try
{
InputStream inputStream = getContentResolver().openInputStream(uri);
Bitmap bitmap = BitmapFactory.decodeStream(inputStream);
if (bitmap == null)
{
Log.e("TAG", "uri is not a bitmap," + uri.toString());
return;
}
int width = bitmap.getWidth(), height = bitmap.getHeight();
int[] pixels = new int[width * height];
bitmap.getPixels(pixels, 0, width, 0, 0, width, height);
bitmap.recycle();
bitmap = null;
RGBLuminanceSource source = new RGBLuminanceSource(width, height, pixels);
BinaryBitmap bBitmap = new BinaryBitmap(new HybridBinarizer(source));
MultiFormatReader reader = new MultiFormatReader();
try
{
Result result = reader.decode(bBitmap);
Toast.makeText(this, "The content of the QR image is: " + result.getText(), Toast.LENGTH_SHORT).show();
}
catch (NotFoundException e)
{
Log.e("TAG", "decode exception", e);
}
}
catch (FileNotFoundException e)
{
Log.e("TAG", "can not open file" + uri.toString(), e);
}
break;
}
}
Я в промежутках между таким же сценарием, где бросить NotFoundException, официальный документ говорит
Выдается, когда штрих-код не был найден на изображении. Это могло быть частично обнаружено, но не могло быть подтверждено.
Решение первого уровня в некоторой степени ответ @Laurent работал почти для каждого образца, который у меня был, но не для некоторых.
Решение следующего уровня, добавляющее decodeHints до reader.decode(..), предложенное @Lucien Mendela, помогло.
Hashtable<DecodeHintType, Object> decodeHints = new Hashtable<DecodeHintType, Object>();
decodeHints.put(DecodeHintType.TRY_HARDER, Boolean.TRUE); // Not required in my case
decodeHints.put(DecodeHintType.PURE_BARCODE, Boolean.TRUE);
Но последнее, что сработало для меня
- Если у вас большое изображение, возможно, вам придется уменьшить его масштаб.
- Также требуется ~ прямоугольная форма (421*402 в моем случае).
Ссылка Добавление пары похожих проблем, поданных через:
Инструменты, которые можно использовать для проверки имеющегося у вас изображения:
- qrdecode (не забудьте установить флажок "Производить отладочный вывод" для получения отладочной информации)
- Zxing Decoder онлайн
Решение первого уровня в некоторой степени ответ @Laurent работал почти для каждого образца, который у меня был, но не для некоторых.
Решение следующего уровня, добавляющее decodeHints до reader.decode(..), предложенное @Lucien Mendela, помогло.
Хеш-таблица decodeHints = новая хеш-таблица (); decodeHints.put(DecodeHintType.TRY_HARDER, Boolean.TRUE); // Не требуется в моем случае decodeHints.put(DecodeHintType.PURE_BARCODE, Boolean.TRUE); Но последнее, что сработало для меня
Если у вас большое изображение, возможно, вам придется уменьшить его масштаб. Также требуется ~ прямоугольная форма (421*402 в моем случае). Ссылка Добавление пары похожих проблем, поданных через:
Произвольное исключение com.google.zxing.NotFoundException во время чтения действительных инструментов QRCode, которые можно использовать для проверки имеющегося изображения:
qrdecode (не забудьте установить флажок "Производить отладочный вывод" для получения информации об отладке) Zxing Decoder online