转载请注明出处:http://blog.csdn.net/allen315410/article/details/39994913

最近从以前的项目中扒下来一个常用的模块,在这里有必要记录一下的,就是android上获取图片以及裁剪图片,怎么样?这个功能是不是很常用啊,你随便打开一个App,只要它有注册功能都会有设置人物头像的功能,尤其在内容型的app中更为常见,那么这些功能是怎么实现的呢?今天,在这里就记录一下好了,防止以后的项目中也会用到,就直接拿来用好了。

1.通过拍照或者图册获取图片(不需要剪裁)

这种获取图片的方式就比较次了,因为不设置图片的剪裁功能,有可能因为图片过大,导致OOM,但是这种方式也是有必要讲一下的,其获取图片的方式有两种,一是调用系统相机实时拍摄一张图片,二十打开设备上已有的图库,在图库中选择一张照片。这两种方式实现方法都是一个道理,无非就是通过Intent调用系统的东西。下面是源码,首先是图片选择方式的Activity,这个Activity被设置成了Dialog模式,需要进行设置一下。

布局文件/res/layout/activity_select_photo.xml:

<?xml version="1.0" encoding="utf-8"?><RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="fill_parent"    android:layout_height="wrap_content"    android:gravity="center_horizontal" >    <LinearLayout        android:id="@+id/dialog_layout"        android:layout_width="fill_parent"        android:layout_height="wrap_content"        android:layout_alignParentBottom="true"        android:layout_marginLeft="10dip"        android:layout_marginRight="10dip"        android:gravity="center_horizontal"        android:orientation="vertical" >        <LinearLayout            android:layout_width="fill_parent"            android:layout_height="wrap_content"            android:background="@drawable/select_photo_up_bg"            android:orientation="vertical"            android:paddingBottom="5dp"            android:paddingTop="5dp" >            <Button                android:id="@+id/btn_take_photo"                android:layout_width="fill_parent"                android:layout_height="35dp"                android:background="@drawable/select_photo_bg"                android:text="拍照选取"                android:textStyle="bold" />            <View                android:layout_width="fill_parent"                android:layout_height="0.5px"                android:background="#828282" />            <Button                android:id="@+id/btn_pick_photo"                android:layout_width="fill_parent"                android:layout_height="35dp"                android:layout_marginTop="0dip"                android:background="@drawable/select_photo_bg"                android:text="相册选取"                android:textStyle="bold" />        </LinearLayout>        <Button            android:id="@+id/btn_cancel"            android:layout_width="fill_parent"            android:layout_height="35dp"            android:layout_marginTop="20dip"            android:background="@drawable/select_photo_bg"            android:paddingBottom="5dp"            android:paddingTop="5dp"            android:text="取消"            android:textColor="#ffff0000"            android:textStyle="bold" />    </LinearLayout></RelativeLayout>
接着是获取图片Activity里的代码SelectPhotoActivity:

public class SelectPhotoActivity extends Activity implements OnClickListener {/** 使用照相机拍照获取图片 */public static final int SELECT_PIC_BY_TACK_PHOTO = 1;/** 使用相册中的图片 */public static final int SELECT_PIC_BY_PICK_PHOTO = 2;/** 开启相机 */private Button btn_take_photo;/** 开启图册 */private Button btn_pick_photo;/** 取消 */private Button btn_cancel;/** 获取到的图片路径 */private String picPath;private Intent lastIntent;private Uri photoUri;/** 从Intent获取图片路径的KEY */public static final String KEY_PHOTO_PATH = "photo_path";@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_select_photo);btn_take_photo = (Button) findViewById(R.id.btn_take_photo);btn_pick_photo = (Button) findViewById(R.id.btn_pick_photo);btn_cancel = (Button) findViewById(R.id.btn_cancel);lastIntent = getIntent();btn_take_photo.setOnClickListener(this);btn_pick_photo.setOnClickListener(this);btn_cancel.setOnClickListener(this);}@Overridepublic void onClick(View v) {switch (v.getId()) {case R.id.btn_take_photo : // 开启相机takePhoto();break;case R.id.btn_pick_photo : // 开启图册pickPhoto();break;case R.id.btn_cancel : // 取消操作this.finish();break;default :break;}}/** * 拍照获取图片 */private void takePhoto() {// 执行拍照前,应该先判断SD卡是否存在String SDState = Environment.getExternalStorageState();if (SDState.equals(Environment.MEDIA_MOUNTED)) {Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);// "android.media.action.IMAGE_CAPTURE"/*** * 需要说明一下,以下操作使用照相机拍照,拍照后的图片会存放在相册中的 这里使用的这种方式有一个好处就是获取的图片是拍照后的原图 * 如果不实用ContentValues存放照片路径的话,拍照后获取的图片为缩略图不清晰 */ContentValues values = new ContentValues();photoUri = this.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);intent.putExtra(android.provider.MediaStore.EXTRA_OUTPUT, photoUri);startActivityForResult(intent, SELECT_PIC_BY_TACK_PHOTO);} else {Toast.makeText(getApplicationContext(), "内存卡不存在",Toast.LENGTH_SHORT).show();}}/*** * 从相册中取图片 */private void pickPhoto() {Intent intent = new Intent();intent.setType("image/*");intent.setAction(Intent.ACTION_GET_CONTENT);startActivityForResult(intent, SELECT_PIC_BY_PICK_PHOTO);}@Overridepublic boolean onTouchEvent(MotionEvent event) {finish();return super.onTouchEvent(event);}@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {if (resultCode == Activity.RESULT_OK) {doPhoto(requestCode, data);}super.onActivityResult(requestCode, resultCode, data);}/** * 选择图片后,获取图片的路径 *  * @param requestCode * @param data */private void doPhoto(int requestCode, Intent data) {if (requestCode == SELECT_PIC_BY_PICK_PHOTO) {// 从相册取图片,有些手机有异常情况,请注意if (data == null) {Toast.makeText(getApplicationContext(), "选择图片文件出错",Toast.LENGTH_SHORT).show();return;}photoUri = data.getData();if (photoUri == null) {Toast.makeText(getApplicationContext(), "选择图片文件出错",Toast.LENGTH_SHORT).show();return;}}String[] pojo = {MediaStore.Images.Media.DATA};Cursor cursor = managedQuery(photoUri, pojo, null, null, null);if (cursor != null) {int columnIndex = cursor.getColumnIndexOrThrow(pojo[0]);cursor.moveToFirst();picPath = cursor.getString(columnIndex);cursor.close();}if (picPath != null&& (picPath.endsWith(".png") || picPath.endsWith(".PNG")|| picPath.endsWith(".jpg") || picPath.endsWith(".JPG"))) {lastIntent.putExtra(KEY_PHOTO_PATH, picPath);setResult(Activity.RESULT_OK, lastIntent);finish();} else {Toast.makeText(getApplicationContext(), "选择图片文件不正确",Toast.LENGTH_SHORT).show();}}}
因为这Activity是要设置成Dialog模式的,所以需要在清单文件中设置一下style,/res/values/styles.xml里添加如下:

<!-- 选取照片的Activity的样式风格,采取对话框的风格 -->    <style name="AnimBottom" parent="@android:style/Animation">        <item name="android:windowEnterAnimation">@anim/push_bottom_in</item>        <item name="android:windowExitAnimation">@anim/push_bottom_out</item>    </style>    <style name="DialogStyleBottom" parent="android:Theme.Dialog">        <item name="android:windowAnimationStyle">@style/AnimBottom</item>        <item name="android:windowFrame">@null</item>        <!-- 边框 -->        <item name="android:windowIsFloating">false</item>        <!-- 是否浮现在activity之上 -->        <item name="android:windowIsTranslucent">true</item>        <!-- 半透明 -->        <item name="android:windowNoTitle">true</item>        <!-- 无标题 -->        <item name="android:windowBackground">@android:color/transparent</item>        <!-- 背景透明 -->        <item name="android:backgroundDimEnabled">true</item>        <!-- 模糊 -->    </style>
在Activity的节点下,设置这个style:

<activity            android:name="com.example.croppictrue.SelectPhotoActivity"            android:screenOrientation="portrait"            android:theme="@style/DialogStyleBottom" >        </activity>
添加权限:

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
运行效果如下:

2.通过拍照或者图册获取图片(需要剪裁)

上面第一种方式获取图片是没有经过剪裁的,但是大多项目需求是需要剪裁图片后再使用,例如修改用户头像等等功能。那么,下面,就奉上剪裁图片的代码吧:

public class CropPictureActivity extends Activity {/** ImageView对象 */private ImageView iv_photo;private String[] items = new String[]{"选择本地图片", "拍照"};/** 头像名称 */private static final String IMAGE_FILE_NAME = "image.jpg";/** 请求码 */private static final int IMAGE_REQUEST_CODE = 0;private static final int CAMERA_REQUEST_CODE = 1;private static final int RESULT_REQUEST_CODE = 2;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_crop);iv_photo = (ImageView) findViewById(R.id.iv_photo);iv_photo.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {showDialog();}});}/** * 显示选择对话框 */private void showDialog() {new AlertDialog.Builder(this).setTitle("设置头像").setItems(items, new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {switch (which) {case 0 :Intent intentFromGallery = new Intent();intentFromGallery.setType("image/*"); // 设置文件类型intentFromGallery.setAction(Intent.ACTION_GET_CONTENT);startActivityForResult(intentFromGallery,IMAGE_REQUEST_CODE);break;case 1 :Intent intentFromCapture = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);// 判断存储卡是否可以用,可用进行存储String state = Environment.getExternalStorageState();if (state.equals(Environment.MEDIA_MOUNTED)) {File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM);File file = new File(path, IMAGE_FILE_NAME);intentFromCapture.putExtra(MediaStore.EXTRA_OUTPUT,Uri.fromFile(file));}startActivityForResult(intentFromCapture,CAMERA_REQUEST_CODE);break;}}}).setNegativeButton("取消", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {dialog.dismiss();}}).show();}@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {// 结果码不等于取消时候if (resultCode != RESULT_CANCELED) {switch (requestCode) {case IMAGE_REQUEST_CODE :startPhotoZoom(data.getData());break;case CAMERA_REQUEST_CODE :// 判断存储卡是否可以用,可用进行存储String state = Environment.getExternalStorageState();if (state.equals(Environment.MEDIA_MOUNTED)) {File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM);File tempFile = new File(path, IMAGE_FILE_NAME);startPhotoZoom(Uri.fromFile(tempFile));} else {Toast.makeText(getApplicationContext(),"未找到存储卡,无法存储照片!", Toast.LENGTH_SHORT).show();}break;case RESULT_REQUEST_CODE : // 图片缩放完成后if (data != null) {getImageToView(data);}break;}}super.onActivityResult(requestCode, resultCode, data);}/** * 裁剪图片方法实现 *  * @param uri */public void startPhotoZoom(Uri uri) {Intent intent = new Intent("com.android.camera.action.CROP");intent.setDataAndType(uri, "image/*");// 设置裁剪intent.putExtra("crop", "true");// aspectX aspectY 是宽高的比例intent.putExtra("aspectX", 1);intent.putExtra("aspectY", 1);// outputX outputY 是裁剪图片宽高intent.putExtra("outputX", 340);intent.putExtra("outputY", 340);intent.putExtra("return-data", true);startActivityForResult(intent, RESULT_REQUEST_CODE);}/** * 保存裁剪之后的图片数据 *  * @param picdata */private void getImageToView(Intent data) {Bundle extras = data.getExtras();if (extras != null) {Bitmap photo = extras.getParcelable("data");Drawable drawable = new BitmapDrawable(this.getResources(), photo);iv_photo.setImageDrawable(drawable);}}}
效果图:

在这个Activity里为了简便处理,我没有在选择图片时候start一个Dialog风格的Activity了,就直接一个普通的对话框提示用户选择,效果也许。其实实现的原理都比较简单,实现图片的剪裁就是发一个Intent请求,调用设备上所有具有剪裁图片功能的app去剪裁图片,我的设备上除了android系统自带的图库以外,还装有“快图浏览”这个app,这个app也自带一个图片剪裁的功能,所有当选择好图片后,会出现一个选择提示,用户可以根据提示选择到底使用哪个app提供的剪裁功能区剪裁图片。
以上代码均在模拟器上测试过,由于模拟器对相机支持的不好,所以就没有演示打开相机拍摄图片了,有兴趣的朋友可以先请下载这个Demo的源码,运行在手机上试试看效果如何,如若疏漏之后,欢迎大家批评指正!


源码请在这里下载


更多相关文章

  1. Android笔记:AlbumSaver图片视频保存工具类
  2. Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右
  3. 【Android】说做就做:带图标的list item
  4. Android中Fragment的强大功能
  5. Android(安卓)大图预览、高效大图加载
  6. android显示缓存大小和清除缓存功能
  7. 转载 Android仿今日头条详情页实现
  8. Android(安卓)PNG渐变背景图片失真问题
  9. 【android】ImageView的src和background的区别以及两者的妙用

随机推荐

  1. Android键盘映射
  2. android使用notifyDataSetChanged()方法,a
  3. Android(安卓)封装自己的工具Jar
  4. Android(安卓)P 系统启动-System Server
  5. android设置Webview的滚动条属性
  6. sdut 2162:The Android(安卓)University
  7. Android消息机制
  8. android textview属性
  9. 教你如何创建类似QQ的android弹出菜单
  10. Android中intent如何传递自定义数据类型