Android实现简单图库辅助器

 更新时间:2017年04月06日 10:34:20   作者:YegodXin  
这篇文章主要为大家详细介绍了Android实现简单图库辅助器的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

写在前面

实际开发经常会遇到读取相册或者拍照功能,网上也很非常多图库框架,都各有风格,也因此与自己的项目格格不入。再者,框架API太多,需要的配置太多,还要吃力研究。所以,本人摸索大多图库框架,提炼核心,写一个只提供核心的辅助类,剩下的就可以自行玩耍。

实现步骤

第一步,创建辅助类,使用弱引用持有Activity,防止内存溢出。

  public class GalleryHelper{

    private Activity mActivity;

    public GalleryHelper(Activity activity) {
      mActivity = new WeakReference<>(activity).get();
    }
  }

第二步,创建文件夹实体类,代表文件夹数据。paths集合是文件夹下的所有路径。

public final class FolderEntity {

  private int num;
  private String name;
  private List<String> paths = new ArrayList<>();

  public int getNum() {
    return paths.size();
  }

  public List<String> getPaths() {
    return paths;
  }

  public void setName(String name) {
    this.name = name;
  }
}

第三步,首先获取手机所有的图片,在Activity里有getLoaderManager方法获取一个LoaderManager实例,该类用于异步加载手机内数据监测,这里不做多分析。我们调用它的initLoader方法,前两个参数这里不需要,只要实现LoaderCallbacks接口,并且指定Cursor类型。LoaderCallbacks接口有三个覆盖方法,我们需要用到的是onCreateLoader方法和onLoadFinished方法,前者是初始化Loader,后者是加载完成后的回调。

  mActivity.getLoaderManager().initLoader(0, null, new LoaderManager.LoaderCallbacks<Cursor>() {
      @Override
      public Loader<Cursor> onCreateLoader(int i, Bundle bundle) {
      }

      @Override
      public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {
      }

      @Override
      public void onLoaderReset(Loader<Cursor> loader) {

      }
    });

在onCreateLoader方法里,初始化CursorLoader,参数跟Cursor类下的query一样,第一个参数指定外部村粗多媒体URI;第二个参数是查找结果字段,这里只要了路径;第三个参数和第四个参数是搜索条件,条件为搜索jpeg格式和png格式,最后一个是按时间倒序搜索。

  public Loader<Cursor> onCreateLoader(int i, Bundle bundle) {
     return new CursorLoader(mActivity,
         MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
         new String[]{MediaStore.Images.Media.DATA},
         MediaStore.Images.Media.MIME_TYPE + "=? or "
             + MediaStore.Images.Media.MIME_TYPE + "=?",
         new String[]{"image/jpeg", "image/png"},
         MediaStore.Images.Media.DATE_ADDED + " DESC");
   }

CursorLoader初始化完成之后,搜索的结果会回调在onLoadFinished方法。这时就可以处理搜索出来的图片路径。因为图片路径是没有分类,这里采用HashMap分类,以文件夹路径为key,具体文件夹FolderEntity类为value,如果当前图片路径的文件夹不存在则创建FolderEntity并且放入HashMap,存在则获取FolderEntity。

  public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {
     HashMap<String, FolderEntity> folderEntityHashMap = new HashMap<String, FolderEntity>();
      if (cursor != null) {
        while (cursor.moveToNext()) {
          //图片路径
          String path = cursor.getString(
          cursor.getColumnIndex(MediaStore.Images.Media.DATA));

          // 路径不存在或者文件不存在就跳过
          File file = new File(path);
          if (TextUtils.isEmpty(path) || !file.exists()) {
            continue;
          }
          String folerPath = file.getParent();
          FolderEntity folderEntity;
          if (folderEntityHashMap.containsKey(folerPath)) {
            folderEntity = folderEntityHashMap.get(folerPath);
          } else {
            folderEntity = new FolderEntity();
            folderEntityHashMap.put(file.getParentFile().getName(), folderEntity);
          }
          folderEntity.getPaths().add(path);
        }
        cursor.close();
      }

    }

但是数据是要提供给外部的,HashMap就显得很麻烦,所以要转换ArrayList,并且按数量大小进行顺序。

    private ArrayList<FolderEntity> map2List(HashMap<String, FolderEntity> mediaBeanMap) {
      Iterator<FolderEntity> iterator = mediaBeanMap.values().iterator();
      ArrayList<FolderEntity> list = new ArrayList<FolderEntity>();
      while (iterator.hasNext()) {
        list.add(iterator.next());
      }
      Collections.sort(list, new Comparator<FolderEntity>() {
        @Override
        public int compare(FolderEntity lhs, FolderEntity rhs) {
          return lhs.getNum() > rhs.getNum() ? 1 : -1;
        }
      });
      return list;
    }

使用接口将数据提供给外部。

  public interface GalleryCallback {
    void complete(List<FolderEntity> list);
  }

  callback.complete(map2List(folderEntityHashMap));

第四步是实现拍照功能,这里实现是创建文件再启动拍照功能。

  File imageStoreDir = new File(Environment.getExternalStorageDirectory(),
      "/DCIM/" + mActivity.getResources().getString(R.string.app_name));
  if (!imageStoreDir.exists()) {
    imageStoreDir.mkdir();
  }
  Intent captureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
  if (captureIntent.resolveActivity(mActivity.getPackageManager()) != null) {
    SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMddHHmmss", Locale.CHINA);
    String filename = String.format("IMG%s", dateFormat.format(new Date()));
    imagePath = new File(imageStoreDir, filename).getAbsolutePath();
    captureIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(imagePath)));
    mActivity.startActivityForResult(new Intent(
        MediaStore.ACTION_IMAGE_CAPTURE), REQUEST_CODE);
  }

定义回调接口,接收Activit的onActivityResult方法,表示回调成功把上面创建好的文件路径提供外部。

  public interface CameraCallback {
    void complete(String path);
  }

  public void onActivityResult(int requestCode, int resultCode) {
    if (Activity.RESULT_OK == resultCode && REQUEST_CODE == requestCode) {
      if (cameraCallback != null) {
        cameraCallback.complete(imagePath);
      }
    }
  }

外部调用

  galleryHelper = new GalleryHelper(this);
    galleryHelper.loadImages(new GalleryHelper.GalleryCallback() {
      @Override
      public void complete(List<FolderEntity> list) {
        //加载本地图片返回结果
      }
    });

 findViewById(R.id.btn_camera).setOnClickListener(new View.OnClickListener() {
     @Override
     public void onClick(View v) {
       galleryHelper.openCamera(new GalleryHelper.CameraCallback() {
         @Override
         public void complete(String path) {
         //拍照返回结果
         }
       });
     }
   });

  @Override
  protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    //使用拍照,必须接收Activity的onActivityResult方法
    galleryHelper.onActivityResult(requestCode, resultCode);
  }

别忘了加权限,这里为了简单实现,我把targetSdkVersion设置23以下,23和23以上的需要自行加上动态权限。

  <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
  <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
  <uses-permission android:name="android.permission.CAMERA" />
  <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />

整体就完成了,使用非常方便,有了这图库辅助器就可以自定义风格,再也不用受约束。网上大多开源图片选择器的搜索图片都是如此,但这例子难免有bug,不足之处望指教。

github地址:https://github.com/tanxinye/GalleryHelper

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

相关文章

  • Android实现SQLite添加、更新及删除行的方法

    Android实现SQLite添加、更新及删除行的方法

    这篇文章主要介绍了Android实现SQLite添加、更新及删除行的方法,涉及Android基于SQLiteDatabase类操作SQLite数据库的基本技巧,需要的朋友可以参考下
    2016-08-08
  • 老生常谈Android HapticFeedback(震动反馈)

    老生常谈Android HapticFeedback(震动反馈)

    下面小编就为大家带来一篇老生常谈Android HapticFeedback(震动反馈)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-04-04
  • Android实现视图轮播效果

    Android实现视图轮播效果

    这篇文章主要为大家详细介绍了Android实现视图轮播效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-10-10
  • Android实现倒计时结束后跳转页面功能

    Android实现倒计时结束后跳转页面功能

    最近在工作中遇到一个需求,需要在倒计时一段时间后进行跳转页面,通过查找相关资料发现其中涉及的知识还不少,所以分享出来,下面这篇文章主要给大家介绍了关于Android实现倒计时结束后跳转页面功能的相关资料,需要的朋友可以参考下。
    2017-11-11
  • HandlerThread的使用场景和用法详解

    HandlerThread的使用场景和用法详解

    这篇文章主要介绍了HandlerThread的使用场景和用法详解,HandlerThread是Android中的一个线程类,它是Thread的子类,并且内部封装了Looper和Handler,提供了更方便的消息处理和线程操作,需要的朋友可以参考下
    2023-07-07
  • Android自定义圆点指示器

    Android自定义圆点指示器

    这篇文章主要为大家详细介绍了Android自定义圆点指示器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-06-06
  • Kotlin RadioGroup与ViewPager实现底层分页按钮方法

    Kotlin RadioGroup与ViewPager实现底层分页按钮方法

    安卓的控件是挺多的,没有办法一个一个的来说明,我们挑出了一些重点的控件,组成一些常见的布局,这样以后在遇到相同功能的界面时,就会有自己的思路,或者进行复用
    2022-12-12
  • Android studio 解决logcat无过滤工具栏的操作

    Android studio 解决logcat无过滤工具栏的操作

    这篇文章主要介绍了Android studio 解决logcat无过滤工具栏的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-04-04
  • Android 获取屏幕尺寸实例代码

    Android 获取屏幕尺寸实例代码

    这篇文章主要介绍了Android 获取屏幕尺寸实例代码的相关资料,需要的朋友可以参考下
    2017-06-06
  • Android Activity打开后被应用快照遮住的问题

    Android Activity打开后被应用快照遮住的问题

    这篇文章主要介绍了Android Activity打开后被应用快照遮住的问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-01-01

最新评论