[置顶] Android实现仿微信朋友圈发布动态(拍照、图库选择、照片压缩、显示、保存、缩略图、点击缩略图删除对应文件等)附源码
2016-08-02 12:11
856 查看
原创作品,转载请注明出处:http://blog.csdn.net/zhang3776813/article/details/52092591
最近项目需求中要用到类似微信朋友圈发布动态选择图片的UI效果,研究了一下,特来分享成果,多的不说来看代码。
Activity类
图片压缩的工具类:
九宫格缩略图适配器类:
package com.whfp.adapter;
import java.util.ArrayList;
import java.util.List;
import com.whfp.circle_of_friends.R;
import android.content.Context;
import android.graphics.Bitmap;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
/**
* 九宫格缩略图适配器
* @author 张明杨
*
*/
public class GridViewAdapter extends BaseAdapter {
private Context mContext;
private List<Bitmap> list = new ArrayList<Bitmap>();
public GridViewAdapter() {
super();
}
/**
* 获取列表数据
* @param list
*/
public void setList(List<Bitmap> list){
this.list = list;
this.notifyDataSetChanged();
}
public GridViewAdapter(Context mContext,List<Bitmap> list) {
super();
this.mContext = mContext;
this.list = list;
}
@Override
public int getCount() {
Log.e(" ", list.size()+"");
if(list==null){
return 1;
}else if(list.size()==9){
return 9;
}else{
return list.size()+1;
}
}
@Override
public Object getItem(int position) {
if (list != null
&& list.size() == 9)
{
return list.get(position);
}
else if (list == null || position - 1 < 0
|| position > list.size())
{
return null;
}
else
{
return list.get(position - 1);
}
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if(convertView==null){
convertView = LayoutInflater.from(mContext).inflate(R.layout.item_published_grida, null);
holder = new ViewHolder();
holder.item_grida_image = (ImageView) convertView.findViewById(R.id.item_grida_image);
convertView.setTag(holder);
}else{
holder = (ViewHolder) convertView.getTag();
}
if (isShowAddItem(position))
{
holder.item_grida_image.setImageResource(R.drawable.btn_add_pic);
holder.item_grida_image.setBackgroundResource(R.color.bg_gray);
}
else
{
holder.item_grida_image.setImageBitmap(list.get(position));
holder.item_grida_image.setBackgroundResource(R.color.bg_gray);
}
return convertView;
}
/**
* 判断当前下标是否是最大值
* @param position 当前下标
* @return
*/
private boolean isShowAddItem(int position)
{
int size = list == null ? 0 : list.size();
return position == size;
}
class ViewHolder{
ImageView item_grida_image;
}
}
效果图 :
不会做动态图,就随便截了几张
九宫格展示压缩后的缩略图:
点击“+”号添加图片:
点击缩略图可删除对应的源文件:
点此下载源码
最近项目需求中要用到类似微信朋友圈发布动态选择图片的UI效果,研究了一下,特来分享成果,多的不说来看代码。
Activity类
package com.whfp.circle_of_friends; import java.io.File; import java.io.FileNotFoundException; import java.io.IOException; import java.util.ArrayList; import java.util.List; import com.whfp.adapter.GridViewAdapter; import com.whfp.utils.FileUtils; import com.whfp.utils.ImageTools; import android.content.ContentResolver; import android.content.DialogInterface; import android.content.Intent; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.net.Uri; import android.os.Bundle; import android.os.Environment; import android.provider.MediaStore; import android.view.View; import android.widget.AdapterView; import android.widget.Toast; import android.widget.AdapterView.OnItemClickListener; import android.widget.GridView; /** * 仿微信朋友圈发布动态 * 拍照或图库选择 * 压缩图片并保存 * @author 张明杨 * */ public class MainActivity extends BasicActivity implements OnItemClickListener { // 图片 九宫格 private GridView gv; // 图片 九宫格适配器 private GridViewAdapter gvAdapter; // 用于保存图片资源文件 private List<Bitmap> lists = new ArrayList<Bitmap>(); // 用于保存图片路径 private List<String> list_path = new ArrayList<String>(); // 拍照 public static final int IMAGE_CAPTURE = 1; // 从相册选择 public static final int IMAGE_SELECT = 2; // 照片缩小比例 private static final int SCALE = 5; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); init(); } /** * 初始化控件 */ private void init() { gv = (GridView) findViewById(R.id.noScrollgridview); gvAdapter = new GridViewAdapter(this, lists); gv.setOnItemClickListener(this); gv.setAdapter(gvAdapter); gvAdapter.setList(lists); } @Override protected void onDestroy() { //删除文件夹及文件 FileUtils.deleteDir(); super.onDestroy(); } /** * 拍照 * * @param path * 照片存放的路径 */ public void captureImage(String path) { Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); // 指定照片保存路径(SD卡),image.jpg为一个临时文件,每次拍照后这个图片都会被替换 Uri uri = Uri.fromFile(new File(Environment.getExternalStorageDirectory(), "image.jpg")); intent.putExtra(MediaStore.EXTRA_OUTPUT, uri); startActivityForResult(intent, IMAGE_CAPTURE); } /** * 从图库中选取图片 */ public void selectImage() { Intent intent = new Intent(); intent.setType("image/*"); intent.setAction(Intent.ACTION_PICK); startActivityForResult(intent, IMAGE_SELECT); } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (resultCode == RESULT_OK && resultCode != RESULT_CANCELED) { String fileName; switch (requestCode) { case IMAGE_CAPTURE:// 拍照返回 // 将保存在本地的图片取出并缩小后显示在界面上 Bitmap bitmap = BitmapFactory.decodeFile(Environment.getExternalStorageDirectory() + "/image.jpg"); Bitmap newBitmap = ImageTools.zoomBitmap(bitmap,bitmap.getWidth() / SCALE, bitmap.getHeight() / SCALE); // 由于Bitmap内存占用较大,这里需要回收内存,否则会报out of memory异常 bitmap.recycle(); // 生成一个图片文件名 fileName = String.valueOf(System.currentTimeMillis()); // 将处理过的图片添加到缩略图列表并保存到本地 ImageTools.savePhotoToSDCard(newBitmap, FileUtils.SDPATH,fileName); lists.add(newBitmap); list_path.add(fileName+".jpg"); for (int i = 0; i < list_path.size(); i++) { logI("第"+i+"张照片的地址:"+list_path.get(i)); } // 更新GrideView gvAdapter.setList(lists); break; case IMAGE_SELECT:// 选择照片返回 ContentResolver resolver = getContentResolver(); // 照片的原始资源地址 Uri originalUri = data.getData(); try { // 使用ContentProvider通过URI获取原始图片 Bitmap photo = MediaStore.Images.Media.getBitmap(resolver,originalUri); if (photo != null) { // 为防止原始图片过大导致内存溢出,这里先缩小原图显示,然后释放原始Bitmap占用的内存 Bitmap smallBitmap = ImageTools.zoomBitmap(photo,photo.getWidth() / SCALE, photo.getHeight()/ SCALE); // 释放原始图片占用的内存,防止out of memory异常发生 photo.recycle(); // 生成一个图片文件名 fileName = String.valueOf(System.currentTimeMillis()); // 将处理过的图片添加到缩略图列表并保存到本地 ImageTools.savePhotoToSDCard(smallBitmap, FileUtils.SDPATH,fileName); lists.add(smallBitmap); list_path.add(fileName+".jpg"); for (int i = 0; i < list_path.size(); i++) { logI("第"+i+"照片的地址:"+list_path.get(i)); } // 更新GrideView gvAdapter.setList(lists); } } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } break; default: break; } } } @Override public void onItemClick(AdapterView<?> parent, View view, final int position, long id) { Toast.makeText(getApplicationContext(), "" + position + getDataSize(),Toast.LENGTH_SHORT).show(); if (position == getDataSize()) {// 点击“+”号位置添加图片 showAlertDialog(false, "提示", new String[] { "拍照", "从图库选择", "取消" }, new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { switch (which + 1) { case 1:// 拍照 captureImage(FileUtils.SDPATH); dialog.dismiss(); break; case 2:// 从图库选择 selectImage(); dialog.dismiss(); break; case 3:// 取消 dialog.dismiss(); break; default: break; } } }); } else {// 点击图片删除 showAlertDialog("提示", "是否删除此图片?", "确定", "取消", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { lists.remove(position); FileUtils.delFile(list_path.get(position)); list_path.remove(position); gvAdapter.setList(lists); dialog.dismiss(); } }, new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { dialog.dismiss(); } }); } } private int getDataSize() { return lists == null ? 0 : lists.size(); } }
图片压缩的工具类:
package com.whfp.utils; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Matrix; import android.graphics.drawable.BitmapDrawable; import android.graphics.drawable.Drawable; import android.os.Environment; import android.util.Log; public class FileUtils { /** * 生成文件夹路径 */ public static String SDPATH = Environment.getExternalStorageDirectory() + "/TEST_PY/"; /** * 将图片压缩保存到文件夹 * * @param bm * @param picName */ public static void saveBitmap(Bitmap bm, String picName) { try { // 如果没有文件夹就创建一个程序文件夹 if (!isFileExist("")) { File tempf = createSDDir(""); } File f = new File(SDPATH, picName + ".JPEG"); // 如果该文件夹中有同名的文件,就先删除掉原文件 if (f.exists()) { f.delete(); } FileOutputStream out = new FileOutputStream(f); bm.compress(Bitmap.CompressFormat.JPEG, 90, out); out.flush(); out.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } /** * 质量压缩 并返回Bitmap * * @param image * 要压缩的图片 * @return 压缩后的图片 */ private Bitmap compressImage(Bitmap image) { ByteArrayOutputStream baos = new ByteArrayOutputStream(); image.compress(Bitmap.CompressFormat.JPEG, 100, baos);// 质量压缩方法,这里100表示不压缩,把压缩后的数据存放到baos中 int options = 100; while (baos.toByteArray().length / 1024 > 100) { // 循环判断如果压缩后图片是否大于100kb,大于继续压缩 baos.reset();// 重置baos即清空baos image.compress(Bitmap.CompressFormat.JPEG, options, baos);// 这里压缩options%,把压缩后的数据存放到baos中 options -= 10;// 每次都减少10 } ByteArrayInputStream isBm = new ByteArrayInputStream(baos.toByteArray());// 把压缩后的数据baos存放到ByteArrayInputStream中 Bitmap bitmap = BitmapFactory.decodeStream(isBm, null, null);// 把ByteArrayInputStream数据生成图片 return bitmap; } /** * 质量压缩 * * @param bitmap * @param picName */ public static void compressImageByQuality(final Bitmap bitmap, String picName) { // 如果没有文件夹就创建一个程序文件夹 if (!isFileExist("")) { try { File tempf = createSDDir(""); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } File f = new File(SDPATH, picName + ".JPEG"); // 如果该文件夹中有同名的文件,就先删除掉原文件 if (f.exists()) { f.delete(); } ByteArrayOutputStream baos = new ByteArrayOutputStream(); int options = 100; // 质量压缩方法,这里100表示不压缩,把压缩后的数据存放到baos中 bitmap.compress(Bitmap.CompressFormat.JPEG, options, baos); // 循环判断如果压缩后图片是否大于200kb,大于继续压缩 while (baos.toByteArray().length / 1024 > 500) { // 重置baos即让下一次的写入覆盖之前的内容 baos.reset(); // 图片质量每次减少5 options -= 5; // 如果图片质量小于10,则将图片的质量压缩到最小值 if (options < 0) options = 0; // 将压缩后的图片保存到baos中 bitmap.compress(Bitmap.CompressFormat.JPEG, options, baos); // 如果图片的质量已降到最低则,不再进行压缩 if (options == 0) break; } // 将压缩后的图片保存的本地上指定路径中 FileOutputStream fos; try { fos = new FileOutputStream(new File(SDPATH, picName + ".JPEG")); fos.write(baos.toByteArray()); fos.flush(); fos.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } /** * 创建文件夹 * * @param dirName * 文件夹名称 * @return 文件夹路径 * @throws IOException */ public static File createSDDir(String dirName) throws IOException { File dir = new File(SDPATH + dirName); if (Environment.getExternalStorageState().equals( Environment.MEDIA_MOUNTED)) { System.out.println("createSDDir:" + dir.getAbsolutePath()); System.out.println("createSDDir:" + dir.mkdir()); } return dir; } /** * 判断改文件是否是一个标准文件 * * @param fileName * 判断的文件路径 * @return 判断结果 */ public static boolean isFileExist(String fileName) { File file = new File(SDPATH + fileName); file.isFile(); return file.exists(); } /** * 删除指定文件 * * @param fileName */ public static void delFile(String fileName) { File file = new File(SDPATH + fileName); if (file.isFile()) { file.delete(); } file.exists(); } /** * 删除指定文件 * @param file */ public static void deleteFile(File file) { if (file.exists()) { // 判断文件是否存在 if (file.isFile()) { // 判断是否是文件 file.delete(); // delete()方法 你应该知道 是删除的意思; } else if (file.isDirectory()) { // 否则如果它是一个目录 File files[] = file.listFiles(); // 声明目录下所有的文件 files[]; for (int i = 0; i < files.length; i++) { // 遍历目录下所有的文件 deleteFile(files[i]); // 把每个文件 用这个方法进行迭代 } } file.delete(); } else { Log.i("TAG", "文件不存在!"); } } /** * 删除指定文件夹中的所有文件 */ public static void deleteDir() { File dir = new File(SDPATH); if (dir == null || !dir.exists() || !dir.isDirectory()) return; for (File file : dir.listFiles()) { if (file.isFile()) file.delete(); else if (file.isDirectory()) deleteDir(); } dir.delete(); } /** * 判断是否存在该文件 * * @param path * 文件路径 * @return */ public static boolean fileIsExists(String path) { try { File f = new File(path); if (!f.exists()) { return false; } } catch (Exception e) { return false; } return true; } }
九宫格缩略图适配器类:
package com.whfp.adapter;
import java.util.ArrayList;
import java.util.List;
import com.whfp.circle_of_friends.R;
import android.content.Context;
import android.graphics.Bitmap;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
/**
* 九宫格缩略图适配器
* @author 张明杨
*
*/
public class GridViewAdapter extends BaseAdapter {
private Context mContext;
private List<Bitmap> list = new ArrayList<Bitmap>();
public GridViewAdapter() {
super();
}
/**
* 获取列表数据
* @param list
*/
public void setList(List<Bitmap> list){
this.list = list;
this.notifyDataSetChanged();
}
public GridViewAdapter(Context mContext,List<Bitmap> list) {
super();
this.mContext = mContext;
this.list = list;
}
@Override
public int getCount() {
Log.e(" ", list.size()+"");
if(list==null){
return 1;
}else if(list.size()==9){
return 9;
}else{
return list.size()+1;
}
}
@Override
public Object getItem(int position) {
if (list != null
&& list.size() == 9)
{
return list.get(position);
}
else if (list == null || position - 1 < 0
|| position > list.size())
{
return null;
}
else
{
return list.get(position - 1);
}
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if(convertView==null){
convertView = LayoutInflater.from(mContext).inflate(R.layout.item_published_grida, null);
holder = new ViewHolder();
holder.item_grida_image = (ImageView) convertView.findViewById(R.id.item_grida_image);
convertView.setTag(holder);
}else{
holder = (ViewHolder) convertView.getTag();
}
if (isShowAddItem(position))
{
holder.item_grida_image.setImageResource(R.drawable.btn_add_pic);
holder.item_grida_image.setBackgroundResource(R.color.bg_gray);
}
else
{
holder.item_grida_image.setImageBitmap(list.get(position));
holder.item_grida_image.setBackgroundResource(R.color.bg_gray);
}
return convertView;
}
/**
* 判断当前下标是否是最大值
* @param position 当前下标
* @return
*/
private boolean isShowAddItem(int position)
{
int size = list == null ? 0 : list.size();
return position == size;
}
class ViewHolder{
ImageView item_grida_image;
}
}
效果图 :
不会做动态图,就随便截了几张
九宫格展示压缩后的缩略图:
点击“+”号添加图片:
点击缩略图可删除对应的源文件:
点此下载源码
相关文章推荐
- Android实现仿微信朋友圈发布动态(拍照、图库选择、照片压缩、显示、保存、缩略图、点击缩略图删除对应文件等)
- android拍照选择图库后将照片剪裁压缩显示到imageview上
- 如何兼容所有Android版本选择照片或拍照然后裁剪图片--基于FileProvider和动态权限的实现
- TakePhoto是一款用于在Android设备上获取照片(拍照或从相册、文件中选择)、裁剪图片、压缩图片的开源工具库,目前最新版本3.0.0。
- [置顶] 利用photoView实现可点击放大到全屏显示的控件,效果类似于微信朋友圈点开看大图
- Android实现照片选择显示-从相册选择或拍照选择
- 一款用于在Android设备上获取照片(拍照或从相册、文件中选择)、裁剪图片、压缩图片的开源工具库
- Android 把bitmap转换成String,计算图片的缩放值,根据路径获得突破并压缩返回bitmap用于显示,获取保存 隐患检查的图片文件夹名称,获取保存图片的目录,根据路径删除图片,添加到图库
- android 打开系统相机拍照并保存文件,显示图片。
- Android 选择文件对话框并将选择内容添加至ListView,并实现listview的删除与事件触发
- Ionic系列——调用摄像头拍照和选择图库照片功能的实现
- Android仿微信选择照片,拍照
- [置顶] Android仿微信APP实现源码
- Android 高仿微信 获取最近刚刚拍照的缩略图 功能实现
- Android调用系统相机拍照,从图库中选择照片,调用系统摄像机录像
- Android中照相,从相册选取照片,压缩,保存到手机内存,展示到界面,点击放大,上传。(含demo)
- Android 三星手机拍照,从图库选择照片旋转问题完美解决
- android 图片压缩,自动旋转保存,查看照片信息,拍照设备类型等
- Android拍照或图库选择照片上传(附服务器端代码)
- Android 仿QQ首页ListView左滑置顶、删除等实现源码