热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android中通过访问本地相册或者相机设置用户头像实例

目前几乎所有的APP在用户注册时都会有设置头像的需求,大致分为三种情况: (1)通过获取本地相册的图片,经过裁剪后作为头像。 (2)通过

目前几乎所有的APP在用户注册时都会有设置头像的需求,大致分为三种情况:

(1)通过获取本地相册的图片,经过裁剪后作为头像。

(2)通过启动手机相机,现拍图片然后裁剪作为头像。

(3)在APP中添加一些自带的头像资源,供用户选择(不够人性化,目前很少使用)。

这次我们简单介绍下通过获取本地相册以及相机拍摄的方法设置头像,实现思路如下:

(1)通过startActivityForResult方法,分别传递调用系统相册的Intent和调用相机拍照的Intent来做选择

(2)调用Android系统中自带的图片剪裁,实现图片的剪裁并在onActivityResult方法中获取数据。

本次演示效果如下(分别为从本地相册获取以及从相机拍摄获取头像):

     

简单布局文件这里不再做赘述,本次试验使用隐式intent调用相机以及本地相册,未在配置清单上添加权限,依然可以调用。

java实现代码如下:

import android.content.Intent;
import android.graphics.Bitmap;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.Toast;
import java.io.File;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {
  private Button buttonLocal, buttonCamera;
  private ImageView imageView;
  //相机拍摄的头像文件(本次演示存放在SD卡根目录下)
  private static final File USER_ICON = new File(Environment.getExternalStorageDirectory(), "user_icon.jpg");
  //请求识别码(分别为本地相册、相机、图片裁剪)
  private static final int CODE_PHOTO_REQUEST = 1;
  private static final int CODE_CAMERA_REQUEST = 2;
  private static final int CODE_PHOTO_CLIP = 3;
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    buttOnLocal= (Button) findViewById(R.id.buttonLocal);
    buttOnCamera= (Button) findViewById(R.id.buttonCamera);
    imageView = (ImageView) findViewById(R.id.imageView);
    buttonLocal.setOnClickListener(this);
    buttonCamera.setOnClickListener(this);
  }
  //设置点击事件
  @Override
  public void onClick(View view) {
    switch (view.getId()) {
      case R.id.buttonLocal:
        //调用获取本地图片的方法
        getPicFromLocal();
        break;
      case R.id.buttonCamera:
        //调用相机拍照的方法
        getPicFromCamera();
        break;
      default:
        break;
    }
  }
  /**
   * 从本机相册获取图片
   */
  private void getPicFromLocal() {
    Intent intent = new Intent();
    // 获取本地相册方法一
    intent.setAction(Intent.ACTION_GET_CONTENT);
    intent.setType("image/*");
    //获取本地相册方法二
//    intent.setAction(Intent.ACTION_PICK);
//    intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
//        "image/*");
    startActivityForResult(intent, CODE_PHOTO_REQUEST);
  }
  /**
   * 通过相机拍摄获取图片,
   * 并存入设置的路径中
   */
  private void getPicFromCamera() {
    Intent intent = new Intent();
    intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
    // 下面这句指定调用相机拍照后的照片存储的路径
    intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(USER_ICON));
    startActivityForResult(intent, CODE_CAMERA_REQUEST);
  }
  /**
   * 图片裁剪
   *
   * @param uri
   */
  private void photoClip(Uri uri) {
    // 调用系统中自带的图片剪裁
    Intent intent = new Intent();
    intent.setAction("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 是裁剪图片宽高
     *这里仅仅是头像展示,不建议将值设置过高
     * 否则超过binder机制的缓存大小的1M限制
     * 报TransactionTooLargeException
     */
    intent.putExtra("outputX", 150);
    intent.putExtra("outputY", 150);
    intent.putExtra("return-data", true);
    startActivityForResult(intent, CODE_PHOTO_CLIP);
  }
  /**
   * 提取保存裁剪之后的图片数据,并设置头像部分的View
   */
  private void setImageToHeadView(Intent intent) {
    Bundle extras = intent.getExtras();
    if (extras != null) {
      Bitmap photo = extras.getParcelable("data");
      imageView.setImageBitmap(photo);
    }
  }
  @Override
  protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    // 用户没有进行有效的设置操作,返回
    if (resultCode == RESULT_CANCELED) {
      Toast.makeText(MainActivity.this, "取消", Toast.LENGTH_LONG).show();
      return;
    }
    switch (requestCode) {
      case CODE_CAMERA_REQUEST:
        if (USER_ICON.exists()) {
          photoClip(Uri.fromFile(USER_ICON));
        }
        break;
      case CODE_PHOTO_REQUEST:
        if (data != null) {
          photoClip(data.getData());
        }
        break;
      case CODE_PHOTO_CLIP:
        if (data != null) {
          setImageToHeadView(data);
        }
        break;
    }
    super.onActivityResult(requestCode, resultCode, data);
  }
}

这里要注意的是在裁剪图片时,长和宽不要设置太大,否则超过binder机制的缓存大小的限制(受手机配置影响).报TransactionTooLargeException,在代码中已经做了详细标注,请各位看官在实现的时候万万注意。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 本文将详细介绍多个流行的 Android 视频处理开源框架,包括 ijkplayer、FFmpeg、Vitamio、ExoPlayer 等。每个框架都有其独特的优势和应用场景,帮助开发者更高效地进行视频处理和播放。 ... [详细]
  • 气象对比分析
    本文探讨了不同地区和时间段的天气模式,通过详细的图表和数据分析,揭示了气候变化的趋势及其对环境和社会的影响。 ... [详细]
  • Android 6.0 切换指定 Wi-Fi 的解决方案
    本文详细介绍了在 Android 6.0 系统中切换到指定 Wi-Fi 的方法,包括常见的问题、原因分析及解决方案。通过官方文档和代码示例,帮助开发者更好地理解和实现这一功能。 ... [详细]
  • vivo Y5s配备了联发科Helio P65八核处理器,这款处理器采用12纳米工艺制造,具备两颗高性能Cortex-A75核心和六颗高效能Cortex-A55核心。此外,它还集成了先进的图像处理单元和语音唤醒功能,为用户提供卓越的性能体验。 ... [详细]
  • 深入解析 Android IPC 中的 Messenger 机制
    本文详细介绍了 Android 中基于消息传递的进程间通信(IPC)机制——Messenger。通过实例和源码分析,帮助开发者更好地理解和使用这一高效的通信工具。 ... [详细]
  • Android Studio 中查看应用程序崩溃日志的方法
    本文介绍如何在 Android Studio 中配置环境变量并使用 ADB 工具查看应用程序的崩溃日志,帮助开发者快速定位和解决问题。 ... [详细]
  • Python自动化测试入门:Selenium环境搭建
    本文详细介绍如何在Python环境中安装和配置Selenium,包括开发工具PyCharm的安装、Python环境的设置以及Selenium包的安装方法。此外,还提供了编写和运行第一个自动化测试脚本的步骤。 ... [详细]
  • 了解责任的苦与乐,是人生的重要课题。本文精选了一系列关于责任的经典语录,帮助读者更深入地理解责任感的重要性,并从中汲取力量。更多相关内容,请继续阅读。 ... [详细]
  • 本文详细介绍如何在 iOS 7 环境下申请苹果开发者账号,涵盖从访问开发者网站到最终激活账号的完整流程。包括选择个人或企业账号类型、付款方式及注意事项等。 ... [详细]
  • 本文详细介绍了如何在Android 4.4及以上版本中配置WebView以实现内容的自动高度调整和屏幕适配,确保中文显示正常,并提供代码示例。 ... [详细]
  • 解决MacOS上Android Studio Gradle版本不匹配问题
    在MacOS系统中,升级Android Studio后可能会遇到Gradle版本不兼容的问题。当网络下载更新受阻时,可以使用本地已安装的Gradle版本来解决问题。本文将详细介绍如何配置本地Gradle环境以确保开发工作的顺利进行。 ... [详细]
  • 本文介绍了如何通过Java代码计算一个整数的位数,并展示了多个基础编程示例,包括求和、平均分计算、条件判断等。 ... [详细]
  • 本题要求在一组数中反复取出两个数相加,并将结果放回数组中,最终求出最小的总加法代价。这是一个经典的哈夫曼编码问题,利用贪心算法可以有效地解决。 ... [详细]
  • 主调|大侠_重温C++ ... [详细]
  • 本篇文章介绍如何将两个分别表示整数的链表进行相加,并生成一个新的链表。每个链表节点包含0到9的数值,如9-3-7和6-3相加得到1-0-0-0。通过反向处理链表、逐位相加并处理进位,最终再将结果链表反向,即可完成计算。 ... [详细]
author-avatar
心胸宽大的榛子lcf
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有