保存照片和视频到相册显示

照片和视频保存到本地的方法大致都是通过流的方式写入文件里面就可以达到保存到文件夹的目的,但是你保存到文件夹的资源却不一定能够在相册显示出来,只能翻看文件管理。
怎么能够将保存到本地的照片视频显示在系统相册中,最常用的方式是发送广播扫描的方式来通知系统扫描文件夹,但是这种方式经常不起作用,根本原因在于,向系统发送广播时系统只会去扫描系统资源的相册,如果你保存的文件夹是自己建立的,那么你发广播时系统是不会扫描到的。
解决这个问题,想要显示在相册需要将数据插入到ContentProvider中,因此对于非系统能够扫描到的相册我们使用手动插入的方式。

2. 保存到系统资源相册

系统相册指的的是CameraDCIM等等这些目录对应的相册,可能还有其他系统会自动扫描的相册,但是暂时没有去整理这些。
对于系统相册来说,只需要发送广播进行扫描即可,数据会自动添加到ContentProvider中,当然如果不发送广播,在手机重启或者过一段时间之后,扫描操作仍会开启,因此绝对绝对不可以自己去进行插入操作,否则相册中会出现两张相同的照片。

/**
 * 针对系统文夹只需要扫描,不用插入内容提供者,不然会重复
 *
 * @param context  上下文
 * @param filePath 文件路径
 */
private static void scanFile(Context context, String filePath) {
    if (!checkFile(filePath))
        return;
    Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
    intent.setData(Uri.fromFile(new File(filePath)));
    context.sendBroadcast(intent);
}

3. 保存到非系统资源相册

保存到非系统资源相册中时,我们就需要进行ContentProvider的插入更新,来达到可以在相册显示的目的。

3.1 初始化ContentValues公共字段

照片和视频是有一些公共字段,写一个初始化公共字段的方法,简化MediaStore字段的写入操作。

/**
 * 插入时初始化公共字段
 *
 * @param filePath 文件
 * @param time     ms
 * @return ContentValues
 */
private static ContentValues initCommonContentValues(String filePath, long time) {
    ContentValues values = new ContentValues();
    File saveFile = new File(filePath);
    long timeMillis = getTimeWrap(time);
    values.put(MediaStore.MediaColumns.TITLE, saveFile.getName());
    values.put(MediaStore.MediaColumns.DISPLAY_NAME, saveFile.getName());
    values.put(MediaStore.MediaColumns.DATE_MODIFIED, timeMillis);
    values.put(MediaStore.MediaColumns.DATE_ADDED, timeMillis);
    values.put(MediaStore.MediaColumns.DATA, saveFile.getAbsolutePath());
    values.put(MediaStore.MediaColumns.SIZE, saveFile.length());
    return values;
}

3.2 插入照片资源

保存照片到本地,并通知相册显示,需要注意的时间的单位必须是ms

/**
 * 保存到照片到本地,并插入MediaStore以保证相册可以查看到
 * 这是更优化的方法,防止读取的照片获取不到宽高
 *
 * @param context    上下文
 * @param filePath   文件路径
 * @param createTime 创建时间 <=0时为当前时间 ms
 * @param width      宽度
 * @param height     高度
 */
public static void insertImageToMediaStore(Context context, String filePath,
                                           long createTime, int width, int height) {
    if (!checkFile(filePath))
        return;
    createTime = getTimeWrap(createTime);
    ContentValues values = initCommonContentValues(filePath, createTime);
    values.put(MediaStore.Images.ImageColumns.DATE_TAKEN, createTime);
    values.put(MediaStore.Images.ImageColumns.ORIENTATION, 0);
    values.put(MediaStore.Images.ImageColumns.ORIENTATION, 0);
    if (Build.VERSION.SDK_INT > Build.VERSION_CODES.JELLY_BEAN) {
        if (width > 0) values.put(MediaStore.Images.ImageColumns.WIDTH, 0);
        if (height > 0) values.put(MediaStore.Images.ImageColumns.HEIGHT, 0);
    }
    values.put(MediaStore.MediaColumns.MIME_TYPE, getPhotoMimeType(filePath));
    context.getApplicationContext().getContentResolver()
            .insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
}

3.3 插入视频资源

保存视频到本地,并通知相册显示,需要注意的时间的单位必须是ms

/**
 * 保存到视频到本地,并插入MediaStore以保证相册可以查看到
 * 这是更优化的方法,防止读取的视频获取不到宽高
 *
 * @param context    上下文
 * @param filePath   文件路径
 * @param createTime 创建时间 <=0时为当前时间 ms
 * @param duration   视频长度 ms
 * @param width      宽度
 * @param height     高度
 */
public static void insertVideoToMediaStore(Context context, String filePath, long createTime,
                                           int width, int height, long duration) {
    if (!checkFile(filePath))
        return;
    createTime = getTimeWrap(createTime);
    ContentValues values = initCommonContentValues(filePath, createTime);
    values.put(MediaStore.Video.VideoColumns.DATE_TAKEN, createTime);
    if (duration > 0)
        values.put(MediaStore.Video.VideoColumns.DURATION, duration);
    if (Build.VERSION.SDK_INT > Build.VERSION_CODES.JELLY_BEAN) {
        if (width > 0) values.put(MediaStore.Video.VideoColumns.WIDTH, width);
        if (height > 0) values.put(MediaStore.Video.VideoColumns.HEIGHT, height);
    }
    values.put(MediaStore.MediaColumns.MIME_TYPE, getVideoMimeType(filePath));
    context.getContentResolver().insert(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, values);
}

4. 注意

在应用过程中发现,vivo手机和魅族手机部分机型只支持在文件管理中查看视频,使用本文描述的方法添加后没有效果,用微信保存视频试了一下同样不能在相册中将视频显示出来,应该是手机的原因,特此声明。

5.源码

贴一下工具类源代码,多了一些辅助方法,比如获取mime_type的参数等方法。

/**
 * CreateAt : 2017/5/24
 * Describe : 相册更新通知帮助类
 * 创建时间单位ms
 * 视频时长单位ms
 *
 * @author chendong
 */
public class AlbumNotifyHelper {

    public static final String TAG = AlbumNotifyHelper.class.getSimpleName();


    ///////////////////////////////////////////////////////////////////////////
    // 下面是对外公开的重载的方法
    ///////////////////////////////////////////////////////////////////////////

    public static void notifyScanDcim(Context context, String filePath) {
        scanFile(context, filePath);
    }

    public static void insertVideoToMediaStore(Context context, String filePath, long dateTaken, long duration) {
        insertVideoToMediaStore(context, filePath, dateTaken, 0, 0, duration);
    }

    public static void insertVideoToMediaStore(Context context, VideoUtil.VideoInfo videoInfo) {
        insertVideoToMediaStore(context, videoInfo.originalVideoFilePath, videoInfo.dateTaken, videoInfo.width, videoInfo.height, videoInfo.duringTime);
    }

    public static void insertImageToMediaStore(Context context, String filePath, long createTime) {
        insertImageToMediaStore(context, filePath, createTime, 0, 0);
    }


    ///////////////////////////////////////////////////////////////////////////
    // 扫描系统相册核心方法
    ///////////////////////////////////////////////////////////////////////////

    /**
     * 针对系统文夹只需要扫描,不用插入内容提供者,不然会重复
     *
     * @param context  上下文
     * @param filePath 文件路径
     */
    public static void scanFile(Context context, String filePath) {
        if (!checkFile(filePath))
            return;
        Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
        intent.setData(Uri.fromFile(new File(filePath)));
        context.sendBroadcast(intent);
    }


    ///////////////////////////////////////////////////////////////////////////
    // 非系统相册像MediaContent中插入数据,核心方法
    ///////////////////////////////////////////////////////////////////////////

    /**
     * 针对非系统文件夹下的文件,使用该方法
     * 插入时初始化公共字段
     *
     * @param filePath 文件
     * @param time     ms
     * @return ContentValues
     */
    private static ContentValues initCommonContentValues(String filePath, long time) {
        ContentValues values = new ContentValues();
        File saveFile = new File(filePath);
        long timeMillis = getTimeWrap(time);
        values.put(MediaStore.MediaColumns.TITLE, saveFile.getName());
        values.put(MediaStore.MediaColumns.DISPLAY_NAME, saveFile.getName());
        values.put(MediaStore.MediaColumns.DATE_MODIFIED, timeMillis);
        values.put(MediaStore.MediaColumns.DATE_ADDED, timeMillis);
        values.put(MediaStore.MediaColumns.DATA, saveFile.getAbsolutePath());
        values.put(MediaStore.MediaColumns.SIZE, saveFile.length());
        return values;
    }

    /**
     * 保存到照片到本地,并插入MediaStore以保证相册可以查看到,这是更优化的方法,防止读取的照片获取不到宽高
     *
     * @param context    上下文
     * @param filePath   文件路径
     * @param createTime 创建时间 <=0时为当前时间 ms
     * @param width      宽度
     * @param height     高度
     */
    public static void insertImageToMediaStore(Context context, String filePath, long createTime, int width, int height) {
        if (!checkFile(filePath))
            return;
        createTime = getTimeWrap(createTime);
        ContentValues values = initCommonContentValues(filePath, createTime);
        values.put(MediaStore.Images.ImageColumns.DATE_TAKEN, createTime);
        values.put(MediaStore.Images.ImageColumns.ORIENTATION, 0);
        values.put(MediaStore.Images.ImageColumns.ORIENTATION, 0);
        if (Build.VERSION.SDK_INT > Build.VERSION_CODES.JELLY_BEAN) {
            if (width > 0) values.put(MediaStore.Images.ImageColumns.WIDTH, 0);
            if (height > 0) values.put(MediaStore.Images.ImageColumns.HEIGHT, 0);
        }
        values.put(MediaStore.MediaColumns.MIME_TYPE, getPhotoMimeType(filePath));
        context.getApplicationContext().getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
    }

    /**
     * 保存到视频到本地,并插入MediaStore以保证相册可以查看到,这是更优化的方法,防止读取的视频获取不到宽高
     *
     * @param context    上下文
     * @param filePath   文件路径
     * @param createTime 创建时间 <=0时为当前时间 ms
     * @param duration   视频长度 ms
     * @param width      宽度
     * @param height     高度
     */
    public static void insertVideoToMediaStore(Context context, String filePath, long createTime, int width, int height, long duration) {
        if (!checkFile(filePath))
            return;
        createTime = getTimeWrap(createTime);
        ContentValues values = initCommonContentValues(filePath, createTime);
        values.put(MediaStore.Video.VideoColumns.DATE_TAKEN, createTime);
        if (duration > 0)
            values.put(MediaStore.Video.VideoColumns.DURATION, duration);
        if (Build.VERSION.SDK_INT > Build.VERSION_CODES.JELLY_BEAN) {
            if (width > 0) values.put(MediaStore.Video.VideoColumns.WIDTH, width);
            if (height > 0) values.put(MediaStore.Video.VideoColumns.HEIGHT, height);
        }
        values.put(MediaStore.MediaColumns.MIME_TYPE, getVideoMimeType(filePath));
        context.getContentResolver().insert(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, values);
    }


    // 是不是系统相册
    private static boolean isSystemDcim(String path) {
        return path.toLowerCase().contains("dcim") || path.toLowerCase().contains("camera");
    }

    // 获取照片的mine_type
    private static String getPhotoMimeType(String path) {
        String lowerPath = path.toLowerCase();
        if (lowerPath.endsWith("jpg") || lowerPath.endsWith("jpeg")) {
            return "image/jpeg";
        } else if (lowerPath.endsWith("png")) {
            return "image/png";
        } else if (lowerPath.endsWith("gif")) {
            return "image/gif";
        }
        return "image/jpeg";
    }

    // 获取video的mine_type,暂时只支持mp4,3gp
    private static String getVideoMimeType(String path) {
        String lowerPath = path.toLowerCase();
        if (lowerPath.endsWith("mp4") || lowerPath.endsWith("mpeg4")) {
            return "video/mp4";
        } else if (lowerPath.endsWith("3gp")) {
            return "video/3gp";
        }
        return "video/mp4";
    }

    // 获得转化后的时间
    private static long getTimeWrap(long time) {
        if (time <= 0) {
            return System.currentTimeMillis();
        }
        return time;
    }

    // 检测文件存在
    private static boolean checkFile(String filePath) {
        boolean result = FileUtil.fileIsExist(filePath);
        Log.e(TAG, "文件不存在 path = " + filePath);
        return result;
    }
}
  • 3
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
在您的新标签中显示来自您的Google相册帐户的个性化照片。 在与您的Google相册帐户相关联的新标签中查看个性化的随机照片,全天为您留下美好的回忆! 每次您打开新标签页时,都会从您的Google相册帐户中看到随机照片。 您可以选择要查看和隐藏的相册,以免令人尴尬的内容突然弹出。 您也可以暂时禁用演示文稿或公共场所的个性化设置。 该产品不是Google制造的,与Google相册没有官方关联。 希望你喜欢! ===================== 0.24中的新增功能错误修复了API请求未返回任何数据时生的意外错误。 ------------------------ 0.23版中的新增功能已修复还修复了必须重新连接帐户两次的错误。 ------------------------ 0.22版中的新功能-添加了从扩展程序内部断开帐户连接的功能-UX更改以符合Google品牌准则-更好的用户体验如果找不到以前的安装,还可以修复必须重新连接帐户两次的错误。 ------------------------ 0.21版中的新功能Google已停用Picasa照片API,因此可以将所有内容切换到Google相册。 这样可以解决有关看到恒定的“正在加载”屏幕或无法更新专辑列表的所有报告问题。 在大多数情况下,体验应该是相同的,但是由于必要的更改,升级后将需要重新连接帐户,并且您将丢失本地配置。 对于那个很抱歉! 其他较小的更改:-照片质量应根据您的浏览器窗口大小自动更新-所有相册链接现在都将您直接带到Google相册,而不是Picasa-修复了当相册照片数量过多时扩展名停止工作的错误-防止创建重复的历史记录条目这是预行版,是对我最终称为1.0版的重大升级。 ------------------------ 0.20版中的新功能-历史记录模式! 现在,滚动查看您看到的最后10张照片。 ------------------------ 0.19版中的新功能-允许切换图像如何适合浏览器(缩放还是包含)-选项UI大修--- --------------------- 0.18版中的新功能-在多个设备上使用扩展程序时的错误修复--------------- --------- 0.17版中的新功能-防止显示常见的视频缩略图类型-添加速率和捐赠按钮---------------------- -版本0.16中的新功能-添加了隐藏/显示相册信息的功能-添加了紧凑的控件模式-修复了导致从相册中删除照片后扩展名中断的错误---------------- -------- 0.15版中的新增功能-添加了使用浏览器控件轻松复制/保存图像的功能------------------------新增功能在0.14版中:-从配置页面临时禁用个性化照片-禁用个性化/未设置个性化设置时,图像加载速度更快-选择要显示照片的更好算法 支持语言:English

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值