1. 程式人生 > >Android學習之呼叫相機和相簿

Android學習之呼叫相機和相簿

呼叫攝像頭拍照

修改佈局檔案中程式碼:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="match_parent">
    <Button
        android:id="@+id/take_photo"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Take Photo"/>    //用來開啟攝像頭進行拍照

    <ImageView
        android:id="@+id/picture"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"/>   //將拍到的圖片顯示出來
</LinearLayout>

編寫呼叫攝像頭的具體邏輯。修改活動檔案:

public class MainActivity extends AppCompatActivity {
    public static final int TAKE_PHOTO = 1;
    private ImageView picture;
    private Uri imageUri;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button takePhoto = (Button) findViewById(R.id.take_photo);
        picture = (ImageView) findViewById(R.id.picture);
        takePhoto.setOnClickListener(new View.OnClickListener(){
            @Override
            public  void onClick(View v){
                File outputImage = new File(getExternalCacheDir(),"output_image.jpg");   //建立了一個File物件,用於存放攝像頭拍下的圖片,並將它儲存在SD卡的應用快取目錄下,用getExternalCacheDir()得到目錄
                try{
                    if(outputImage.exists()){
                        outputImage.delete();
                    }
                    outputImage.createNewFile();
                }catch (IOException e) {
                    e.printStackTrace();
                }
                if(Build.VERSION.SDK_INT>=24){   //若裝置系統版本低於Android7.0,就呼叫Uri的fromFile()將File物件轉換成Uri物件(表示著圖片的本地真實路徑)
                    imageUri = FileProvider.getUriForFile(MainActivity.this,"com/example.cameraalbumtest.fileprovider",outputImage);//否則就呼叫FileProvider的getUriForFile()
                }else{
                    imageUri = Uri.fromFile(outputImage);   方法將File物件轉換成一個封裝過的Uri物件
                }
                Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");//將這個Intent的action指定為android.media.action.IMAGE_CAPTURE
                intent.putExtra(MediaStore.EXTRA_OUTPUT,imageUri);//置頂圖片的輸出地址
                startActivityForResult(intent,RAKE_PHOTO);
            }
        });
    }
    @Override
    protected void onActivityResult(int requestCode,int resultCode, Intent data){  //用startActivityForResult()啟動活動會有結果返回到此方法中
        switch (requestCode){
            case TAKE_PHOTO:
                if(resultCode == RESULT_OK){
                    try{
                        //將拍攝的照片顯示出來
                        Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream((imageUri));
                        picture.setImageBitmap(bitmap);
                    }catch (FileNotFoundException e){
                        e.printStackTrace();
                    }
                }
                break;
            default:
                break;
        }
    }
}

在AndroidMainfest.xml中對內容提供器(FileProvider)進行註冊:

    <provider
            android:authorities="com.example.cameraalbumtest.fileprovider"  //必須和乾菜FileProvider.getUriForFile()方法中第二個引數一致
            android:name="android.support.v4.content.FileProvider"  //此屬性值是固定的
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data              //用<meta-data>來指定Uri的共享路徑,並引用了一個@xml/file_path資源(下面我們來建立)
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/file_paths"/>
        </provider>

建立上面說的@xml/file_path資源

<?xml version="1.0" encoding="utf-8" ?>
<paths xmlns:andriod="http://schemas.android.com/apk/res/android">
    <enternal-path name="my_images" path=""/>    //用來指定Uri共享,name屬性值可以隨便填,path屬性的指標是共享的具體路徑,此處為空表示共
</paths>                                         //享整個SD卡

在Android4.4之前,需要宣告許可權:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
          package="com.example.cameraalbumtest">
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

從相簿中選擇照片

在上面的基礎上在佈局檔案中新增:

   <Button
        android:id="@+id/choose_from_album"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text=""Choose From Album/>

然後修改MainActivity檔案中程式碼,加入從相簿選擇照片的邏輯:(紅色為新增)

public class MainActivity extends AppCompatActivity {
    public static final int TAKE_PHOTO = 1;
    private ImageView picture;
    private Uri imageUri;
    public static final int CHOOSE_PHOTO = 2;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button takePhoto = (Button) findViewById(R.id.take_photo);
        Button chooseFromAlbum = (Button) findViewById(R.id.choose_from_album);
        picture = (ImageView) findViewById(R.id.picture);
        takePhoto.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                File outputImage = new File(getExternalCacheDir(), "output_image.jpg");
                try {
                    if (outputImage.exists()) {
                        outputImage.delete();
                    }
                    outputImage.createNewFile();
                } catch (IOException e) {
                    e.printStackTrace();
                }
                if (Build.VERSION.SDK_INT >= 24) {
                    imageUri = FileProvider.getUriForFile(MainActivity.this, "com/example.cameraalbumtest.fileprovider", outputImage);
                } else {
                    imageUri = Uri.fromFile(outputImage);
                }
                Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
                intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
                startActivityForResult(intent, RAKE_PHOTO);
            }
        });
        chooseFromAlbum.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {   //申請許可權
                    ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);
                } else {
                    openAlbum();
                }
            }
        });
    }

    private void openAlbum() {
        Intent intent = new Intent("android.intent.action.GET_CONTENT");//將他的action指定為“”中的
        intent.setType("image/*");
        startActivityForResult(intent, CHOOSE_PHOTO);  //開啟相簿。第二個引數值為CHOOSE_PHOTO,
    }   //當從相簿選擇完圖片回到onActivityResult方法時,就會進入CHOOSE_PHOTO的case來處理圖片

    @Override
    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
        switch (requestCode) {
            case 1:
                if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                    openAlbum();
                } else {
                    Toast.makeText(this, "You denied the permission", Toast.LENGTH_SHORT).show();
                }
                break;
            default:
        }
    }
    @Override
    protected void onActivityResult(int requestCode,int resultCode, Intent data){
        switch (requestCode){
            case TAKE_PHOTO:
                if(resultCode == RESULT_OK){
                    try{
                        //將拍攝的照片顯示出來
                        Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));
                        picture.setImageBitmap(bitmap);
                    }catch (FileNotFoundException e){
                        e.printStackTrace();
                    }
                }
                break;
            case CHOOSE_PHOTO:
                if(resultCode == RESULT_OK) {
                    if (Build.VERSION.SDK_INT >= 19) {  //進行版本判斷
                        handleImageOnKitKat(data);
                    } else {
                        handleImageBeforeKitKat(data);
                    }
                }
                break;
            default:
                break;
        }
    }
    @Target(19)
private void handleImageOnKitKat(Intent data) {   //如何解析這個封裝過的Uri
        String imagePath = null;
        Uri uri = data.getData();
        if( DocumentsContract.isDocumentUri(this,Uri)){
            String docId = DocumentsContract.getDocumentId(uri);
            if ("com.android.providers.media.documents".equals(uri.getAuthority())){
                String id = docId.split(":")[1];
                String selection = MediaStore.Images.Media._ID + "=" + id;
                imagePath = getImagePath(MediaStore.Inages.Media.EXTERNAL_CONTENT_URI,selection);
            }else if("com.android.providers.downloads.documents"。equals(uri.getAuthority())){
                Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"),Long.valueOf(docId));
                imagePath = getImagePath(contentUri, null);}
        }else if ("content".equalsIgnoreCase(uri.getScheme())){
                imagePath = getImagePath(uri,null);
        }else if ("file".equalsIgnoreCase(uri.getScheme())){
            imagePath = uri.getPath();
        }
        displayImage(imagePath);
    }
    private void handleImageBeforeKitKat(Intent data) {
        Uri uri = data.getData();
        String imagePath = getImagePath(rui,null);
        displayImage(imagePath);
    }

    private String getImagePath(Uri uri,String selection){
        String path = null;
        Cursor cursor = getContentResolver().query(uri,null,selection,null,null);
        if(cursor != null){
            if(cursor.moveToFirst()){
                path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
            }
            cursor.close();
        }
        return path;
    }
    private void displayImage(String imagePath){
        if(imagePath != null){
            Bitmap bitmap = BitmapFactory.decodeFile(imagePath);
            picture.setImageBitmap(bitmap);
        }else{
            Toast.makeText(this,"failed to get image",Toast.LENGTH_SHORT).show();
        }
    }
}

應該有錯,不過可以根據報錯修改