更换头像后,只要不换虚拟机,头像一直都是最后一次更换的图片。重新运行或者卸载了重新安装,头像一直不变。
这里只有更换头像,所以简陋的弄一下,方便观察
先上效果图:运行环境eclipse Android4.4.2
在这里插入图片描述
卸载重新运行效果:
在这里插入图片描述

先在布局文件activity_main.xml中添加一个ImageView控件用来显示头像图片:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical"
    tools:context="${relativePackage}.${activityClass}" >

    <LinearLayout 
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"
        android:layout_marginTop="80dp"
        android:orientation="vertical">
        <ImageView 
            android:id="@+id/iv_head"
            android:layout_width="100dp"
            android:layout_height="100dp"
        	android:contentDescription="@null"
            android:layout_gravity="center_horizontal"
            android:src="@drawable/default_icon"/>
        </LinearLayout>
</LinearLayout>

注:android:src="@drawable/default_icon"这个是ImageView控件的默认背景图,如果不想要呢可以删除或者换成背景颜色,一点都不影响

然后再新建一个弹窗的布局文件activity_dialog_select.xml,效果如下:
在这里插入图片描述
具体代码:

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

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:background="@drawable/dialog_bg_shape"
        android:orientation="vertical">

        <TextView
            android:id="@+id/dialog_title"
            android:layout_width="match_parent"
            android:layout_height="37dp"
            android:gravity="center"
            android:text="请选择更换方式"
            android:textColor="#777777"
            android:textSize="14sp" />

        <View
            android:layout_width="match_parent"
            android:layout_height="0.2dp"
            android:background="#e5e5e5" />

        <TextView
            android:id="@+id/photograph"
            android:layout_width="match_parent"
            android:layout_height="41dp"
            android:gravity="center"
            android:text="拍照"
            android:textColor="#444444"
            android:textSize="18sp" />

        <View
            android:layout_width="match_parent"
            android:layout_height="0.2dp"
            android:background="#e5e5e5" />

        <TextView
            android:id="@+id/photo"
            android:layout_width="match_parent"
            android:layout_height="41dp"
            android:gravity="center"
            android:text="从相册中选择"
            android:textColor="#444444"
            android:textSize="18sp" />
    </LinearLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="10dp"
        android:background="@drawable/dialog_bg_shape">
        <TextView
            android:id="@+id/cancel"
            android:layout_width="match_parent"
            android:layout_height="47dp"
            android:gravity="center"
            android:text="取消"
            android:textColor="#444444"
            android:textSize="18sp" />
    </LinearLayout>
</LinearLayout>

在这里,我们需要引入样式,使得这个弹窗的样子比较好看:dialog_bg_shape.xml一般放在drawable文件夹里面

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
    android:shape="rectangle"
    >
    <corners android:radius="15dp"/>
    <solid android:color="#FFFFFF"/>
</shape>

之后就可以编写逻辑代码了:
先生成弹窗:这里使用的是dialog

	builder = new AlertDialog.Builder(this);//创建对话框
    inflater = getLayoutInflater();
    layout = inflater.inflate(R.layout.activity_dialog_select, null);//获取自定义布局
    builder.setView(layout);//设置对话框的布局
    dialog = builder.create();//生成最终的对话框
    dialog.setCanceledOnTouchOutside(true);//设置点击Dialog外部任意区域关闭
    dialog.show();//显示对话框

我用eclipse Android4.0运行发现弹窗后面有一个黑色的背景,怎么也去不掉,就像下面这样:
在这里插入图片描述
但是用Android studio的Android8.0运行又没有(项目6.0)
在这里插入图片描述
拍照用的是:

try {
                    Intent intent2 = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);//开启相机应用程序获取并返回图片(capture:俘获)
                    intent2.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(Environment.getExternalStorageDirectory(),
                            "head.jpg")));//指明存储图片或视频的地址URI
                    startActivityForResult(intent2, 2);//采用ForResult打开
                } catch (Exception e) {
                    Toast.makeText(MainActivity.this, "相机无法启动,请先开启相机权限", Toast.LENGTH_LONG).show();
                }

没有写权限哦,所以Android6.0及以上请自行添加权限

图片裁剪:

		Intent intent = new Intent("com.android.camera.action.CROP");
        //找到指定URI对应的资源图片
        intent.setDataAndType(uri, "image/*");
        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("return-data", true);
        //进入系统裁剪图片的界面
        startActivityForResult(intent, 3);

同样是高版本的不能用,我发现这就是一鸡肋。但为了应付老师,这几天也就弄出来这个,在网上疯狂的搜代码,靠,全部只适合高版本的,但那个老师叫我们弄的项目是4.0的,太气人了。

话不多说,别忘了在AndroidManifest.xml清单文件里面添加权限:

	<uses-permission android:name="android.permission.CAMERA" />
    <uses-feature android:name="android.hardware.camera"/>
    <uses-feature android:name="android.hardware.camera.autofocus"/>
 
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />

下面是完整的代码:这是两个布局文件共用一个逻辑代码文件

package china.ynyx.heyunhui;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

import android.app.Activity;
import android.app.AlertDialog;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.Drawable;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends Activity implements OnClickListener{

    private ImageView iv_icon;
    
    private AlertDialog.Builder builder;
    private AlertDialog dialog;
    private LayoutInflater inflater;
    private View layout;
    private TextView takePhoto;
    private TextView choosePhoto;
    private TextView cancel;
    private static String path = "/sdcard/DemoHead/";//sd路径
    private Bitmap head;//头像Bitmap

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        intView();
    }
    
    
  private void intView() {
		// TODO Auto-generated method stub
	  iv_icon = (ImageView) findViewById(R.id.iv_head);

	  Bitmap bt = BitmapFactory.decodeFile(path + "head.jpg");//从Sd中找头像,转换成Bitmap
	  if(bt!=null){
		@SuppressWarnings("deprecation")
		Drawable drawable = new BitmapDrawable(bt);//转换成drawable
		iv_icon.setImageDrawable(drawable);
	  }else{
		/**
		*	如果SD里面没有则需要从服务器取头像,取回来的头像再保存在SD中
		* 
		*/
	  }
      iv_icon.setOnClickListener(new OnClickListener() {

          @Override
          public void onClick(View v) {
          	viewInit();
          }
      });
	}

  	//初始化控件方法
    public void viewInit() {
    builder = new AlertDialog.Builder(this);//创建对话框
    inflater = getLayoutInflater();
    layout = inflater.inflate(R.layout.activity_dialog_select, null);//获取自定义布局
    builder.setView(layout);//设置对话框的布局
    dialog = builder.create();//生成最终的对话框
    
    dialog.setCanceledOnTouchOutside(true);//设置点击Dialog外部任意区域关闭
    dialog.show();//显示对话框
    
    takePhoto = (TextView) layout.findViewById(R.id.photograph);
    choosePhoto = (TextView) layout.findViewById(R.id.photo);
    cancel = (TextView) layout.findViewById(R.id.cancel);
    //设置监听
    takePhoto.setOnClickListener(this);
    choosePhoto.setOnClickListener(this);
    cancel.setOnClickListener(this);
}
    @Override
    public void onClick(View view) {
        switch (view.getId()){
            case R.id.photograph://调用相机拍照
            	//最好用try/catch包裹一下,防止因为用户未给应用程序开启相机权限,而使程序崩溃
                try {
                    Intent intent2 = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);//开启相机应用程序获取并返回图片(capture:俘获)
                    intent2.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(Environment.getExternalStorageDirectory(),
                            "head.jpg")));//指明存储图片或视频的地址URI
                    startActivityForResult(intent2, 2);//采用ForResult打开
                } catch (Exception e) {
                    Toast.makeText(MainActivity.this, "相机无法启动,请先开启相机权限", Toast.LENGTH_LONG).show();
                }
                dialog.dismiss();
                break;
            case R.id.photo://从相册里面取照片
                Intent intent1 = new Intent(Intent.ACTION_PICK, null);//返回被选中项的URI
                intent1.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");//得到所有图片的URI
                startActivityForResult(intent1, 1);
                dialog.dismiss();
                break;
            case R.id.cancel:
                dialog.dismiss();//关闭对话框
                break;
            default:break;
        }
    }
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        switch (requestCode) {
            //从相册里面取相片的返回结果
            case 1:
                if (resultCode == RESULT_OK) {
                    cropPhoto(data.getData());//裁剪图片
                }
 
                break;
            //相机拍照后的返回结果
            case 2:
                if (resultCode == RESULT_OK) {
                    File temp = new File(Environment.getExternalStorageDirectory()
                            + "/head.jpg");
                    cropPhoto(Uri.fromFile(temp));//裁剪图片
                }
 
                break;
            //调用系统裁剪图片后
            case 3:
                if (data != null) {
                    Bundle extras = data.getExtras();
                    head = extras.getParcelable("data");
                    if (head != null) {
                        /**
                         	* 上传服务器代码
                         */
                        setPicToView(head);//保存在SD卡中
                        iv_icon.setImageBitmap(head);//用ImageView显示出来
                    }
                }
                break;
            default:
                break;
 
        }
        super.onActivityResult(requestCode, resultCode, data);
    }
 
    ;
 
    /**
                * 调用系统的裁剪
     *
     * @param uri
     */
    public void cropPhoto(Uri uri) {
        Intent intent = new Intent("com.android.camera.action.CROP");
        //找到指定URI对应的资源图片
        intent.setDataAndType(uri, "image/*");
        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("return-data", true);
        //进入系统裁剪图片的界面
        startActivityForResult(intent, 3);
        
    }
 
    private void setPicToView(Bitmap mBitmap) {
        String sdStatus = Environment.getExternalStorageState();
        if (!sdStatus.equals(Environment.MEDIA_MOUNTED)) { // 检测sd卡是否可用
            return;
        }
        FileOutputStream b = null;
        File file = new File(path);
        file.mkdirs();// 创建以此File对象为名(path)的文件夹
        String fileName = path + "head.jpg";//图片名字
        try {
            b = new FileOutputStream(fileName);
            mBitmap.compress(Bitmap.CompressFormat.JPEG, 100, b);// 把数据写入文件(compress:压缩)
 
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } finally {
            try {
                //关闭流
                b.flush();
                b.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
 
        }
    }
}

AndroidManifest.xml文件:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.yuanjiaotouxiang"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="21" />
    
    <uses-permission android:name="android.permission.CAMERA" />
    <uses-feature android:name="android.hardware.camera"/>
    <uses-feature android:name="android.hardware.camera.autofocus"/>
 
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
    	android:theme="@android:style/Theme.Light.NoTitleBar">
        <activity
            android:name=".MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

demo地址:
在这里插入图片描述   在这里插入图片描述

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐