`
yxwang0615
  • 浏览: 560811 次
  • 性别: Icon_minigender_1
  • 来自: 青岛
社区版块
存档分类
最新评论

从相册或拍照选择照片到EditText

 
阅读更多

【方法一】使用自定义AlertDialog

 

private static final int PHOTO_SUCCESS = 1;
private static final int CAMERA_SUCCESS = 2; 
private ImageButton pic; //图片选择按钮

// 从相册或相机选择图片
pic.setOnClickListener(new View.OnClickListener() {

	@Override
	public void onClick(View v) {
		final CharSequence[] items = { "手机相册", "相机拍摄" };
		AlertDialog dlg = new AlertDialog.Builder(NewTopic.this).setTitle("选择图片").setItems(items, 
			new DialogInterface.OnClickListener() { 
				public void onClick(DialogInterface dialog,int item) { 
					//这里item是根据选择的方式,
					//在items数组里面定义了两种方式, 拍照的下标为1所以就调用拍照方法       
					if(item==1){ 
						Intent getImageByCamera= new Intent("android.media.action.IMAGE_CAPTURE");   
						startActivityForResult(getImageByCamera, CAMERA_SUCCESS);   
					}else{ 
						Intent getImage = new Intent(Intent.ACTION_GET_CONTENT); 
						getImage.addCategory(Intent.CATEGORY_OPENABLE); 
						getImage.setType("image/*"); 
						startActivityForResult(getImage, PHOTO_SUCCESS); 
					 } 
				} 
			}).create(); 
		dlg.show(); 
	}
});


protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
	ContentResolver resolver = getContentResolver(); 
	if (resultCode == RESULT_OK) {
		switch (requestCode) {
		case PHOTO_SUCCESS:
			//获得图片的uri 
			Uri originalUri = intent.getData(); 
			Bitmap bitmap = null;
			try {
				Bitmap originalBitmap = BitmapFactory.decodeStream(resolver.openInputStream(originalUri));
				bitmap = resizeImage(originalBitmap, 200, 200);
			} catch (FileNotFoundException e) {
				e.printStackTrace();
			}
			if(bitmap != null){
				//根据Bitmap对象创建ImageSpan对象
				ImageSpan imageSpan = new ImageSpan(NewTopic.this, bitmap);
				//创建一个SpannableString对象,以便插入用ImageSpan对象封装的图像
				SpannableString spannableString = new SpannableString("[local]"+1+"[/local]");
				//  用ImageSpan对象替换face
				spannableString.setSpan(imageSpan, 0, "[local]1[local]".length()+1, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
				//将选择的图片追加到EditText中光标所在位置
				int index = edit.getSelectionStart(); //获取光标所在位置
				Editable edit_text = edit.getEditableText();
				if(index <0 || index >= edit_text.length()){
					edit_text.append(spannableString);
				}else{
					edit_text.insert(index, spannableString);
				}
			}else{
				Toast.makeText(NewTopic.this, "获取图片失败", Toast.LENGTH_SHORT).show();
			}
			break;
		case CAMERA_SUCCESS:
			Bundle extras = intent.getExtras(); 
			Bitmap originalBitmap1 = (Bitmap) extras.get("data");
			if(originalBitmap1 != null){
				bitmap = resizeImage(originalBitmap1, 200, 200);
				//根据Bitmap对象创建ImageSpan对象
				ImageSpan imageSpan = new ImageSpan(NewTopic.this, bitmap);
				//创建一个SpannableString对象,以便插入用ImageSpan对象封装的图像
				SpannableString spannableString = new SpannableString("[local]"+1+"[/local]");
				//  用ImageSpan对象替换face
				spannableString.setSpan(imageSpan, 0, "[local]1[local]".length()+1, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
				//将选择的图片追加到EditText中光标所在位置
				int index = edit.getSelectionStart(); //获取光标所在位置
				Editable edit_text = edit.getEditableText();
				if(index <0 || index >= edit_text.length()){
					edit_text.append(spannableString);
				}else{
					edit_text.insert(index, spannableString);
				}
			}else{
				Toast.makeText(NewTopic.this, "获取图片失败", Toast.LENGTH_SHORT).show();
			}
			break;
		default:
			break;
		}
	}
}
/**
 * 图片缩放
 * @param originalBitmap 原始的Bitmap
 * @param newWidth 自定义宽度
 * @param newHeight自定义高度
 * @return 缩放后的Bitmap
 */
private Bitmap resizeImage(Bitmap originalBitmap, int newWidth, int newHeight){
	int width = originalBitmap.getWidth();
	int height = originalBitmap.getHeight();
	//定义欲转换成的宽、高
//		int newWidth = 200;
//		int newHeight = 200;
	//计算宽、高缩放率
	float scanleWidth = (float)newWidth/width;
	float scanleHeight = (float)newHeight/height;
	//创建操作图片用的matrix对象 Matrix
	Matrix matrix = new Matrix();
	// 缩放图片动作
	matrix.postScale(scanleWidth,scanleHeight);
	//旋转图片 动作
	//matrix.postRotate(45);
	// 创建新的图片Bitmap
	Bitmap resizedBitmap = Bitmap.createBitmap(originalBitmap,0,0,width,height,matrix,true);
	return resizedBitmap;
}

alertDialog 的运行效果类似这样:

 

 

【方法二】使用Intent.createChooser

 还可以使用Intent选择器+隐式Intent的方法达到效果,只是这种方式比较繁琐,界面效果也差点。

 首先,定义两个Activity来接受隐式intent, AndroidManifest.xml:

       <activity android:name=".activity.action_get_content.PickPicFromLocalFile" android:label="手机相册">     
            <intent-filter>     
                <action android:name="open_pic_intent" />
                 <category android:name="android.intent.category.DEFAULT" />     
                 <category android:name="android.intent.category.OPENABLE" />     
                 <data android:mimeType="image/*" />     
            </intent-filter>     
        </activity>
        <activity android:name=".activity.action_get_content.PickPicFromCamera" android:label="相机拍摄" android:icon="@drawable/logo1"> 
            <intent-filter>     
                <action android:name="open_pic_intent" />     
                 <category android:name="android.intent.category.DEFAULT" />     
                 <category android:name="android.intent.category.OPENABLE" />     
                 <data android:mimeType="image/*" />     
            </intent-filter>     
        </activity>  
 

 

 PickPicFromLocalFile.java:

 调用了系统的打开本地图库的方法,把intent传给了我们自定义的页面,自定义页面再把Intent传给事件源页面。

 

/**
 * 手机相册 使用Intent.createChooser弹出的菜单
 * @author wangyx
 * @version 1.0.0 2011-12-16
 */

public class PickPicFromLocalFile extends Activity {
	
	protected static final int PICRESULT_LOCAL = 0;
	private Intent intent;
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		// TODO Auto-generated method stub
		super.onCreate(savedInstanceState);
		intent = getIntent();
		Intent innerIntent = new Intent(Intent.ACTION_GET_CONTENT);   
		//action_get_content是通过intent中设置的type属性来判断具体调用哪个程序的
		innerIntent.setType("image/*"); 
		startActivityForResult(innerIntent,PICRESULT_LOCAL);
	}
	
	@Override 
	public void onActivityResult(int reqCode, int resultCode, Intent data){
		super.onActivityResult(reqCode, resultCode, data); 
		if(resultCode == RESULT_OK){
			 switch (reqCode) {
		         case (PICRESULT_LOCAL):  
	 	           intent.setData(data.getData());
		           intent.putExtra("flag", 1); //回传用于判断是哪个网页传递的数据
	        	   setResult(RESULT_OK, intent); 
		           finish();
		         break;  
			 }
		} 
	}
}

 PickPicFromCamera.java:调用系统打开相机的方法,照片的数据保存在intent的bundle中,key值为"data"。

 

/**
 * 手机相册 使用Intent.createChooser弹出的菜单
 * @author wangyx
 * @version 1.0.0 2011-12-16
 */

public class PickPicFromCamera extends Activity {
	protected static final int PICRESULT_CAMERA = 0;
	private Intent intent;
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		// TODO Auto-generated method stub
		super.onCreate(savedInstanceState);
		intent = this.getIntent();
		Intent getImageByCamera= new Intent("android.media.action.IMAGE_CAPTURE");   
        startActivityForResult(getImageByCamera, PICRESULT_CAMERA); 
	}
	@Override 
	public void onActivityResult(int reqCode, int resultCode, Intent data){
		super.onActivityResult(reqCode, resultCode, data); 
		if(resultCode == RESULT_OK){
			 switch (reqCode) {
		         case (PICRESULT_CAMERA):  
		        	Bundle extras = data.getExtras(); 
		         	intent.putExtra("flag", 2);
		         	intent.putExtras(extras);
	        	    setResult(RESULT_OK, intent); 
		            finish();
		         break;  
			 }
		} 
	}
}

 处理回传的数据,显示在ImageView上:

 

protected static final int PICRESULT_CODE = 0;
private ImageButton pic; //图片选择按钮
private ImageView iv;

// 从相册或相机选择图片
pic.setOnClickListener(new View.OnClickListener() {
	Intent wrapperIntent = new Intent("open_pic_intent"); 
	wrapperIntent.setType("image/*");  
	startActivityForResult(Intent.createChooser(wrapperIntent, "设置"), PICRESULT_CODE);
}

@Override 
	public void onActivityResult(int reqCode, int resultCode, Intent data){
		 ContentResolver resolver = getContentResolver();
		if(resultCode == RESULT_OK){
			 switch (reqCode) {
			case PICRESULT_CODE:
				Bitmap myBitmap = null;
				if(data.getIntExtra("flag", 0) == 1){
					//处理本地图库返回的数据
					   try { 
				            //获得图片的uri 
				            Uri originalUri = data.getData(); 
				            //将图片内容解析成字节数组 
				            byte[] mContent=readStream(resolver.openInputStream(Uri.parse(originalUri.toString()))); 
				            //将字节数组转换为ImageView可调用的Bitmap对象 
				            myBitmap = getPicFromBytes(mContent, null); 
				        } catch (Exception e) { 
				            System.out.println(e.getMessage()); 
				        } 
				}else if(data.getIntExtra("flag", 0) == 2){
					  try { 
	                        Bundle extras = data.getExtras(); 
	                        myBitmap = (Bitmap) extras.get("data"); 
	                        ByteArrayOutputStream baos = new ByteArrayOutputStream();     
	                        myBitmap.compress(Bitmap.CompressFormat.JPEG , 100, baos);     
					  } catch (Exception e) { 
						  e.printStackTrace(); 
					  } 
				}
			   //把得到的图片绑定在控件上显示 
			   iv.setImageBitmap(myBitmap); 
				break;
			default:
				break;
			}
		}
    } 
	
	public static Bitmap getPicFromBytes(byte[] bytes, BitmapFactory.Options opts) { 
	    if (bytes != null) 
	        if (opts != null) 
	            return BitmapFactory.decodeByteArray(bytes, 0, bytes.length,opts); 
	        else 
	            return BitmapFactory.decodeByteArray(bytes, 0, bytes.length); 
	    return null; 
	} 

	public static byte[] readStream(InputStream inStream) throws Exception { 
	    byte[] buffer = new byte[1024]; 
	    int len = -1; 
	    ByteArrayOutputStream outStream = new ByteArrayOutputStream(); 
	    while ((len = inStream.read(buffer)) != -1) { 
	             outStream.write(buffer, 0, len); 
	    } 
	    byte[] data = outStream.toByteArray(); 
	    outStream.close(); 
	    inStream.close(); 
	    return data; 

	} 
 

    这两个自定义页中中你也可以直接处理图片,在回传的Intent中保存流给接收页面处理,这样做的好处是接受页不必知道是哪个页面回传的数据,少了层逻辑判断,直接将流转为BitMap显示即可,但缺点也很明显:

public Intent putExtra (String name, byte[] value)这个方法传递的字节数不能>40KB,如果超出会报 错。

在数返回显示的逻辑中使用了getPicFromBytes和readStream两个自定义方法,这里只是提供了一种方案,你仍然可以使用方法一中的处理方法。

 

效果图:

选项前的图标是AndroidManifest.xml中相关Activity的icon属性,默认会调用Activity所在项目的icon,目前还没有解决如何不显示选项之前的图标,希望知道的朋友,不吝赐教,留言告知。

分享到:
评论

相关推荐

    Android从相册或拍照选择照片插入到EditText

    总的来说,实现“Android从相册或拍照选择照片插入到EditText”这一功能,需要处理用户权限、图片选择、图片处理和数据显示等多个步骤。以上代码示例可以帮助你理解整个流程,但具体实现可能需要根据项目的实际需求...

    Android实现个人登陆,之后实现照或选择相册上传

    9. **错误处理和反馈**:在登录、拍照、选择相册和上传过程中,可能出现各种错误,如网络连接问题、权限未授予、文件读取失败等。良好的错误处理和用户反馈机制能够提升用户体验。 10. **UI响应式**:Android应用应...

    聊天,相册,拍照

    "HRChatDemo"可能是这个项目的名称,它很可能包含了一个完整的聊天应用示例,包括聊天界面、相册选择图片和拍照功能的实现。通过查看和分析这个项目的源代码,我们可以学习到如何在Android Studio中整合这些功能,...

    微信聊天界面,相册,相机拍照

    4. **图片上传与显示**:与相册选择图片的步骤相似,将拍摄的照片上传到服务器,获取URL后显示在聊天界面。 在Android Studio中,你需要创建相应的Activity或Fragment来实现这些功能,编写适配器Adapter来处理...

    微信文字语音聊天相册拍照

    在Android应用开发中,微信文字语音聊天相册拍照功能的实现涉及到多个技术点,这些功能在许多社交应用中都是核心部分。以下将详细介绍如何利用Android Studio来开发这样的应用。 首先,我们要关注的是**文字聊天**...

    Android高仿qq聊天界面,发送表情,添加本地图库照片,可拍照

    为了从本地图库选择照片,需要使用`Intent.ACTION_PICK`启动系统相册。获取用户选择的图片后,可以使用`ContentResolver`读取其内容。图片可能需要进行缩放、裁剪等处理,以适应聊天界面的展示需求。可以使用`...

    android 图片转文字

    8. **用户交互设计**:为了提供良好的用户体验,可能包含图像的选择方式(如从相册选取或直接拍照),以及清晰的反馈提示。 9. **性能优化**:由于OCR可能会消耗大量计算资源,开发者可能需要考虑异步处理,避免UI...

    Android多媒体日记本Demo

    - **Intent选择器**:允许用户从相册、录音库中选择多媒体文件,通过Intent.createChooser()创建选择器。 8. **UI动画**: - **动画效果**:为提高用户体验,可添加过渡动画,如页面切换、按钮点击反馈等,...

    富文本编辑器(可以同时添加文字和图片)

    1. **图片选择器**:为了从本地存储中选择图片,通常会使用Intent来启动系统图片选择器,用户可以从相册中挑选图片。 2. **拍照功能**:通过调用相机API,用户可以直接拍摄新照片并插入到编辑器中。 3. **图片上传**...

    android小例子

    照相机应用的例子可能涉及启动相机预览、捕获照片、调整设置(如闪光灯、焦距等)以及保存图片到相册。对于新版本的Android,CameraX库提供了更现代、更易于使用的接口,简化了相机功能的实现。 4. **GPS地图**:...

    [Android实例] 继人员列表,聊天的实现,包括图片,语音.zip

    图片发送功能可以利用Android的Camera API或者选择器Intent让用户拍照或从相册选取图片,然后使用Bitmap进行压缩,减少内存占用。上传图片到服务器通常需要用到HTTP请求库,如OkHttp或Retrofit,将图片转换为Base64...

    仿微信发表图文功能

    - 图片选择接口:通常我们会集成系统级别的图片库或者相机接口,让用户能够从手机相册中选取图片或直接拍照。Android系统可以使用`Intent.ACTION_PICK`或`Intent.ACTION_IMAGE_CAPTURE`,iOS则可以使用`...

    android 记事本

    3. 图片处理:Android提供了多种方式处理图片,如Bitmap、Glide库或者Picasso库,用于从相册选择图片或拍照并将其保存到应用内部存储。 4. 闹钟功能:Android的AlarmManager服务可以用来设置定时提醒。`...

    android从入门到精通sl(实例源程序)

    ### Android从入门到精通SL(实例源程序) #### 一、Android基础知识 1. **Android系统简介** - Android是一个基于Linux内核的开源移动操作系统,由Google公司维护。 - 它支持多种设备,包括智能手机、平板电脑...

    android安卓Diary

    3. **图片和文件选择器**: 使用Intent调用系统文件选择器,让用户能够从相册或其他位置选择图片和文件。 七、加密与安全 1. **数据加密**: 对敏感的日记内容进行加密,保障用户隐私。 2. **安全传输**: 如果数据...

    安卓设计报告文档.docx

    - **头像更换**:集成相机和相册权限,让用户可以拍照或选择已有的图片作为头像。 7. **实验结果与分析**: - 分析应用的性能、用户体验和功能实现的完整性,评估是否达到预期目标。 8. **小结与心得体会**: -...

    Android 高仿微信朋友圈拍照上传功能

    这个功能涉及到用户界面设计、权限管理、图片选择和预览以及数据处理等多个方面。下面我们将详细探讨如何实现这样一个功能。 首先,对于图片选择部分,我们可以利用第三方库PhotoPicker。PhotoPicker是一个支持多图...

    android实现定时拍照功能

    // 处理拍照后的图片,如保存到相册 saveImage(data); camera.startPreview(); // 重新开始预览 } }; ``` 5. **图片处理**: 拍照后,你需要将字节数组转换为Bitmap,然后可以保存到设备的存储空间: ```...

    android安卓demo

    5. **多媒体**:学习如何处理图像、音频和视频,包括从相册选取、相机拍照、播放音频和视频等。 6. **通知与服务**:理解如何发送本地通知,以及后台运行的服务(Services)如何执行长时间任务。 7. **权限管理**...

Global site tag (gtag.js) - Google Analytics