Android 实现 相机、相册功能 + 图片存取数据库操作

效果图:
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

MainActivity.java:

public class MainActivity extends AppCompatActivity {
    
    

    public static final int TAKE_PHOTO =1;
    public static final int CHOOSE_PHOTO=2;
    private ImageView picture;
    private Button tiao;
    private Uri imageUri;
    String index = "";		//	作为拍照还是相册的标识
    String info = "";		//	uri变成字符串


    @Override
    protected void onCreate(Bundle savedInstanceState) {
    
    
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button takePhoto = findViewById(R.id.take_photo);
        Button chooseFromAlbum = findViewById(R.id.choose_from_album);
        tiao = findViewById(R.id.btn_tiao);
        tiao.setOnClickListener(new View.OnClickListener() {
    
    
            @Override
            public void onClick(View v) {
    
    
                Intent intent = new Intent(MainActivity.this,Xian_Activity.class);
                initData();
                startActivity(intent);
            }
        });
        picture = findViewById(R.id.picture);
        //  拍照
        takePhoto.setOnClickListener(new View.OnClickListener() {
    
    
            @Override
            public void onClick(View v) {
    
    

                 //          获取本地时间,作为图片的名字,防止拍了多张照片时,出现图片覆盖导致之前图片消失的问题
                SimpleDateFormat format = new SimpleDateFormat("yyyy年MM月dd日HH:mm:ss");
                Date curDate = new Date(System.currentTimeMillis());
                String str = format.format(curDate);

                /**
                 *          创建File对象,用于存储拍照后的照片
                 *          第一个参数:  是这张照片存放在手机SD卡的对应关联缓存应用
                 *          第二个参数:  这张图片的命名
                 */
                File outputImage = new File(getExternalCacheDir(),str+".jpg");
                try {
    
    
                    if (outputImage.exists()){
    
              //  检查与File对象相连接的文件和目录是否存在于磁盘中
                        outputImage.delete();           //  删除与File对象相连接的文件和目录
                    }
                    outputImage.createNewFile();        //  如果与File对象相连接的文件不存在,则创建一个空文件
                } catch (IOException e) {
    
    
                    e.printStackTrace();
                }
                if (Build.VERSION.SDK_INT >= 24){
    
           //  如果运行设备的系统版本高于 Android7.0
                    /**
                     *          将File对象转换成一个封装过的Uri对象
                     *          第一个参数:  要求传入Context参数
                     *          第二个参数:  可以是任意唯一的字符串
                     *          第三个参数:  我们刚刚创建的File对象
                     */
                    imageUri = FileProvider.getUriForFile(MainActivity.this,"ccv.turbosnail.cameraalbumtest.fileprovider",outputImage);
                }else{
    
                      //  如果运行设备的系统版本低于 Android7.0
                    //  将File对象转换成Uri对象,这个Uri对象表示着output_image.jpg 这张图片的本地真实路径
                    imageUri = Uri.fromFile(outputImage);
                }
                /**
                 *      启动相机程序
                 */
                //  将Intent的action指定为 拍照到指定目录 —— android.media.action.IMAGE_CAPTURE
                Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
                //  指定图片的输出地址
                intent.putExtra(MediaStore.EXTRA_OUTPUT,imageUri);
                //  在通过startActivityForResult(),来启动活动,因此拍完照后会有结果返回到 onActivityResult()方法中
                startActivityForResult(intent,TAKE_PHOTO);  //  打开相机,用自定义常量 —— TAKE_PHOTO来作为case处理图片的标识
            }
        });

        //  相册
        chooseFromAlbum.setOnClickListener(new View.OnClickListener() {
    
    
            @Override
            public void onClick(View v) {
    
    
                //  动态申请WRITE_EXTERNAL_STORAGE 这个危险权限。因为相册中的照片时存储在SD卡上的,我们从SD卡中读取照片就需要申请这个权限
                //  WRITE_EXTERNAL_STORAGE  ——  同时授权程序对SD卡读和写的能力
                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 initData() {
    
    

        MySQLite dbHelper = new MySQLite(MainActivity.this,"user",null,1);
        SQLiteDatabase db = dbHelper.getReadableDatabase();
        ContentValues cv = new ContentValues();

        cv.put("PHOTO",info);
        cv.put("FLAG",index);

        db.insert("user",null,cv);
        db.close();
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
    
    
        switch (requestCode){
    
    
            case TAKE_PHOTO:
                if (resultCode == RESULT_OK){
    
    
                    try{
    
    
                            //  根据Uri找到这张照片的位置,将它解析成Bitmap对象,然后将把它设置到imageView中显示出来
                        Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));
                        info = ""+imageUri;
                        index = "2";
                        picture.setImageBitmap(bitmap);
                    } catch (FileNotFoundException e) {
    
    
                        e.printStackTrace();
                    }
                }
                break;
            case CHOOSE_PHOTO:
                /**
                 *          之所以这样做是是因为,Android 系统从4.4版本开始,选取相册图片不再返回图片真实的Uri了,而是一个封装过的Uri,因此
                 *          如果是4.4版本以上的手机就需要对这个Uri进行解析才行
                 */
                if (resultCode == RESULT_OK){
    
    
                    if (Build.VERSION.SDK_INT  >= 19){
    
          //  如果是在4.4及以 上 系统的手机就调用该方法来处理图片
                        handleImageOnKitKat(data);
                    }else{
    
    
                        handleImageBeforeKitKat(data);      //  如果是在4.4以 下 系统的手机就调用该方法来处理图片
                    }
                }
                break;
                default:
                    break;
        }
    }

    /**
     *          如何解析这个封装过的Uri
     */
    @RequiresApi(api = Build.VERSION_CODES.KITKAT)
    private void handleImageOnKitKat(Intent data) {
    
    
        String imagePath = null;
        Uri uri = data.getData();
        if (DocumentsContract.isDocumentUri(this,uri)){
    
    
            //  如果返回的Uri是 document 类型的话,那就取出 document id 进行处理
            String docId = DocumentsContract.getDocumentId(uri);
            if ("com.android.providers.media.documents".equals(uri.getAuthority())){
    
    
                String id = docId.split(":")[1];        //  解析出数字格式id
                String selection = MediaStore.Images.Media._ID + "=" + id;
                imagePath = getImagePath(MediaStore.Images.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())){
    
    
            //  如果是 content 类型的Uri,则使用普通方式处理
            imagePath = getImagePath(uri,null);
        }else if ("file".equalsIgnoreCase(uri.getScheme())){
    
    
            //  如果是 file 类型的 Uri ,直接获取图片路径即可
            imagePath = uri.getPath();
        }
        displayImage(imagePath);        //  拿到图片路径后,在调用 displayImage() 方法将图片显示到界面上
    }

    private void handleImageBeforeKitKat(Intent data) {
    
    
        Uri uri = data.getData();
        String imagePath = getImagePath(uri,null);
        displayImage(imagePath);
    }

    private void displayImage(String imagePath) {
    
    
        if (imagePath != null){
    
    
            Bitmap bitmap = BitmapFactory.decodeFile(imagePath);
            index = "1";
            info = imagePath;
            picture.setImageBitmap(bitmap);
        }else{
    
    
            Toast.makeText(this,"failed to gei image",Toast.LENGTH_SHORT).show();
        }
    }

    private String getImagePath(Uri uri, String selection) {
    
    
        String path = null;
        //  通过 Uri 和 selection 来获取真实图片的路径
        Cursor cursor = getContentResolver().query(uri,null,selection,null,null);
        if (cursor != null){
    
    
            if (cursor.moveToNext()){
    
    
                //  MediaStore.Images.Media.insertImage —— 得到保存图片的原始路径
                path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
            }
            cursor.close();
        }
        return path;
    }


    private void openAlbum() {
    
    
        /**
         *      启动相册程序
         */
        //  action —— android.intent.action.GET_CONTENT
        Intent intent = new Intent("android.intent.action.GET_CONTENT");
        //  该函数表示要查找文件的mime类型(如*/*),这个和组件在manifest里定义的相对应,但在源代码里
        intent.setType("image/*");
        startActivityForResult(intent,CHOOSE_PHOTO);    //打开相册,用自定义常量 —— CHOOSE_PHOTO来作为case处理图片的标识
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull 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:
       }
    }
}

MySQLite.java

public class MySQLite extends SQLiteOpenHelper {
    
    
    public MySQLite(Context context, String name, SQLiteDatabase.CursorFactory factory, int version) {
    
    
        super(context, name, factory, version);
    }

    @Override
    public void onCreate(SQLiteDatabase db) {
    
    
        db.execSQL("CREATE TABLE user(_id INTEGER PRIMARY KEY AUTOINCREMENT," +
                "PHOTO TEXT," +
                "FLAG TEXT)");
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
    
    

    }
}

Xian_Activity.java

public class Xian_Activity extends AppCompatActivity {
    
    
    private Button btnText;
    private ImageView picture;
    String info = "",index = "";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
    
    
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_xian_);
        picture = findViewById(R.id.img_picture);
        btnText = findViewById(R.id.btn_text);

        btnText.setOnClickListener(new View.OnClickListener() {
    
    
            @Override
            public void onClick(View v) {
    
    
                initData();
            }
        });
    }

    private void initData() {
    
    

        MySQLite dbHelper = new MySQLite(Xian_Activity.this,"user",null,1);
        SQLiteDatabase db = dbHelper.getReadableDatabase();
        Cursor cursor = db.query("user",null,null,null,null,null,null);

        while (cursor.moveToNext()){
    
    
            index = cursor.getString(cursor.getColumnIndex("FLAG"));
            info = cursor.getString(cursor.getColumnIndex("PHOTO"));

            if (index.equals("1")){
    
    		//	相册
                Bitmap bitmap = BitmapFactory.decodeFile(info);
                    picture.setImageBitmap(bitmap);
            }else if (index.equals("2")){
    
    	//	拍照
                Uri imgUri = Uri.parse(info);
                    try {
    
    
                        Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imgUri));
                        picture.setImageBitmap(bitmap);
                    } catch (FileNotFoundException e) {
    
    
                        e.printStackTrace();
                    }
            }
        }

    }
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout  xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">
    
    <Button
        android:id="@+id/btn_tiao"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:text="跳转"/>

    <Button
        android:id="@+id/take_photo"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:text="拍照"/>

    <Button
        android:id="@+id/choose_from_album"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="相册"/>

    <ImageView
        android:id="@+id/picture"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"/>

</LinearLayout>

activity_xian.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".Xian_Activity">

    <Button
        android:id="@+id/btn_text"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="查看"/>

    <ImageView
        android:id="@+id/img_picture"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"/>

</LinearLayout>

点击运行即可!
最后,我真诚的希望能评论一句嘛,让我知道你来过,我会很开心的

猜你喜欢

转载自blog.csdn.net/qq_27494201/article/details/96170013#comments_26167070