يمكنك استخدام حزمة تعلّم الآلة للتعرّف على الرموز الشريطية وفك ترميزها.
الميزة | غير مجمعة | مجمعة |
---|---|---|
التنفيذ | يتم تنزيل النموذج بشكل ديناميكي من خلال "خدمات Google Play". | النموذج مرتبط بشكل ثابت بتطبيقك في وقت الإصدار. |
حجم التطبيق | زيادة عن حجم 200 كيلوبايت تقريبًا. | زيادة بمقدار 2.4 ميغابايت تقريبًا. |
وقت الإعداد | قد تحتاج إلى الانتظار حتى يتم تنزيل النموذج قبل الاستخدام الأول. | الطراز متاح على الفور. |
تجربة السمات والبيانات
- يمكنك تجربة نموذج التطبيق للتعرّف على مثال عن استخدام واجهة برمجة التطبيقات هذه.
- يمكنك الاطّلاع على تطبيق عرض التصميم المتعدد الأبعاد للحصول على تنفيذ شامل لواجهة برمجة التطبيقات هذه.
قبل البدء
في ملف
build.gradle
على مستوى المشروع، تأكَّد من تضمين مستودع Google Maven في كل من قسمَيbuildscript
وallprojects
.أضِف الاعتماديات الخاصة بمكتبات تعلُّم الآلة في Android إلى ملف Gradle الخاص بالوحدة على مستوى التطبيق، وهو عادةً
app/build.gradle
. اختَر إحدى الاعتماديات التالية وفقًا لاحتياجاتك:لتجميع النموذج مع تطبيقك:
dependencies { // ... // Use this dependency to bundle the model with your app implementation 'com.google.mlkit:barcode-scanning:17.1.0' }
لاستخدام النموذج في "خدمات Google Play":
dependencies { // ... // Use this dependency to use the dynamically downloaded model in Google Play Services implementation 'com.google.android.gms:play-services-mlkit-barcode-scanning:18.2.0' }
إذا اخترت استخدام النموذج في خدمات Google Play، يمكنك ضبط تطبيقك لتنزيل النموذج تلقائيًا على الجهاز بعد تثبيت تطبيقك من "متجر Play". لإجراء ذلك، أضِف البيان التالي إلى ملف
AndroidManifest.xml
الخاص بتطبيقك:<application ...> ... <meta-data android:name="com.google.mlkit.vision.DEPENDENCIES" android:value="barcode" > <!-- To use multiple models: android:value="barcode,model2,model3" --> </application>
يمكنك أيضًا التحقّق بشكل صريح من مدى توفّر النموذج وطلب تنزيل التطبيق من خلال خدمات PlayInstallClient API في خدمات Google Play.
إذا لم تفعّل عمليات تنزيل نماذج وقت التثبيت أو تطلب تنزيلًا صريحًا، يتم تنزيل النموذج في المرة الأولى التي تشغّل فيها الماسح الضوئي. لا تعرض الطلبات التي تجريها قبل اكتمال عملية التنزيل أي نتائج.
إرشادات متعلقة بصورة الإدخال
-
ليتمكّن ML Kit من قراءة الرموز الشريطية بدقة، يجب أن تحتوي صور الإدخال على رموز شريطية يتم تمثيلها ببيانات بكسل كافية.
وتعتمد متطلبات بيانات البكسل المحدّدة على نوع الرمز الشريطي ومقدار البيانات التي تمّ ترميزه، لأنّ العديد من الرموز الشريطية تتوافق مع حمولة ذات حجم متغيّر. بصورة عامة، يجب ألا يقل عرض وحدة الرمز الشريطي ذات المغزى عن 2 بكسل، وبالنسبة إلى الرموز ثنائية الأبعاد، يجب أن يبلغ طول وحدة البكسل 2 بكسل.
على سبيل المثال، تتألف الرموز الشريطية EAN-13 من أشرطة ومسافات بعرضها 1 أو 2 أو 3 أو 4 وحدات، لذلك يجب أن تتضمن صورة الرمز الشريطي EAN-13 أشرطة و/أو مساحات لا يقل عرضها عن 2 و4 و6 و8 بكسل. يجب ألا يزيد عرض الرمز الشريطي عن EAN-13 عن 95 وحدة، ويجب ألا يقل عرض الرمز الشريطي عن 190 بكسل.
تحتاج تنسيقات الكثافة، مثل PDF417، إلى سمات بكسل أكبر كي تتمكن حزمة تعلُّم الآلة من قراءتها بشكل موثوق. على سبيل المثال، يمكن أن يحتوي رمز PDF417 على ما يصل إلى 34 "كلمة" بعرض 17 وحدة في صف واحد، وألا يزيد عرضها عن 1156 بكسل.
-
يمكن أن يؤثّر تركيز الصورة الضعيف في دقة المسح. في حال لم يتلقَّ تطبيقك أي نتائج مقبولة، اطلب من المستخدم إعادة التقاط الصورة.
-
بالنسبة إلى التطبيقات العادية، ننصح بتقديم صورة بدقة أعلى، مثل 1280×720 أو 1920×1080، ما يجعل الرموز الشريطية قابلة للمسح من مسافة أبعد من الكاميرا.
ويمكنك تحسين الأداء في التطبيقات التي يكون فيها وقت الاستجابة أمرًا بالغ الأهمية، وذلك من خلال التقاط الصور بدقة أقل، شرط أن يكون الرمز الشريطي الذي يشكّل معظم الصور التي يتم إدخالها. راجِع أيضًا نصائح لتحسين الأداء في الوقت الفعلي
1- ضبط الماسح الضوئي للرموز الشريطية
إذا كنت تعرف أشكال الرموز الشريطية التي تتوقع قراءتها، يمكنك تحسين سرعة أداة رصد الرموز الشريطية من خلال ضبطها لاكتشاف تلك التنسيقات فقط.على سبيل المثال، لاكتشاف رمز Aztec ورموز الاستجابة السريعة فقط، أنشِئ عنصرًا
في BarcodeScannerOptions
كما في المثال التالي:
Kotlin
val options = BarcodeScannerOptions.Builder() .setBarcodeFormats( Barcode.FORMAT_QR_CODE, Barcode.FORMAT_AZTEC) .build()
لغة Java
BarcodeScannerOptions options = new BarcodeScannerOptions.Builder() .setBarcodeFormats( Barcode.FORMAT_QR_CODE, Barcode.FORMAT_AZTEC) .build();
تتوفّر التنسيقات التالية:
- الرمز 128 (
FORMAT_CODE_128
) - الرمز 39 (
FORMAT_CODE_39
) - الرمز البرمجي 93 (
FORMAT_CODE_93
) - كودابار (
FORMAT_CODABAR
) - رقم EAN-13 (
FORMAT_EAN_13
) - رقم EAN-8 (
FORMAT_EAN_8
) - فريق تكنولوجيا المعلومات (
FORMAT_ITF
) - الرمز العالمي للمنتج (UPC) (
FORMAT_UPC_A
) - الرمز العالمي للمنتج (UPC) (
FORMAT_UPC_E
) - رمز الاستجابة السريعة (
FORMAT_QR_CODE
) - PDF417 (
FORMAT_PDF417
) - الأزتك (
FORMAT_AZTEC
) - مصفوفة البيانات (
FORMAT_DATA_MATRIX
)
بدءًا من النموذج المجمّع 17.1.0 والنموذج غير المجمّع 18.2.0، يمكنك أيضًا طلب enableAllPotentialBarcodes()
لعرض جميع الرموز الشريطية المحتملة حتى إذا
تعذّر فك ترميزها. ويمكن استخدامه لتسهيل اكتشاف المحتوى بشكل أكبر، مثلاً من خلال تكبير الكاميرا للحصول على صورة أكثر وضوحًا لأي رمز شريطي في مربع الإحاطة التي تم عرضها.
Kotlin
val options = BarcodeScannerOptions.Builder() .setBarcodeFormats(...) .enableAllPotentialBarcodes() // Optional .build()
Java
BarcodeScannerOptions options = new BarcodeScannerOptions.Builder() .setBarcodeFormats(...) .enableAllPotentialBarcodes() // Optional .build();
2. Prepare the input image
To recognize barcodes in an image, create anInputImage
object
from either a Bitmap
, media.Image
, ByteBuffer
, byte array, or a file on
the device. Then, pass the InputImage
object to the
BarcodeScanner
's process
method.
You can create an InputImage
object from different sources, each is explained below.
Using a media.Image
To create an InputImage
object from a media.Image
object, such as when you capture an image from a
device's camera, pass the media.Image
object and the image's
rotation to InputImage.fromMediaImage()
.
If you use the
CameraX library, the OnImageCapturedListener
and
ImageAnalysis.Analyzer
classes calculate the rotation value
for you.
Kotlin
private class YourImageAnalyzer : ImageAnalysis.Analyzer { override fun analyze(imageProxy: ImageProxy) { val mediaImage = imageProxy.image if (mediaImage != null) { val image = InputImage.fromMediaImage(mediaImage, imageProxy.imageInfo.rotationDegrees) // Pass image to an ML Kit Vision API // ... } } }
لغة Java
private class YourAnalyzer implements ImageAnalysis.Analyzer { @Override public void analyze(ImageProxy imageProxy) { Image mediaImage = imageProxy.getImage(); if (mediaImage != null) { InputImage image = InputImage.fromMediaImage(mediaImage, imageProxy.getImageInfo().getRotationDegrees()); // Pass image to an ML Kit Vision API // ... } } }
إذا كنت لا تستخدم مكتبة كاميرا تمنحك درجة تدوير الصورة، يمكنك احتسابها من درجة تدوير الجهاز واتجاه جهاز استشعار الكاميرا في الجهاز:
Kotlin
private val ORIENTATIONS = SparseIntArray() init { ORIENTATIONS.append(Surface.ROTATION_0, 0) ORIENTATIONS.append(Surface.ROTATION_90, 90) ORIENTATIONS.append(Surface.ROTATION_180, 180) ORIENTATIONS.append(Surface.ROTATION_270, 270) } /** * Get the angle by which an image must be rotated given the device's current * orientation. */ @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) @Throws(CameraAccessException::class) private fun getRotationCompensation(cameraId: String, activity: Activity, isFrontFacing: Boolean): Int { // Get the device's current rotation relative to its "native" orientation. // Then, from the ORIENTATIONS table, look up the angle the image must be // rotated to compensate for the device's rotation. val deviceRotation = activity.windowManager.defaultDisplay.rotation var rotationCompensation = ORIENTATIONS.get(deviceRotation) // Get the device's sensor orientation. val cameraManager = activity.getSystemService(CAMERA_SERVICE) as CameraManager val sensorOrientation = cameraManager .getCameraCharacteristics(cameraId) .get(CameraCharacteristics.SENSOR_ORIENTATION)!! if (isFrontFacing) { rotationCompensation = (sensorOrientation + rotationCompensation) % 360 } else { // back-facing rotationCompensation = (sensorOrientation - rotationCompensation + 360) % 360 } return rotationCompensation }
لغة Java
private static final SparseIntArray ORIENTATIONS = new SparseIntArray(); static { ORIENTATIONS.append(Surface.ROTATION_0, 0); ORIENTATIONS.append(Surface.ROTATION_90, 90); ORIENTATIONS.append(Surface.ROTATION_180, 180); ORIENTATIONS.append(Surface.ROTATION_270, 270); } /** * Get the angle by which an image must be rotated given the device's current * orientation. */ @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) private int getRotationCompensation(String cameraId, Activity activity, boolean isFrontFacing) throws CameraAccessException { // Get the device's current rotation relative to its "native" orientation. // Then, from the ORIENTATIONS table, look up the angle the image must be // rotated to compensate for the device's rotation. int deviceRotation = activity.getWindowManager().getDefaultDisplay().getRotation(); int rotationCompensation = ORIENTATIONS.get(deviceRotation); // Get the device's sensor orientation. CameraManager cameraManager = (CameraManager) activity.getSystemService(CAMERA_SERVICE); int sensorOrientation = cameraManager .getCameraCharacteristics(cameraId) .get(CameraCharacteristics.SENSOR_ORIENTATION); if (isFrontFacing) { rotationCompensation = (sensorOrientation + rotationCompensation) % 360; } else { // back-facing rotationCompensation = (sensorOrientation - rotationCompensation + 360) % 360; } return rotationCompensation; }
بعد ذلك، أدخِل العنصر media.Image
وقيمة درجة التدوير إلى السمة InputImage.fromMediaImage()
:
Kotlin
val image = InputImage.fromMediaImage(mediaImage, rotation)
Java
InputImage image = InputImage.fromMediaImage(mediaImage, rotation);
استخدام معرّف الموارد المنتظم (URI) للملف
لإنشاء عنصر InputImage
من معرّف الموارد المنتظم (URI)، يجب تمرير سياق التطبيق ومعرّف الموارد المنتظم (URI) للملف إلى
InputImage.fromFilePath()
. ويُعدّ هذا الإجراء مفيدًا عندما تستخدم
هدف ACTION_GET_CONTENT
لحثّ المستخدم على اختيار
صورة من تطبيق المعرض.
Kotlin
val image: InputImage try { image = InputImage.fromFilePath(context, uri) } catch (e: IOException) { e.printStackTrace() }
Java
InputImage image; try { image = InputImage.fromFilePath(context, uri); } catch (IOException e) { e.printStackTrace(); }
باستخدام ByteBuffer
أو ByteArray
لإنشاء عنصر InputImage
من ByteBuffer
أو ByteArray
، يجب أولاً احتساب
درجة تدوير الصورة كما هو موضّح سابقًا في إدخال media.Image
.
بعد ذلك، يمكنك إنشاء عنصر InputImage
باستخدام المخزن المؤقت أو الصفيف، بالإضافة إلى ارتفاع الصورة وعرضها وتنسيق ترميز الألوان ودرجة التدوير:
Kotlin
val image = InputImage.fromByteBuffer( byteBuffer, /* image width */ 480, /* image height */ 360, rotationDegrees, InputImage.IMAGE_FORMAT_NV21 // or IMAGE_FORMAT_YV12 ) // Or: val image = InputImage.fromByteArray( byteArray, /* image width */ 480, /* image height */ 360, rotationDegrees, InputImage.IMAGE_FORMAT_NV21 // or IMAGE_FORMAT_YV12 )
لغة Java
InputImage image = InputImage.fromByteBuffer(byteBuffer, /* image width */ 480, /* image height */ 360, rotationDegrees, InputImage.IMAGE_FORMAT_NV21 // or IMAGE_FORMAT_YV12 ); // Or: InputImage image = InputImage.fromByteArray( byteArray, /* image width */480, /* image height */360, rotation, InputImage.IMAGE_FORMAT_NV21 // or IMAGE_FORMAT_YV12 );
استخدام Bitmap
لإنشاء عنصر InputImage
من كائن Bitmap
، عليك تقديم البيان التالي:
Kotlin
val image = InputImage.fromBitmap(bitmap, 0)
Java
InputImage image = InputImage.fromBitmap(bitmap, rotationDegree);
يتم تمثيل الصورة بواسطة عنصر Bitmap
مع درجات التدوير.
3. الحصول على مثيل من BarcodeScanner
Kotlin
val scanner = BarcodeScanning.getClient() // Or, to specify the formats to recognize: // val scanner = BarcodeScanning.getClient(options)
لغة Java
BarcodeScanner scanner = BarcodeScanning.getClient(); // Or, to specify the formats to recognize: // BarcodeScanner scanner = BarcodeScanning.getClient(options);
4- معالجة الصورة
تمرير الصورة إلى الطريقةprocess
:
Kotlin
val result = scanner.process(image) .addOnSuccessListener { barcodes -> // Task completed successfully // ... } .addOnFailureListener { // Task failed with an exception // ... }
لغة Java
Task<List<Barcode>> result = scanner.process(image) .addOnSuccessListener(new OnSuccessListener<List<Barcode>>() { @Override public void onSuccess(List<Barcode> barcodes) { // Task completed successfully // ... } }) .addOnFailureListener(new OnFailureListener() { @Override public void onFailure(@NonNull Exception e) { // Task failed with an exception // ... } });
5. الحصول على معلومات من الرموز الشريطية
في حال نجاح عملية التعرّف على الرمز الشريطي، سيتم تمرير قائمةBarcode
بالعناصر إلى المستمع بنجاح. يمثّل كل عنصر Barcode
رمزًا شريطيًا تم رصده في الصورة. بالنسبة إلى كل رمز شريطي، يمكنك الحصول على إحداثياته الحدية في صورة الإدخال،
بالإضافة إلى البيانات الأولية التي تم ترميزها بواسطة الرمز الشريطي. أيضًا، إذا تمكّن الماسح الضوئي للرمز الشريطي من تحديد نوع البيانات التي تم ترميزها بواسطة الرمز الشريطي، يمكنك
الحصول على عنصر يتضمن بيانات مُحلَّلة.
على سبيل المثال:
Kotlin
for (barcode in barcodes) { val bounds = barcode.boundingBox val corners = barcode.cornerPoints val rawValue = barcode.rawValue val valueType = barcode.valueType // See API reference for complete list of supported types when (valueType) { Barcode.TYPE_WIFI -> { val ssid = barcode.wifi!!.ssid val password = barcode.wifi!!.password val type = barcode.wifi!!.encryptionType } Barcode.TYPE_URL -> { val title = barcode.url!!.title val url = barcode.url!!.url } } }
لغة Java
for (Barcode barcode: barcodes) { Rect bounds = barcode.getBoundingBox(); Point[] corners = barcode.getCornerPoints(); String rawValue = barcode.getRawValue(); int valueType = barcode.getValueType(); // See API reference for complete list of supported types switch (valueType) { case Barcode.TYPE_WIFI: String ssid = barcode.getWifi().getSsid(); String password = barcode.getWifi().getPassword(); int type = barcode.getWifi().getEncryptionType(); break; case Barcode.TYPE_URL: String title = barcode.getUrl().getTitle(); String url = barcode.getUrl().getUrl(); break; } }
نصائح لتحسين الأداء في الوقت الفعلي
إذا أردت مسح الرموز الشريطية ضوئيًا في تطبيق في الوقت الفعلي، اتّبِع هذه الإرشادات لتحقيق أفضل معدّلات عرض الإطارات:
-
لا تلتقط صورة بدرجة الدقة الأصلية للكاميرا. وفي بعض الأجهزة، ينتج عن التقاط الصور بدرجة الدقة الأصلية دقة كبيرة جدًا (10 ميغابكسل)، ما يؤدي إلى وقت استجابة منخفض للغاية بدون فائدة إلى الدقة. وبدلاً من ذلك، اطلب حجم الكاميرا المطلوب لرصد الرمز الشريطي والذي لا يزيد عادةً عن 2 ميغابكسل.
إذا كانت سرعة المسح مهمة، يمكنك خفض درجة دقة التقاط الصورة. يُرجى العِلم أنّ الحدّ الأدنى لمتطلبات حجم الرمز الشريطي الموضّح أعلاه.
إذا كنت تحاول التعرّف على الرموز الشريطية من سلسلة من إطارات الفيديو الخاصة بالبث، قد تظهر أداة التعرّف على نتائج مختلفة من إطار إلى آخر. وعليك الانتظار حتى تحصل على سلسلة متتابعة من القيمة نفسها حتى تتأكّد من أنك تعرض نتيجة جيدة.
لا يتوفّر الرقم الاختباري الاختباري لـ ITF وCODE-39.
- إذا كنت تستخدم واجهة برمجة التطبيقات
Camera
أوcamera2
، يمكنك تقييد المكالمات التي تصل إلى أداة الرصد. إذا ظهر إطار فيديو جديد أثناء تشغيل أداة الرصد، أفلِت الإطار. يمكنك الاطّلاع على مثالVisionProcessorBase
في نموذج البدء السريع لتطبيق التطبيق. - إذا كنت تستخدم واجهة برمجة التطبيقات
CameraX
، تأكّد من ضبط استراتيجية الضغط على القيمة التلقائيةImageAnalysis.STRATEGY_KEEP_ONLY_LATEST
. ويضمن ذلك إرسال صورة واحدة فقط للتحليل في كل مرة. وإذا تم إنتاج المزيد من الصور عندما تكون أداة التحليل مشغولة، سيتم إسقاطها تلقائيًا ولن يتم وضعها في قائمة انتظار التسليم. بعد إغلاق الصورة التي يتم تحليلها من خلال استدعاء صورة ImageProxy.close() ، سيتم تسليم أحدث صورة بعد ذلك. - إذا كنت تستخدم أداة الاكتشاف للكشف عن رسومات بهدف تركيب الرسومات على
صورة الإدخال، عليك أولاً الحصول على النتيجة من حزمة تعلّم الآلة، ثم عرض الصورة
والتراكب في خطوة واحدة. ويتم عرض هذا العرض على سطح العرض مرة واحدة فقط لكل إطار إدخال. يمكنك الاطّلاع على
CameraSourcePreview
وGraphicOverlay
في نموذج التطبيق السريع للبدء كمثال. - إذا كنت تستخدم واجهة برمجة التطبيقات Camera2 API، عليك التقاط صور بتنسيق
ImageFormat.YUV_420_888
. إذا كنت تستخدم إصدارًا قديمًا من واجهة برمجة تطبيقات الكاميرا، يمكنك التقاط صور بتنسيقImageFormat.NV21
.