Android拍照+方形剪裁——附代码与效果图

本文链接    http://blog.csdn.net/xiaodongrush/article/details/29173567

参考链接    http://stackoverflow.com/questions/12758425/how-to-set-the-output-image-use-com-android-camera-action-crop

1. 缘起

要开发一个头像上传的模块,头像上传过程分两步。第一步,相机拍照或者从图库选取照片,产生一个照片,第二步,提供头像剪裁,一般是剪裁为方形的。第三步,上传头像,删除不必要的缓存文件。

拍照和图库选择照片都可以使用系统的方案。自制相机可以搞滤镜,这个开发成本比较大,一般的APP也不用支持。图库选择照片这个可以自己做,访问sd卡,比较简单。问题出在图片剪裁上。网上有一些技术方案,迁移过来之后,效果不好,比如缩放的敏感度问题,缩放之后剪裁不准确的问题,缩放不流畅的问题。后来发现使用com.android.camera.action.CROP可以调用系统剪裁页面,但是该页面不是官方公开的页面,所以,某些厂商可能不支持这个。T_T。

看了下几款APP的头像截取,QQ、微信和易信都是自己做的,效果也不是很好,360手机助手是调用的系统的。系统的截取比较流畅,效果较好,除了前面的隐患,还有一个问题就是,在一些定制手机上面,剪裁的页面整体比较暗,或者上面有一层阴影,当然剪裁完之后图片是正常的。

考虑到360手机助手用户量这么大的APP,也在使用系统剪裁,所以我们也考虑使用系统剪裁,如果发现系统剪裁不可用,再调用自己的剪裁。

2. 拍照代码

这里用了MediaStore.EXTRA_OUTPUT,拍照图像不会通过intnet返回,要通过uri来读取,这样可以读小一点的图像进来。如果不这样,系统会在intent里面返回一个压缩图片,这个压缩的图像有多大不是能够控制的,所以可能比较大。

Intent newIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
newIntent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
startActivityForResult(newIntent, REQUEST_CODE_TAKE_PHOTO);

3. 方形剪裁

这里的代码与拍照类似,也用了MediaStore.EXTRA_OUTPUT,处理结果要通过路径来读取。

Intent intent = new Intent("com.android.camera.action.CROP");
intent.setDataAndType(uri, "image/*");
intent.putExtra("crop", "true"); // 开启剪裁
intent.putExtra("aspectX", 1); // 宽高比例
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 150); // 宽高
intent.putExtra("outputY", 150);
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(mOutputFile.getAbsoluteFile() + "tmp")));
startActivityForResult(intent, REQUEST_CODE_CLIP_PHOTO);

4. 相对完整的代码

代码下载链接,内附APK    http://download.csdn.net/detail/u011267546/7460367

public class MainActivity extends Activity {

    private static final int REQUEST_CODE_TAKE_PHOTO = 0;

    private static final int REQUEST_CODE_CLIP_PHOTO = 1;

    private File mOutputFile;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout. activity_main);
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (requestCode == REQUEST_CODE_TAKE_PHOTO) {
            onTakePhotoFinished(resultCode, data);
        } else if (requestCode == REQUEST_CODE_CLIP_PHOTO ) {
            onClipPhotoFinished(resultCode, data);
        }
    }

    public void onClick(View v) {
        if (v.getId() == R.id.take_photo) {
            if (hasCarema() == false) {
                return;
            }
            takePhoto();
        }
    }

    private boolean hasCarema() {
        PackageManager pm = getPackageManager();
        if (!pm.hasSystemFeature(PackageManager.FEATURE_CAMERA )
                && !pm.hasSystemFeature(PackageManager.FEATURE_CAMERA_FRONT )) {
            Toast. makeText(this, "no camera found", Toast.LENGTH_SHORT).show();
            return false ;
        }
        return true ;
    }

    private void takePhoto() {
        String sdPath = Environment.getExternalStorageDirectory()
                .getAbsolutePath();
        mOutputFile = new File(sdPath, System.currentTimeMillis() + ".tmp");
        Uri uri = Uri. fromFile(mOutputFile);
        Intent newIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE );
        newIntent.putExtra(MediaStore. EXTRA_OUTPUT, uri);
        startActivityForResult(newIntent, REQUEST_CODE_TAKE_PHOTO );
    }

    private void onTakePhotoFinished(int resultCode, Intent data) {
        if (resultCode == RESULT_CANCELED) {
            Toast. makeText(this, "take photo canceled", Toast.LENGTH_SHORT)
                    .show();
            return;
        } else if (resultCode != RESULT_OK) {
            Toast. makeText(this, "take photo failed", Toast.LENGTH_SHORT)
                    .show();
        } else {
            clipPhoto(Uri. fromFile(mOutputFile));
        }
    }

    // http://www.xuanyusong.com/archives/1743
    private void clipPhoto(Uri uri) {
        Intent intent = new Intent("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 是裁剪图片宽高
        intent.putExtra( "outputX", 150);
        intent.putExtra( "outputY", 150);
        intent.putExtra(MediaStore. EXTRA_OUTPUT,
                Uri. fromFile(new File(mOutputFile.getAbsoluteFile() + "tmp" )));
        startActivityForResult(intent, REQUEST_CODE_CLIP_PHOTO );
    }

    private void onClipPhotoFinished(int resultCode, Intent data) {
        if (resultCode == RESULT_CANCELED) {
            Toast. makeText(this, "clip photo canceled", Toast.LENGTH_SHORT)
                    .show();
            return;
        } else if (resultCode != RESULT_OK) {
            Toast. makeText(this, "take photo failed", Toast.LENGTH_SHORT)
                    .show();
        }
        Bitmap bm = BitmapFactory.decodeFile(mOutputFile.getAbsolutePath()
                + "tmp");
        ImageView photoIv = (ImageView) findViewById(R.id.photo );
        photoIv.setImageBitmap(bm);
    }
}

5. 效果图

献上可爱的大熊熊,只截取大熊的上半身。

Android拍照+方形剪裁——附代码与效果图

时间: 2024-10-11 05:45:42

Android拍照+方形剪裁——附代码与效果图的相关文章

task 1:完成省城市的三级联动(包括湖南省),附代码和效果图。

<!DOCTYPE html> <html> <head> <title>完成省城市的三级联动(包括湖南省),附代码和效果图.</title> </head> <body> <select id="province"> <option value="-1">请选择</option> </select> <select id=&qu

[Android] 拍照、截图、保存并显示在ImageView控件中

最近在做Android的项目,其中部分涉及到图像处理的内容.这里先讲述如何调用Camera应用程序进行拍照,并截图和保存显示在ImageView控件中以及遇到的困难和解决方法. PS:作者购买了本<Android第一行代码 著:郭霖>,参照里面的内容完成(推荐该书,前面的布局及应用非常不错).网上这类资料非常多,作者仅仅分享给初学者同时在线记录些内容,希望对大家有所帮助. 首先,设置activity_main.xml为LinearLayout布局且 android:orientation=&q

球体的双目视觉定位(matlab,附代码)

球体的双目视觉定位(matlab,附代码) 标签(空格分隔): 机器视觉 引言 双目视觉定位是我们的一个课程设计,最近刚做完,拿出来与大家分享一下,实验的目的是在拍摄的照片中识别球体,并求出该球体到相机的实际距离吗,我们要求需要用matlab,但是matlab调用双目摄像头(一个USB口)却老是只能调用双目摄像头中的一个,但是利用Python的OpenCV库却可以同时调用两个,因此我们选用了Python用于拍摄图片. 1.基本流程 备注:因为根据得出的视差图识别出圆球略困难,我们没有采用视差图深

Android拍照、相册选取、裁剪图片

来自:http://blog.csdn.net/ryantang03/article/details/8656278 package com.example.listactivity; import java.io.ByteArrayOutputStream; import java.io.File; import com.example.model.ImageTools; import android.app.Activity; import android.app.AlertDialog;

Android Permission denied 错误 ( 附Android权限大全 )

Android Permission denied 错误(附Android权限大全) java.net.SocketException: Permission denied (maybe missing INTERNET permission) 这是一个经典错误, Socket不能对外连接,错误不会被报出,调试的时候,能看到Exception, 这个Exception会有非常多变体. Android默认不同意訪问网络,所以,在AndroidManifest.xml中,须要进行例如以下配置: <u

Android拍照,相册选择图片以及Android6.0权限管理

概述 在android开发过程中,拍照或者从相册中选择图片是很常见的功能.下面要说得这个案例比较简单,用户点击按钮选择拍照或者打开相册选择图片,然后将选中的图片显示在手机上.android6.0后,推出了动态权限管理.以往我们将涉及到的权限全部写在清单文件中,只要用户安装了该程序,程序在运行过程中都会获得相应权限.android6.0后,对于一些特别敏感的权限,开发者必须在程序中进行声明.拍照和从相册选择图片都是涉及到用户隐私的敏感权限,必须在程序中进行声明. 大概的流程 创建布局文件,这里不多

QtAndroid详解(3):startActivity实战Android拍照功能

在"QtAndroid详解(1):QAndroidJniObject"中,我们介绍了 QAndroidJniObject 这个 Qt JNI 的核心类,在""中我们介绍了 startActivity 以及与它配套的一些 Android 背景知识,这次我们来看一个实例,演示如何使用 startActivity 来调用 Android 系统功能,同时也演示 QAndroidJniObject 的常见用法. 实例介绍 先看下实例效果,然后再论. 我们只是演示 API 用法

android 拍照注意问题

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); startActivityForResult(intent, reqCode);//reqCode是返回的code. 但是在有些手机会在protected void onActivityResult(int requestCode, int resultCode, Intent data)里返回的data为空,这时候我们就要想一些办法来解决了,这个是手机问题. 我们要设置系统

Android JUnit Test——批量运行测试代码

写Android测试用例有三要素,一是我们用的“安卓模拟器device”(用来显示UI),二是“uiautomatorviewer.bat”(用来定位UI上的元素),三是“Robotium”中提供的类(用来与UI元素进行交互).在写好Android的测试用例后,可通过Android JUnit Test批量运行测试代码.一共分四步—— 第一步:写一个suite类(suite中包含指定的测试用例,如下把SendInfoTest类作为一个测试用例添加进了suite中,LoginTest这个类由于被注