कैमरा लेंस और सुविधाएं

ध्यान दें: यह पेज Camera2 पैकेज के बारे में है. हमारा सुझाव है कि अगर आपके ऐप्लिकेशन को Camera2 में खास और कम लेवल की सुविधाओं की ज़रूरत न हो, तो हमारा सुझाव है कि आप CameraX इस्तेमाल करें. CameraX और Camera2, दोनों ही Android 5.0 (एपीआई लेवल 21) और इसके बाद वाले वर्शन पर काम करते हैं.

कई आधुनिक Android डिवाइसों में आगे, पीछे या दोनों ओर से किए जा सकते हैं. हर लेंस में खास क्षमताएं हो सकती हैं. जैसे, बर्स्ट कैप्चर, मैन्युअल कंट्रोल या मोशन ट्रैकिंग का इस्तेमाल किया जा सकता है. चेक जमा करने वाला ऐप्लिकेशन सिर्फ़ इन ऐप्लिकेशन का इस्तेमाल कर सकता है वाला पहला कैमरा है, जबकि सोशल मीडिया ऐप्लिकेशन में सामने का कैमरा इस्तेमाल कर सकते हैं. हालांकि, इससे उपयोगकर्ताओं को सभी उपलब्ध कैमरों के बीच स्विच करने का विकल्प मिलता है लेंस. यह उनके चुने गए विकल्पों को भी याद रख सकती है.

इस पेज में कैमरा लेंस और उनकी क्षमताओं की सूची बनाने का तरीका बताया गया है. आपके ऐप्लिकेशन में यह तय किया जा सकता है कि किसी स्थिति में कौनसे लेंस का इस्तेमाल किया जाए. नीचे दिया गया कोड स्निपेट सभी कैमरों की सूची हासिल करता है और फिर से उनसे:

KotlinJava
try {
   
val cameraIdList = cameraManager.cameraIdList // may be empty

   
// iterate over available camera devices
   
for (cameraId in cameraIdList) {
       
val characteristics = cameraManager.getCameraCharacteristics(cameraId)
       
val cameraLensFacing = characteristics.get(CameraCharacteristics.LENS_FACING)
       
val cameraCapabilities = characteristics.get(CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES)

       
// check if the selected camera device supports basic features
       
// ensures backward compatibility with the original Camera API
       
val isBackwardCompatible = cameraCapabilities?.contains(
           
CameraMetadata.REQUEST_AVAILABLE_CAPABILITIES_BACKWARD_COMPATIBLE) ?: false
       
...
   
}
} catch (e: CameraAccessException) {
    e
.message?.let { Log.e(TAG, it) }
   
...
}
try {
   
String[] cameraIdList = cameraManager.getCameraIdList(); // may be empty

   
// iterate over available camera devices
   
for (String cameraId : cameraIdList) {
       
CameraCharacteristics characteristics = cameraManager.getCameraCharacteristics(cameraId);
       
int cameraLensFacing = characteristics.get(CameraCharacteristics.LENS_FACING);
       
int[] cameraCapabilities =
            characteristics
.get(CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES);

       
// check if the selected camera device supports basic features
       
// ensures backward compatibility with the original Camera API
       
boolean isBackwardCompatible = false;
       
for (int capability : cameraCapabilities) {
           
if (capability == CameraMetadata.REQUEST_AVAILABLE_CAPABILITIES_BACKWARD_COMPATIBLE) {
                isBackwardCompatible
= true;
               
break;
           
}
       
}
       
...
   
}
} catch (CameraAccessException e) {
   
Log.e(TAG, e.getMessage());
   
...
}

वैरिएबल cameraLensFacing, कैमरे के सामने आने की दिशा के बारे में बताता है के मुताबिक होता है और उसमें नीचे दी गई कोई एक वैल्यू होती है:

लेंस-फ़ेसिंग कॉन्फ़िगरेशन के बारे में ज़्यादा जानकारी के लिए, यहां देखें CameraCharacteristics.LENS_FACING.

पिछले कोड सैंपल के वैरिएबल cameraCapabilities में यह शामिल है कई तरह की क्षमताओं के बारे में जानकारी. इसमें यह जानकारी भी शामिल है कि कैमरा एक आउटपुट के रूप में मानक फ़्रेम्स बना सके (इसके विपरीत, केवल गहराई वाले सेंसर का डेटा). आप यह पता लगा सकते हैं कि क्या CameraMetadata.REQUEST_AVAILABLE_CAPABILITIES_BACKWARD_COMPATIBLE कैमरे की क्षमताओं में से एक है, जिसे isBackwardCompatible.

सही डिफ़ॉल्ट सेटिंग चुनें

अपने ऐप्लिकेशन में, हो सकता है कि आप डिफ़ॉल्ट रूप से कोई खास कैमरा खोलना चाहें (अगर यह उपलब्ध है). उदाहरण के लिए, कोई सेल्फ़ी ऐप्लिकेशन मुख्य रूप से सामने वाला हिस्सा खोलता है कैमरा चालू हो सकता है, जबकि ऑगमेंटेड रिएलिटी (एआर) ऐप्लिकेशन, पीछे वाले कैमरे के साथ चालू हो सकता है. यह फ़ंक्शन वह पहला कैमरा दिखाता है जो किसी दी गई दिशा में होता है:

KotlinJava
fun getFirstCameraIdFacing(cameraManager: CameraManager,
                           facing
: Int = CameraMetadata.LENS_FACING_BACK): String? {
   
try {
       
// Get list of all compatible cameras
       
val cameraIds = cameraManager.cameraIdList.filter {
           
val characteristics = cameraManager.getCameraCharacteristics(it)
           
val capabilities = characteristics.get(CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES)
            capabilities
?.contains(
                   
CameraMetadata.REQUEST_AVAILABLE_CAPABILITIES_BACKWARD_COMPATIBLE) ?: false
       
}

       
// Iterate over the list of cameras and return the first one matching desired
       
// lens-facing configuration
        cameraIds
.forEach {
           
val characteristics = cameraManager.getCameraCharacteristics(it)
           
if (characteristics.get(CameraCharacteristics.LENS_FACING) == facing) {
               
return it
           
}
       
}

       
// If no camera matched desired orientation, return the first one from the list
       
return cameraIds.firstOrNull()
   
} catch (e: CameraAccessException) {
        e
.message?.let { Log.e(TAG, it) }
   
}
}
public String getFirstCameraIdFacing(CameraManager cameraManager, @Nullable Integer facing) {
   
if (facing == null) facing = CameraMetadata.LENS_FACING_BACK;
   
String cameraId = null;

   
try {
       
// Get a list of all compatible cameras
       
String[] cameraIdList = cameraManager.getCameraIdList();

       
// Iterate over the list of cameras and return the first one matching desired
       
// lens-facing configuration and backward compatibility
       
for (String id : cameraIdList) {
           
CameraCharacteristics characteristics = cameraManager.getCameraCharacteristics(id);
           
int[] capabilities = characteristics.get(CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES);
           
for (int capability : capabilities) {
               
if (capability == CameraMetadata.REQUEST_AVAILABLE_CAPABILITIES_BACKWARD_COMPATIBLE
                       
&& characteristics.get(CameraCharacteristics.LENS_FACING).equals(facing)) {
                    cameraId
= id;
                   
break;
               
}
           
}
       
}

       
// If no camera matches the desired orientation, return the first one from the list
        cameraId
= cameraIdList[0];
   
} catch (CameraAccessException e) {
       
Log.e(TAG, "getFirstCameraIdFacing: " + e.getMessage());
   
}

   
return cameraId;
}

कैमरा स्विच करने की सुविधा चालू करें

कई कैमरा ऐप्लिकेशन, उपयोगकर्ताओं को एक से दूसरे कैमरे पर जाने का विकल्प देते हैं:

पहली इमेज. Google Camera ऐप्लिकेशन में कैमरा स्विच करने का बटन

कई डिवाइसों में एक ही दिशा में लगे कई कैमरे होते हैं. कुछ लोगों के पास तो बाहरी यूएसबी कैमरे. उपयोगकर्ताओं को ऐसा यूज़र इंटरफ़ेस (यूआई) उपलब्ध कराने के लिए जो एक से दूसरे खाते पर स्विच कर सके अलग-अलग कैमरे हों, तो हर संभव कैमरे के लिए पहला उपलब्ध कैमरा चुनें लेंस के सामने कॉन्फ़िगरेशन.

वैसे तो अगला कैमरा चुनने के लिए कोई खास वजह नहीं है, लेकिन यह कोड ज़्यादातर इस्तेमाल के लिए काम करता है:

KotlinJava
fun filterCompatibleCameras(cameraIds: Array<String>,
                            cameraManager
: CameraManager): List<String> {
   
return cameraIds.filter {
       
val characteristics = cameraManager.getCameraCharacteristics(it)
        characteristics
.get(CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES)?.contains(
               
CameraMetadata.REQUEST_AVAILABLE_CAPABILITIES_BACKWARD_COMPATIBLE) ?: false
   
}
}

fun filterCameraIdsFacing(cameraIds: List<String>, cameraManager: CameraManager,
                          facing
: Int): List<String> {
   
return cameraIds.filter {
       
val characteristics = cameraManager.getCameraCharacteristics(it)
        characteristics
.get(CameraCharacteristics.LENS_FACING) == facing
   
}
}

fun getNextCameraId(cameraManager: CameraManager, currCameraId: String? = null): String? {
   
// Get all front, back and external cameras in 3 separate lists
   
val cameraIds = filterCompatibleCameras(cameraManager.cameraIdList, cameraManager)
   
val backCameras = filterCameraIdsFacing(
            cameraIds
, cameraManager, CameraMetadata.LENS_FACING_BACK)
   
val frontCameras = filterCameraIdsFacing(
            cameraIds
, cameraManager, CameraMetadata.LENS_FACING_FRONT)
   
val externalCameras = filterCameraIdsFacing(
            cameraIds
, cameraManager, CameraMetadata.LENS_FACING_EXTERNAL)

   
// The recommended order of iteration is: all external, first back, first front
   
val allCameras = (externalCameras + listOf(
            backCameras
.firstOrNull(), frontCameras.firstOrNull())).filterNotNull()

   
// Get the index of the currently selected camera in the list
   
val cameraIndex = allCameras.indexOf(currCameraId)

   
// The selected camera may not be in the list, for example it could be an
   
// external camera that has been removed by the user
   
return if (cameraIndex == -1) {
       
// Return the first camera from the list
        allCameras
.getOrNull(0)
   
} else {
       
// Return the next camera from the list, wrap around if necessary
        allCameras
.getOrNull((cameraIndex + 1) % allCameras.size)
   
}
}
public List<String> filterCompatibleCameras(CameraManager cameraManager, String[] cameraIds) {
   
final List<String> compatibleCameras = new ArrayList<>();

   
try {
       
for (String id : cameraIds) {
           
CameraCharacteristics characteristics = cameraManager.getCameraCharacteristics(id);
           
int[] capabilities = characteristics.get(CameraCharacteristics.REQUEST_AVAILABLE_CAPABILITIES);
           
for (int capability : capabilities) {
               
if (capability == CameraMetadata.REQUEST_AVAILABLE_CAPABILITIES_BACKWARD_COMPATIBLE) {
                    compatibleCameras
.add(id);
               
}
           
}
       
}
   
} catch (CameraAccessException e) {
       
Log.e(TAG, "filterCompatibleCameras: " + e.getMessage());
   
}

   
return compatibleCameras;
}

public List<String> filterCameraIdsFacing(CameraManager cameraManager, List<String> cameraIds, int lensFacing) {
   
final List<String> compatibleCameras = new ArrayList<>();

   
try {
       
for (String id : cameraIds) {
           
CameraCharacteristics characteristics = cameraManager.getCameraCharacteristics(id);
           
if (characteristics.get(CameraCharacteristics.LENS_FACING) == lensFacing) {
                compatibleCameras
.add(id);
           
}
       
}
   
} catch (CameraAccessException e) {
       
Log.e(TAG, "filterCameraIdsFacing: " + e.getMessage());
   
}

   
return compatibleCameras;
}

public String getNextCameraId(CameraManager cameraManager, @Nullable String currentCameraId) {
   
String nextCameraId = null;

   
try {
       
// Get all front, back, and external cameras in 3 separate lists
       
List<String> compatibleCameraIds = filterCompatibleCameras(cameraManager, cameraManager.getCameraIdList());
       
List<String> backCameras = filterCameraIdsFacing(cameraManager, compatibleCameraIds, CameraMetadata.LENS_FACING_BACK);
       
List<String> frontCameras = filterCameraIdsFacing(cameraManager, compatibleCameraIds, CameraMetadata.LENS_FACING_FRONT);
       
List<String>externalCameras = filterCameraIdsFacing(cameraManager, compatibleCameraIds, CameraMetadata.LENS_FACING_EXTERNAL);

       
// The recommended order of iteration is: all external, first back, first front
       
List<String> allCameras = new ArrayList<>(externalCameras);
       
if (!backCameras.isEmpty()) allCameras.add(backCameras.get(0));
       
if (!frontCameras.isEmpty()) allCameras.add(frontCameras.get(0));

       
// Get the index of the currently selected camera in the list
       
int cameraIndex = allCameras.indexOf(currentCameraId);

       
// The selected camera may not be in the list, for example it could be an
       
// external camera that has been removed by the user
       
if (cameraIndex == -1) {
           
// Return the first camera from the list
            nextCameraId
= !allCameras.isEmpty() ? allCameras.get(0) : null;
       
else {
           
if (!allCameras.isEmpty()) {
               
// Return the next camera from the list, wrap around if necessary
                nextCameraId
= allCameras.get((cameraIndex + 1) % allCameras.size());
           
}
       
}
   
} catch (CameraAccessException e) {
       
Log.e(TAG, "getNextCameraId: " + e.getMessage());
   
}

   
return nextCameraId;
}

यह कोड कई अलग-अलग तरह के डिवाइसों के एक बड़े सेट के लिए काम करता है कॉन्फ़िगरेशन. किनारे के केस का हिसाब लगाने के बारे में ज़्यादा जानकारी के लिए, CameraMetadata.REQUEST_AVAILABLE_CAPABILITIES_LOGICAL_MULTI_CAMERA देखें.

इसके साथ काम करने वाले ऐप्लिकेशन बनाएं

अब भी बंद किए गए Camera API का इस्तेमाल करने वाले ऐप्लिकेशन के लिए, कैमरों की संख्या वह Camera.getNumberOfCameras() रिटर्न, ओईएम लागू करने के हिसाब से तय होता है. अगर वीडियो में लॉजिकल मल्टी-कैमरा मौजूद है, तो सिस्टम में, पुराने सिस्टम के साथ काम करने की सुविधा को बनाए रखने के लिए, इस तरीके से सिर्फ़ एक हर लॉजिकल कैमरे और बुनियादी कैमरे के ग्रुप के लिए कैमरा. सभी कैमरे देखने के लिए, Camera2 एपीआई का इस्तेमाल करें.

कैमरा ओरिएंटेशन के बारे में ज़्यादा जानने के लिए, यहां जाएं Camera.CameraInfo.orientation.

सामान्य तौर पर, Camera.getCameraInfo() सभी कैमरे से क्वेरी करने के लिए एपीआई orientation और हर एक उपलब्ध ओरिएंटेशन के लिए, सिर्फ़ एक कैमरा ऐसे उपयोगकर्ताओं को दिखाएं जो कैमरों के बीच स्विच कर रहा होता है.

सभी तरह के डिवाइस में बदलाव करें

यह न मानें कि आपका ऐप्लिकेशन हमेशा हैंडहेल्ड डिवाइस पर चलता है. दो कैमरे. इसके बजाय, ऐप्लिकेशन के हिसाब से सबसे सही कैमरे चुनें. अगर आपको किसी ख़ास कैमरे की ज़रूरत नहीं है. वह पहला कैमरा चुनें जिसकी स्क्रीन आपके हिसाब से है दिशा-निर्देश. अगर कोई बाहरी कैमरा कनेक्ट किया गया है, तो आपको यह लग सकता है कि उपयोगकर्ता उसे डिफ़ॉल्ट के रूप में प्राथमिकता देता है.