android选择头像弹窗,Android App开发常用功能之用户头像选择-Go语言中文社区

前言

现在的APP基本都有个人资料的填写,基本的都有头像的选择,支持拍照和从本地相册选择,剪切圆形头像的功能,现在用个小demo实现以下。

下面看一下效果图

e90dd745a8b4bc540b6f59590e9638ca.gif

上代码:

主界面代码

package com.example.androidpersonal_icon;

import android.os.Bundle;

import android.os.Environment;

import android.provider.MediaStore;

import android.util.Log;

import android.view.Gravity;

import android.view.View;

import android.view.View.OnClickListener;

import android.widget.ImageView;

import java.io.File;

import android.app.Activity;

import android.content.Intent;

import android.graphics.Bitmap;

import android.graphics.BitmapFactory;

import android.net.Uri;

public class MainActivity extends Activity {

protected static final int CHOOSE_PICTURE = 0;

protected static final int TAKE_PICTURE = 1;

private static final int CROP_SMALL_PICTURE = 2;

protected static Uri tempUri;

private ImageView iv_personal_icon;

private SelectPicPopupWindow menuWindow;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

// 新建一个用来存储照片的文件夹

File destDir = new File(Environment.getExternalStorageDirectory() + "/AndroidPersonal_icon");

if (!destDir.exists()) {

destDir.mkdirs();

}

iv_personal_icon = (ImageView) findViewById(R.id.iv_personal_icon);

iv_personal_icon.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

// 实例化SelectPicPopupWindow

menuWindow = new SelectPicPopupWindow(MainActivity.this, itemsOnClick);

// 显示窗口

menuWindow.showAtLocation(MainActivity.this.findViewById(R.id.main),

Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, 0); // 设置layout在PopupWindow中显示的位置

}

});

// 读取上一次剪切的照片

if (destDir.exists() && destDir.isDirectory()) {

if (destDir.list().length > 0) {

Log.d("111111111112", destDir.toString() + "/image_icon.png");

Bitmap bitmap = BitmapFactory.decodeFile(destDir.toString() + "/image_icon.png");

iv_personal_icon.setImageBitmap(bitmap);

} else {

iv_personal_icon.setBackgroundResource(R.drawable.default_personal_image);

}

}

}

// 为弹出窗口实现监听类

private OnClickListener itemsOnClick = new OnClickListener() {

public void onClick(View v) {

menuWindow.dismiss();

switch (v.getId()) {

case R.id.Layout_take_photo:

Intent openCameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

tempUri = Uri.fromFile(

new File(Environment.getExternalStorageDirectory() + "/AndroidPersonal_icon", "image.jpg"));

Log.d("11111111", tempUri.toString());

// 指定照片保存路径(SD卡),image.jpg为一个临时文件,每次拍照后这个图片都会被替换

openCameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, tempUri);

startActivityForResult(openCameraIntent, TAKE_PICTURE);

break;

case R.id.Layout_pick_photo:

Intent openAlbumIntent = new Intent(Intent.ACTION_GET_CONTENT);

openAlbumIntent.setType("image/*");

startActivityForResult(openAlbumIntent, CHOOSE_PICTURE);

break;

default:

break;

}

}

};

@Override

protected void onActivityResult(int requestCode, int resultCode, Intent data) {

super.onActivityResult(requestCode, resultCode, data);

if (resultCode == RESULT_OK) { // 如果返回码是可以用的

switch (requestCode) {

case TAKE_PICTURE:

startPhotoZoom(tempUri); // 开始对图片进行裁剪处理

break;

case CHOOSE_PICTURE:

startPhotoZoom(data.getData()); // 开始对图片进行裁剪处理

break;

case CROP_SMALL_PICTURE:

if (data != null) {

setImageToView(data); // 让刚才选择裁剪得到的图片显示在界面上

}

break;

}

}

}

/**

* 裁剪图片方法实现

*

* @param uri

*/

protected void startPhotoZoom(Uri uri) {

if (uri == null) {

Log.i("tag", "The uri is not exist.");

}

tempUri = 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", 150);

intent.putExtra("outputY", 150);

intent.putExtra("return-data", true);

startActivityForResult(intent, CROP_SMALL_PICTURE);

}

/**

* 保存裁剪之后的图片数据

*

* @param

*

* @param picdata

*/

protected void setImageToView(Intent data) {

Bundle extras = data.getExtras();

if (extras != null) {

Bitmap photo = extras.getParcelable("data");

photo = Utils.toRoundBitmap(photo, tempUri); // 这个时候的图片已经被处理成圆形的了

iv_personal_icon.setImageBitmap(photo);

uploadPic(photo);

}

}

private void uploadPic(Bitmap bitmap) {

// 上传至服务器

// ... 可以在这里把Bitmap转换成file,然后得到file的url,做文件上传操作

// 注意这里得到的图片已经是圆形图片了

// bitmap是没有做个圆形处理的,但已经被裁剪了

String imagePath = Utils.savePhoto(bitmap,

Environment.getExternalStorageDirectory().getAbsolutePath() + "/AndroidPersonal_icon", "image_icon");

Log.d("imagePath", imagePath + "");

if (imagePath != null) {

// 拿着imagePath上传了

// ...

}

}

}

圆形头像剪切代码:

package com.example.androidpersonal_icon;

import java.io.File;

import java.io.FileNotFoundException;

import java.io.FileOutputStream;

import java.io.IOException;

import android.graphics.Bitmap;

import android.graphics.Bitmap.Config;

import android.graphics.Canvas;

import android.graphics.Paint;

import android.graphics.PorterDuff.Mode;

import android.graphics.PorterDuffXfermode;

import android.graphics.Rect;

import android.graphics.RectF;

import android.net.Uri;

import android.util.Log;

public class Utils {

/**

* Save image to the SD card

*

* @param photoBitmap

* @param photoName

* @param path

*/

public static String savePhoto(Bitmap photoBitmap, String path,

String photoName) {

String localPath = null;

if (android.os.Environment.getExternalStorageState().equals(

android.os.Environment.MEDIA_MOUNTED)) {

File dir = new File(path);

if (!dir.exists()) {

dir.mkdirs();

}

File photoFile = new File(path, photoName + ".png");

FileOutputStream fileOutputStream = null;

try {

fileOutputStream = new FileOutputStream(photoFile);

if (photoBitmap != null) {

if (photoBitmap.compress(Bitmap.CompressFormat.PNG, 100,

fileOutputStream)) { // 转换完成

localPath = photoFile.getPath();

fileOutputStream.flush();

}

}

} catch (FileNotFoundException e) {

photoFile.delete();

localPath = null;

e.printStackTrace();

} catch (IOException e) {

photoFile.delete();

localPath = null;

e.printStackTrace();

} finally {

try {

if (fileOutputStream != null) {

fileOutputStream.close();

fileOutputStream = null;

}

} catch (IOException e) {

e.printStackTrace();

}

}

}

return localPath;

}

/**

* 转换图片成圆形

*

* @param bitmap

* 传入Bitmap对象

* @param tempUri

* @return

*/

public static Bitmap toRoundBitmap(Bitmap bitmap, Uri tempUri) {

int width = bitmap.getWidth();

int height = bitmap.getHeight();

float roundPx;

float left, top, right, bottom, dst_left, dst_top, dst_right, dst_bottom;

if (width <= height) {

roundPx = width / 2;

left = 0;

top = 0;

right = width;

bottom = width;

height = width;

dst_left = 0;

dst_top = 0;

dst_right = width;

dst_bottom = width;

} else {

roundPx = height / 2;

float clip = (width - height) / 2;

left = clip;

right = width - clip;

top = 0;

bottom = height;

width = height;

dst_left = 0;

dst_top = 0;

dst_right = height;

dst_bottom = height;

}

Bitmap output = Bitmap.createBitmap(width, height, Config.ARGB_8888);

Canvas canvas = new Canvas(output);

final int color = 0xff424242;

final Paint paint = new Paint();

final Rect src = new Rect((int) left, (int) top, (int) right,

(int) bottom);

final Rect dst = new Rect((int) dst_left, (int) dst_top,

(int) dst_right, (int) dst_bottom);

final RectF rectF = new RectF(dst);

paint.setAntiAlias(true);// 设置画笔无锯齿

canvas.drawARGB(0, 0, 0, 0); // 填充整个Canvas

paint.setColor(color);

// 以下有两种方法画圆,drawRounRect和drawCircle

// canvas.drawRoundRect(rectF, roundPx, roundPx, paint);//

// 画圆角矩形,第一个参数为图形显示区域,第二个参数和第三个参数分别是水平圆角半径和垂直圆角半径。

canvas.drawCircle(roundPx, roundPx, roundPx, paint);

paint.setXfermode(new PorterDuffXfermode(Mode.SRC_IN));// 设置两张图片相交时的模式,参考http://trylovecatch.iteye.com/blog/1189452

canvas.drawBitmap(bitmap, src, dst, paint); // 以Mode.SRC_IN模式合并bitmap和已经draw了的Circle

return output;

}

}

ok,大功告成,最后别忘了在清单文件中添加读写sd可权限,不然得不到imagePath

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/405602.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CentOS7 (64位) 下QT5.5 连接MySQL数据库(driver not loaded)

用qt连接MySQL需要共享库 libqsqlmysql.so的驱动&#xff0c;路径在plugin/sqldrivers目录下&#xff0c;乍看已经可用了&#xff0c;其实不然。 用ldd命令分析一下&#xff0c;libmysqlclient_r.so.16 > not found ,后明显libmysqlclient_r.so.16 缺少另一个共享库&#xf…

linux dhcp 服务(转)

大多数的情况下Linux作为DHCP服务器而windows 95/98作为DHCP客户。Linux也可以作为DHCP客户&#xff0c;即你要安装dhcpcd rpm 软件包 &#xff0c;Linux作为DHCP服务器&#xff0c;只需要安装dhcpd rpm 包。一.DHCP服务器工作的前提条件为了使DHCP服务器为windows机器服务&…

java jdk1.8.0_221 安装步骤

一、下载jdk Oracle JDK下载 官网 https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 下载jdk1.8.0_221. 需要注册账号登陆才能下载。 下载完成&#xff0c;双击jdk-8u221-windows-x64.exe&#xff0c;进行安装。 二、安装jdk 安装前…

gsoap使用心得!

From: http://www.cppblog.com/qiujian5628/archive/2008/06/19/54019.html 完整源码下载 最近换了个工作环境&#xff0c;现在在大望路这边上班&#xff0c;呵&#xff0c;刚上班接到的任务就是熟悉gsoap&#xff01;废话少说&#xff0c;现在开始gSoap学习&#xff01;gSOAP…

android tabhost 多个activity,Android:TabHost中Activity的生命周期问题

用过TabHost制作多个activity的分页效果的朋友应该知道&#xff0c;tabhost中镶嵌的activity的onCreate和onDestroy是和tabhost关联的&#xff0c;在创建了tabhost之后&#xff0c;第一次访问某个activity会执行这个activity的oncreate事件&#xff0c;以后切回来时只会执行onR…

网站决策分析软件WebBI

为了提高网站的访问量、黏合度&#xff0c;网站的管理者投入大量的精力于市场推广、内容发布、网站业务改进等&#xff0c;而对于如何评估推广效果、如何了解网民的行为以提高网站的吸引力和服务质量很多网站都显的无力可施。经过互联网多年的发展&#xff0c;互联网网站之间的…

eclipse启动报错No java virtual machine was found after seearching the locations:XXXXX

一、问题描述 卸载之前的老版本JDK后&#xff0c;重新安装了JDK&#xff0c;再次打开eclipse出现以下问题&#xff1a;在指定的目录中找不到JDK虚拟机。 二、解决办法 eclipse启动时会根据eclipse.ini配置文件来找javaw.exe 启动ecplise GUI。 在eclipse安装的根目录中找到e…

网卡配置

网卡配置文件 &#xff08;1&#xff09;/etc/sysconfig/network-scripts/ifcfg-interface-name 配置文件ifcfg-interface-name包含了初始化接口所需的大部分详细信息。其中interface-name将根据网卡的类型和排序而不同&#xff0c;一般其名字为eth0、eth1、ppp0等&#xff0c…

linux ace platform_linux.gnu,Linux下配置和使用ACE笔记

1. 下载 ACE 5.7.从 http://download.dre.vanderbilt.edu/ 下载 ACETAOCIAO-5.7.tar.gz包。2. 解压将压缩包copy到linux目录/data/ACE里&#xff0c;然后解压, tar -zxvf ACETAOCIAO-5.7.tar.gz。解压后的目录是 /data/ACE_wrappers.3. 配置config.h和platform_macros.GNU。…

Linux 查看磁盘分区、文件系统、使用情况的命令和相关工具介绍

From: http://fedora.linuxsir.org/main/?qnode/41 作者&#xff1a;北南南北 来自&#xff1a;LinuxSir.Org 提要&#xff1a;Linux 磁盘分区表、文件系统的查看、统计的工具很多&#xff0c;有些工具是多功能的&#xff0c;不仅仅是查看磁盘的分区表&#xff0c;而且也能进行…

Android 机顶盒手势、数据分页演示DEMO

上篇讲了一篇机顶盒的一个封装类《GridView》&#xff0c;利用这个封装的GridView 可以实现一些例如移动、位移图标等基础组件“GridView”无法实现的一些功能。本篇将会提出一部分可供运行的代码演示一个分页、手势换屏的DEMO版本&#xff0c;供大家学习。 DEMO 的工程目录如下…

Java jdk中的部分工具javac javaw javaws javadoc javah javap jar jdb jps的作用

一、JDK 工具和实用程序 JDK提供的标准工具和程序 基本工具&#xff1a;javac, java, javadoc, apt, appletviewer, jar, jdb, javah, javap, extcheck安全工具&#xff1a;keytool, jarsigner, policytool, kinit, klist, ktab国际化工具&#xff1a;native2ascii远程方法调…

firefox android 去更新,Android版Firefox Beta发布更新

就在谷歌I/O大会前夕&#xff0c;Android版Firefox Beta悄悄的发布了一次更新。更新内容包括&#xff1a;Flash支持(虽然事实上在上个月的更新中就已经支持了)以及日常bug修复及稳定性提升等&#xff0c;不过依然没有加入复制和粘贴功能。此外Firefox Beta的UI界面也被重新设计…

Linux/UNIX的scp命令用法详解

From: http://www.ha97.com/4169.html PS&#xff1a;偶很喜欢用scp这个命令来传输数据。scp命令跟cp命令类似&#xff0c;只不过cp命令是在同一台机器上用的&#xff0c;scp是在两台机器上复制传输数据的命令。scp实质相当于利用SSH协议来传输数据的cp命令。 用法举例&#xf…

React开发(230):ant design table固定表头

固定表头 方便一页内展示大量数据。需要指定 column 的 width 属性&#xff0c;否则列头和内容可能不对齐。 如果指定 width 不生效或出现白色垂直空隙&#xff0c; 请尝试建议留一列不设宽度以适应弹性布局&#xff0c;或者检查是否有超长连续字段破坏布局。

安装ARM交叉编译工具

1. 下载交叉编译工具链。 可到如下两个网站下载&#xff1a; http://ftp.arm.linux.org.uk/pub/armlinux/toolchain/ http://www.uushare.com/user/zyh006814/file/1434053 2. 安装 # mkdir /usr/local/arm # cd /usr/local/arm # 解压下载的工具链 # 添加环境变量&#xff0c…

Linux下获得线程ID syscall(224)

From: http://blog.163.com/onx0320126/blog/static/164736436201252574720314/ 对于Linux线程&#xff0c;也许大家并不陌生&#xff0c;而我们通常所说的线程是指基于POSIX标准的线程&#xff0c;而Linux下除了兼容POSIX标准&#xff0c;而且提供线程在内核级的支持。早在Lin…

Qt中常用的QChar QByteArry QString数据类型转换方法

一、QChar类型 转 QByteArry类型 char mybuffer[mybufferSize]; QByteArray ByteData QByteArray(mybuffer, mybufferSize);二、QByteArry类型 转 QString类型 QByteArray _byte; QString string&#xff0c;_string1; string QString(_byte); _string1 QString(_byte.toH…