在一个应用中更换用户的头像,一般有拍照和从图库中选择照片两种方法,现在网上也有很多开源的,但是很多都太复杂。而 Android-crop 这个库比较小,代码不复杂,比较适合,但是它没有拍照这个功能,需要我们自己整合进去。
// 调用系统的拍照
private void dispatchTakePictureIntent() {
Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE);
}
}
重写 onActivityResult 方法,系统相机会返回略缩图
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == RESULT_OK) {
Bundle extras = data.getExtras();
Bitmap imageBitmap = (Bitmap) extras.get("data");
mImageView.setImageBitmap(imageBitmap);
}
}
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="18"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
在4.4 及其以上不需要 WRTE_EXTERNAL_STORAGE 权限,所以,我们添加了 android: maxSdkVersion = “18”
. 因为是全图片,图片一般都会很大,我们不能再像上面一样全部数据返回,不然很容易就出现 OOM , 所以,启动拍照的时候设置一个指定路径保存图片,拍照成功之后就可以直接使用这个路径了
// 图片路径
private Uri mCurrentPhotoUri;
// 拍照
private void dispatchTakePictureIntent() {
Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
if (takePictureIntent.resolveActivity(mContext.getPackageManager()) != null) {
File photoFile = null;
try {
photoFile = createImageFile();
} catch (IOException e) {
e.printStackTrace();
}
if (photoFile != null){
Uri photoUri = FileProvider.getUriForFile(mContext, "com.yxhuang.fileprovider", photoFile);
takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);
mCurrentPhotoUri = photoUri;
startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE);
}
}
}
// 创建图片路径
private File createImageFile() throws IOException {
String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());
String imageFileName = "JPEG_" + timeStamp + "_";
File storageDir = mContext.getExternalFilesDir(Environment.DIRECTORY_PICTURES);
File image = File.createTempFile(
imageFileName, /* prefix */
".jpg", /* suffix */
storageDir /* directory */
);
return image;
}
在创建图片 uri 的时候,使用了 FileProvider, FileProvider 只是 ContentProvider 的一个子类。用于方便文件的共享。
FileProvider 需要一些配置
在 Manifext.xml 中 在 节点下
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.yxhuang.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths">
meta-data>
provider>
其中 authorities 的内容要与 FileProvider.getUriForFile()方法中的第二个参数相同。
同时也要对android:resource 进行配置
res/xml/file_paths.xml
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path name="image_file"
path="Android/data/com.yxhuang/files/Pictures"/>
paths>
Android-crop 是一个比较简单的图片裁剪库。具体使用看 GitHub 的 Demo. 我们这里结合拍照一起使用。
上面我们已经准备好拍照了,我们需要在 onActivityForResult() 方法中进行处理
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
// 拍照返回
if (resultCode == RESULT_OK){
if (requestCode == REQUEST_IMAGE_CAPTURE){
beginCrop(mCurrentPhotoUri);
} else if (requestCode == Crop.REQUEST_PICK){
beginCrop(data.getData());
}
}
// 裁剪
if (requestCode == Crop.REQUEST_CROP){
handleCrop(resultCode, data);
}
}
// 开始裁剪
private void beginCrop(Uri source){
Uri destination = Uri.fromFile(new File(mContext.getCacheDir(), "cropped"));
// start() 方法根据其的需求选择不同的重载方法
Crop.of(source, destination).asSquare().start(getActivity(), MineFragment.this);
}
// 将裁剪回来的数据进行处理
private void handleCrop(int resultCode, Intent result){
if (resultCode == RESULT_OK){
mImageView.setImageURI(Crop.getOutput(result));
} else if (resultCode == Crop.RESULT_ERROR){
Toast.makeText(mContext,
Crop.getError(result).getMessage(),
Toast.LENGTH_SHORT).show();
}
}