本文鏈接 http://blog.csdn.net/xiaodongrush/article/details/29173567
1. 緣起
要開發一個頭像上傳的模塊,頭像上傳過程分兩步。第一步,相機拍照或者從圖庫選取照片,產生一個照片,第二步,提供頭像剪裁,一般是剪裁爲方形的。第三步,上傳頭像,刪除不必要的緩存文件。
拍照和圖庫選擇照片都可以使用系統的方案。自制相機可以搞濾鏡,這個開發成本比較大,一般的APP也不用支持。圖庫選擇照片這個可以自己做,訪問sd卡,比較簡單。問題出在圖片剪裁上。網上有一些技術方案,遷移過來之後,效果不好,比如縮放的敏感度問題,縮放之後剪裁不準確的問題,縮放不流暢的問題。後來發現使用com.android.camera.action.CROP可以調用系統剪裁頁面,但是該頁面不是官方公開的頁面,所以,某些廠商可能不支持這個。T_T。
看了下幾款APP的頭像截取,QQ、微信和易信都是自己做的,效果也不是很好,360手機助手是調用的系統的。系統的截取比較流暢,效果較好,除了前面的隱患,還有一個問題就是,在一些定製手機上面,剪裁的頁面整體比較暗,或者上面有一層陰影,當然剪裁完之後圖片是正常的。
考慮到360手機助手用戶量這麼大的APP,也在使用系統剪裁,所以我們也考慮使用系統剪裁,如果發現系統剪裁不可用,再調用自己的剪裁。
2. 拍照代碼
這裏用了MediaStore.EXTRA_OUTPUT,拍照圖像不會通過intnet返回,要通過uri來讀取,這樣可以讀小一點的圖像進來。如果不這樣,系統會在intent裏面返回一個壓縮圖片,這個壓縮的圖像有多大不是能夠控制的,所以可能比較大。
Intent newIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
newIntent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
startActivityForResult(newIntent, REQUEST_CODE_TAKE_PHOTO);
3. 方形剪裁
這裏的代碼與拍照類似,也用了MediaStore.EXTRA_OUTPUT,處理結果要通過路徑來讀取。
Intent intent = new Intent("com.android.camera.action.CROP");
intent.setDataAndType(uri, "image/*");
intent.putExtra("crop", "true"); // 開啓剪裁
intent.putExtra("aspectX", 1); // 寬高比例
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 150); // 寬高
intent.putExtra("outputY", 150);
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(mOutputFile.getAbsoluteFile() + "tmp")));
startActivityForResult(intent, REQUEST_CODE_CLIP_PHOTO);
4. 相對完整的代碼
代碼下載鏈接,內附APK http://download.csdn.net/detail/u011267546/7460367
public class MainActivity extends Activity {
private static final int REQUEST_CODE_TAKE_PHOTO = 0;
private static final int REQUEST_CODE_CLIP_PHOTO = 1;
private File mOutputFile;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout. activity_main);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == REQUEST_CODE_TAKE_PHOTO) {
onTakePhotoFinished(resultCode, data);
} else if (requestCode == REQUEST_CODE_CLIP_PHOTO ) {
onClipPhotoFinished(resultCode, data);
}
}
public void onClick(View v) {
if (v.getId() == R.id.take_photo) {
if (hasCarema() == false) {
return;
}
takePhoto();
}
}
private boolean hasCarema() {
PackageManager pm = getPackageManager();
if (!pm.hasSystemFeature(PackageManager.FEATURE_CAMERA )
&& !pm.hasSystemFeature(PackageManager.FEATURE_CAMERA_FRONT )) {
Toast. makeText(this, "no camera found", Toast.LENGTH_SHORT).show();
return false ;
}
return true ;
}
private void takePhoto() {
String sdPath = Environment.getExternalStorageDirectory()
.getAbsolutePath();
mOutputFile = new File(sdPath, System.currentTimeMillis() + ".tmp");
Uri uri = Uri. fromFile(mOutputFile);
Intent newIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE );
newIntent.putExtra(MediaStore. EXTRA_OUTPUT, uri);
startActivityForResult(newIntent, REQUEST_CODE_TAKE_PHOTO );
}
private void onTakePhotoFinished(int resultCode, Intent data) {
if (resultCode == RESULT_CANCELED) {
Toast. makeText(this, "take photo canceled", Toast.LENGTH_SHORT)
.show();
return;
} else if (resultCode != RESULT_OK) {
Toast. makeText(this, "take photo failed", Toast.LENGTH_SHORT)
.show();
} else {
clipPhoto(Uri. fromFile(mOutputFile));
}
}
// http://www.xuanyusong.com/archives/1743
private void clipPhoto(Uri uri) {
Intent intent = new Intent("com.android.camera.action.CROP" );
intent.setDataAndType(uri, "image/*");
// 下面這個crop=true是設置在開啓的Intent中設置顯示的VIEW可裁剪
intent.putExtra( "crop", "true" );
// aspectX aspectY 是寬高的比例
intent.putExtra( "aspectX", 1);
intent.putExtra( "aspectY", 1);
// outputX outputY 是裁剪圖片寬高
intent.putExtra( "outputX", 150);
intent.putExtra( "outputY", 150);
intent.putExtra(MediaStore. EXTRA_OUTPUT,
Uri. fromFile(new File(mOutputFile.getAbsoluteFile() + "tmp" )));
startActivityForResult(intent, REQUEST_CODE_CLIP_PHOTO );
}
private void onClipPhotoFinished(int resultCode, Intent data) {
if (resultCode == RESULT_CANCELED) {
Toast. makeText(this, "clip photo canceled", Toast.LENGTH_SHORT)
.show();
return;
} else if (resultCode != RESULT_OK) {
Toast. makeText(this, "take photo failed", Toast.LENGTH_SHORT)
.show();
}
Bitmap bm = BitmapFactory.decodeFile(mOutputFile.getAbsolutePath()
+ "tmp");
ImageView photoIv = (ImageView) findViewById(R.id.photo );
photoIv.setImageBitmap(bm);
}
}
5. 效果圖
獻上可愛的大熊熊,只截取大熊的上半身。