iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >Android图像选择之 PictureSelector
  • 950
分享到

Android图像选择之 PictureSelector

androidPictureSelector 2023-12-22 17:12:11 950人浏览 泡泡鱼
摘要

一款针对Android平台下的图片选择器,支持从相册获取图片、视频、音频&拍照,支持裁剪(单图or多图裁剪)、压缩、主题自定义配置等功能,支持动态获取权限&适配Android 5.0+系统的开源图片选择框架。 废话不多。开干!! 添加依赖:

一款针对Android平台下的图片选择器,支持从相册获取图片、视频、音频&拍照,支持裁剪(单图or多图裁剪)、压缩、主题自定义配置等功能,支持动态获取权限&适配Android 5.0+系统的开源图片选择框架

废话不多。开干!!

添加依赖:

    //图片选择器    api 'io.GitHub.lucksiege:pictureselector:v3.11.1'    //图片压缩    api 'io.github.lucksiege:compress:v3.11.1'    //图片裁剪    api 'io.github.lucksiege:ucrop:v3.11.1'    //自定义相机    api 'io.github.lucksiege:camerax:v3.11.1'

实现工具类如下 PictureUtils :

import android.Manifest;import android.content.Context;import android.content.Intent;import android.graphics.Bitmap;import android.graphics.drawable.Drawable;import android.net.Uri;import android.provider.MediaStore;import android.widget.ImageView;import androidx.annotation.NonNull;import androidx.annotation.Nullable;import androidx.fragment.app.Fragment;import com.bumptech.glide.Glide;import com.bumptech.glide.request.target.CustomTarget;import com.bumptech.glide.request.transition.Transition;import com.luck.lib.camerax.SimpleCameraX;import com.luck.picture.lib.basic.PictureSelector;import com.luck.picture.lib.config.PictureMimeType;import com.luck.picture.lib.config.SelectMimeType;import com.luck.picture.lib.engine.CompressFileEngine;import com.luck.picture.lib.engine.CropFileEngine;import com.luck.picture.lib.engine.VideoPlayerEngine;import com.luck.picture.lib.entity.LocalMedia;import com.luck.picture.lib.entity.MediaExtrainfo;import com.luck.picture.lib.interfaces.OnCameraInterceptListener;import com.luck.picture.lib.interfaces.OnExternalPreviewEventListener;import com.luck.picture.lib.interfaces.OnRecordAudioInterceptListener;import com.luck.picture.lib.interfaces.OnResultCallbackListener;import com.luck.picture.lib.permissions.PermissionChecker;import com.luck.picture.lib.permissions.PermissionResultCallback;import com.luck.picture.lib.utils.MediaUtils;import com.luck.picture.lib.utils.ToastUtils;import com.yalantis.ucrop.UCrop;import com.yalantis.ucrop.UCropImageEngine;import java.io.File;import java.util.ArrayList;import top.zibin.luban.Luban;import top.zibin.luban.OnNewCompressListener;public class PictureUtils {        public static void openCamera(Context context, boolean isRotateImage, OnPictureSelectorResultListener onPictureSelectorResultListener) {        openCamera(context, SelectMimeType.ofImage(), isRotateImage, onPictureSelectorResultListener);    }        public static void openVideo(Context context, boolean isRotateImage, OnPictureSelectorResultListener onPictureSelectorResultListener) {        openCamera(context, SelectMimeType.ofVideo(), isRotateImage, onPictureSelectorResultListener);    }        public static void openCamera(Context context, int openCamera, boolean isRotateImage, OnPictureSelectorResultListener onPictureSelectorResultListener) {        PictureSelector.create(context)                .openCamera(openCamera)                .isCameraAroundState(isRotateImage)                .setVideoThumbnailListener(new VideoThumbListener(context))                .setCompressEngine((CompressFileEngine) (context1, source, call) -> Luban.with(context1).load(source).ignoreBy(100)                        .setCompressListener(new OnNewCompressListener() {@Overridepublic void onStart() {}@Overridepublic void onSuccess(String source, File compressFile) {    if (call != null) {        call.onCallback(source, compressFile.getAbsolutePath());    }}@Overridepublic void onError(String source, Throwable e) {    if (call != null) {        call.onCallback(source, null);    }}                        }).launch())                .forResult(new OnResultCallbackListener<LocalMedia>() {                    @Override                    public void onResult(ArrayList<LocalMedia> result) {                        onPictureSelectorResultListener.onResult(result);                    }                    @Override                    public void onCancel() {                    }                });    }        public static void createAvatar(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {        create(mContext, SelectMimeType.ofImage(), selectResult, 1, 1, true, onPictureSelectorResultListener);    }        public static void createImageMin(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {        create(mContext, SelectMimeType.ofImage(), selectResult, 1, 1, false, onPictureSelectorResultListener);    }        public static void createImageMax(Context mContext, int selectMax, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {        create(mContext, SelectMimeType.ofImage(), selectResult, 1, selectMax, false, onPictureSelectorResultListener);    }        public static void createVideo(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {        create(mContext, SelectMimeType.ofVideo(), selectResult, 1, 1, false, onPictureSelectorResultListener);    }        public static void createAudio(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {        create(mContext, SelectMimeType.ofAudio(), selectResult, 1, 1, false, onPictureSelectorResultListener);    }        public static void createPicture(Context mContext, ArrayList<LocalMedia> selectResult, OnPictureSelectorResultListener onPictureSelectorResultListener) {        create(mContext, SelectMimeType.ofAll(), selectResult, 1, 1, false, onPictureSelectorResultListener);    }        public static void create(Context mContext, int selectMimeType, ArrayList<LocalMedia> selectResult, int selectMin, int selectMax, boolean isCrop,  OnPictureSelectorResultListener onPictureSelectorResultListener) {        PictureSelector.create(mContext)                .openGallery(selectMimeType)                .setMaxSelectNum(selectMax)                .setCropEngine(getCropFileEngine(isCrop))                .setMinSelectNum(selectMin)                .setFilterVideoMaxSecond(selectMimeType == SelectMimeType.ofVideo() ? 60 : 60 * 10)                .setFilterVideoMinSecond(5)                .setRecordVideoMaxSecond(selectMimeType == SelectMimeType.ofVideo() ? 60 : 60 * 10)                .setRecordVideoMinSecond(5)                .setFilterMaxFileSize(100 * 1024 * 1024)                .setCameraInterceptListener(new MeOnCameraInterceptListener())                .isFilterSizeDuration(true)                .setSelectedData(selectResult)                .setRecordAudioInterceptListener(new MeOnRecordAudioInterceptListener())                .setCompressEngine((CompressFileEngine) (context, source, call) -> Luban.with(context).load(source).ignoreBy(100)                        .setCompressListener(new OnNewCompressListener() {@Overridepublic void onStart() {}@Overridepublic void onSuccess(String source, File compressFile) {    if (call != null) {        call.onCallback(source, compressFile.getAbsolutePath());    }}@Overridepublic void onError(String source, Throwable e) {    if (call != null) {        call.onCallback(source, null);    }}                        }).launch()).setImageEngine(GlideUtils.createGlideEngine())                .forResult(new OnResultCallbackListener<LocalMedia>() {                    @Override                    public void onResult(ArrayList<LocalMedia> result) {                        for (LocalMedia media : result) {if (media.getWidth() == 0 || media.getHeight() == 0) {    if (PictureMimeType.isHasImage(media.getMimeType())) {        MediaExtraInfo imageExtraInfo = MediaUtils.getImageSize(mContext, media.getPath());        media.setWidth(imageExtraInfo.getWidth());        media.setHeight(imageExtraInfo.getHeight());    } else if (PictureMimeType.isHasVideo(media.getMimeType())) {        MediaExtraInfo videoExtraInfo = MediaUtils.getVideoSize(mContext, media.getPath());        media.setWidth(videoExtraInfo.getWidth());        media.setHeight(videoExtraInfo.getHeight());    }}//LogUtils.e("文件名: " + media.getFileName() + "\n" +//        "是否压缩:" + media.isCompressed() + "\n" +//        "压缩路径:" + media.getCompressPath() + "\n" +//        "初始路径:" + media.getPath() + "\n" +//        "绝对路径:" + media.getRealPath() + "\n" +//        "是否裁剪:" + media.isCut() + "\n" +//        "裁剪路径:" + media.getCutPath() + "\n" +//        "是否开启原图:" + media.isOriginal() + "\n" +//        "原图路径:" + media.getOriginalPath() + "\n" +//        "沙盒路径:" + media.getSandboxPath() + "\n" +//        "水印路径:" + media.getWatermarkPath() + "\n" +//        "视频缩略图:" + media.getVideoThumbnailPath() + "\n" +//        "原始宽高: " + media.getWidth() + "x" + media.getHeight() + "\n" +//        "裁剪宽高: " + media.getCropImageWidth() + "x" + media.getCropImageHeight() + "\n" +//        "文件大小: " + PictureFileUtils.fORMatAccurateUnitFileSize(media.getSize()) + "\n" +//        "文件大小: " + media.getSize() + "\n" +//        "文件时长: " + media.getDuration()//);                        }                        onPictureSelectorResultListener.onResult(result);                    }                    @Override                    public void onCancel() {                    }                });    }        public static void openImage(Context mContext, int position, ArrayList<LocalMedia> localMedia) {        PictureSelector.create(mContext)                .openPreview()                .isHidePreviewDownload(true)                .setImageEngine(GlideUtils.createGlideEngine())                .setExternalPreviewEventListener(new OnExternalPreviewEventListener() {                    @Override                    public void onPreviewDelete(int position) {                    }                    @Override                    public boolean onLongPressDownload(Context context, LocalMedia media) {                        return false;                    }                }).startActivityPreview(position, false, localMedia);    }    public static void openImage(Context mContext, int position, String imageUrl) {        ArrayList<LocalMedia> localMedia = new ArrayList<>();        for (String url : imageUrl.split(",")) {            localMedia.add(LocalMedia.generateHttpAsLocalMedia(url));        }        PictureSelector.create(mContext)                .openPreview()                .setImageEngine(GlideUtils.createGlideEngine())                .setExternalPreviewEventListener(new OnExternalPreviewEventListener() {                    @Override                    public void onPreviewDelete(int position) {                    }                    @Override                    public boolean onLongPressDownload(Context context, LocalMedia media) {                        return false;                    }                }).startActivityPreview(position, false, localMedia);    }        public static void openVideo(Context mContext, int position, ArrayList<LocalMedia> localMedia) {        openVideo(mContext, position, localMedia, null);    }        public static void openVideo(Context mContext, int position, ArrayList<LocalMedia> localMedia, VideoPlayerEngine videoPlayerEngine) {        PictureSelector.create(mContext)                .openPreview()                .setImageEngine(GlideUtils.createGlideEngine())                .setVideoPlayerEngine(videoPlayerEngine)                .isAutoVideoPlay(true)                .setExternalPreviewEventListener(new OnExternalPreviewEventListener() {                    @Override                    public void onPreviewDelete(int position) {                    }                    @Override                    public boolean onLongPressDownload(Context context, LocalMedia media) {                        return false;                    }                }).startActivityPreview(position, false, localMedia);    }        private static ImageFileCropEngine getCropFileEngine(boolean isCrop) {        return isCrop ? new ImageFileCropEngine() : null;    }        private static class ImageFileCropEngine implements CropFileEngine {        @Override        public void onStartCrop(Fragment fragment, Uri srcUri, Uri destinationUri, ArrayList<String> dataSource, int requestCode) {            UCrop.Options options = buildOptions();            UCrop uCrop = UCrop.of(srcUri, destinationUri, dataSource);            uCrop.withOptions(options);            uCrop.setImageEngine(new UCropImageEngine() {                @Override                public void loadImage(Context context, String url, ImageView imageView) {                    Glide.with(context).load(url).override(180, 180).into(imageView);                }                @Override                public void loadImage(Context context, Uri url, int maxWidth, int maxHeight, OnCallbackListener<Bitmap> call) {                    Glide.with(context).asBitmap().load(url).override(maxWidth, maxHeight).into(new CustomTarget<Bitmap>() {    @Override    public void onResourceReady(@NonNull Bitmap resource, @Nullable Transition<? super Bitmap> transition) {        if (call != null) {            call.onCall(resource);        }    }    @Override    public void onLoadCleared(@Nullable Drawable placeholder) {        if (call != null) {            call.onCall(null);        }    }});                }            });            uCrop.start(fragment.requiReactivity(), fragment, requestCode);        }    }        private static UCrop.Options buildOptions() {        UCrop.Options options = new UCrop.Options();        options.setHideBottomControls(true);        options.setFreeStyleCropEnabled(true);        options.setShowCropFrame(true);        options.setShowCropGrid(false);        options.setCircleDimmedLayer(false);        options.withAspectRatio(1, 1);        options.isCropDragSmoothToCenter(false);        options.setMaxScaleMultiplier(100);        return options;    }    public interface OnPictureSelectorResultListener {        void onResult(ArrayList<LocalMedia> result);    }        private static class MeOnCameraInterceptListener implements OnCameraInterceptListener {        @Override        public void openCamera(Fragment fragment, int cameraMode, int requestCode) {            SimpleCameraX camera = SimpleCameraX.of();            camera.isAutoRotation(true);            camera.setCameraMode(cameraMode);            camera.setVideoFrameRate(50);            camera.setVideoBitRate(5 * 1024 * 1024);            camera.isDisplayRecordChangeTime(true);            camera.isManualFocusCameraPreview(true);            camera.isZoomCameraPreview(true);            camera.setImageEngine((context, url, imageView) -> Glide.with(context).load(url).into(imageView));            camera.start(fragment.requireActivity(), fragment, requestCode);        }    }        private static class MeOnRecordAudioInterceptListener implements OnRecordAudioInterceptListener {        @Override        public void onRecordAudio(Fragment fragment, int requestCode) {            String[] recordAudio = {Manifest.permission.RECORD_AUDIO};            if (PermissionChecker.isCheckSelfPermission(fragment.getContext(), recordAudio)) {                startRecordSoundAction(fragment, requestCode);            } else {                PermissionChecker.getInstance().requestPermissions(fragment,                        new String[]{Manifest.permission.RECORD_AUDIO}, new PermissionResultCallback() {@Overridepublic void onGranted() {    startRecordSoundAction(fragment, requestCode);}@Overridepublic void onDenied() {}                        });            }        }    }        private static void startRecordSoundAction(Fragment fragment, int requestCode) {        Intent recordAudioIntent = new Intent(MediaStore.Audio.Media.RECORD_SOUND_ACTION);        if (recordAudioIntent.resolveActivity(fragment.requireActivity().getPackageManager()) != null) {            fragment.startActivityForResult(recordAudioIntent, requestCode);        } else {            ToastUtils.showToast(fragment.getContext(), "The system is missing a recording component");        }    }}

上文监听 VideoThumbListener:

import android.content.Context;import android.graphics.Bitmap;import android.graphics.drawable.Drawable;import androidx.annotation.NonNull;import androidx.annotation.Nullable;import com.bumptech.glide.Glide;import com.bumptech.glide.request.target.CustomTarget;import com.bumptech.glide.request.transition.Transition;import com.luck.picture.lib.interfaces.OnKeyValueResultCallbackListener;import com.luck.picture.lib.interfaces.OnVideoThumbnailEventListener;import com.luck.picture.lib.utils.PictureFileUtils;import java.io.ByteArrayOutputStream;import java.io.File;import java.io.FileOutputStream;import java.io.IOException;public class VideoThumbListener implements OnVideoThumbnailEventListener {    private Context context;    public VideoThumbListener(Context context) {        this.context = context;    }    @Override    public void onVideoThumbnail(Context context, String videoPath, OnKeyValueResultCallbackListener call) {        Glide.with(context).asBitmap().sizeMultiplier(0.6F).load(videoPath).into(new CustomTarget<Bitmap>() {            @Override            public void onResourceReady(@NonNull Bitmap resource, @Nullable Transition<? super Bitmap> transition) {                ByteArrayOutputStream stream = new ByteArrayOutputStream();                resource.compress(Bitmap.CompressFormat.JPEG, 60, stream);                FileOutputStream fos = null;                String result = null;                try {                    File targetFile = new File(getVideoThumbnailDir(), "thumbnails_" + System.currentTimeMillis() + ".jpg");                    fos = new FileOutputStream(targetFile);                    fos.write(stream.toByteArray());                    fos.flush();                    result = targetFile.getAbsolutePath();                } catch (IOException e) {                    e.printStackTrace();                } finally {                    PictureFileUtils.close(fos);                    PictureFileUtils.close(stream);                }                if (call != null) {                    call.onCallback(videoPath, result);                }            }            @Override            public void onLoadCleared(@Nullable Drawable placeholder) {                if (call != null) {                    call.onCallback(videoPath, "");                }            }        });    }    private String getVideoThumbnailDir() {        File externalFilesDir = context.getExternalFilesDir("");        File customFile = new File(externalFilesDir.getAbsolutePath(), "Thumbnail");        if (!customFile.exists()) {            customFile.mkdirs();        }        return customFile.getAbsolutePath() + File.separator;    }}

使用:

使用摄像头拍照:

PictureUtils.openCamera(requireActivity(), false) {val localMedia = it[0]               }

选择单张图片

  PictureUtils.createImageMin(this, ArrayList()) {val localMedia = it[0]              }

选择多张图片

PictureUtils.createImageMax(this, 9, arrayListOf()) { result: ArrayList<LocalMedia> ->   ...                         }

简单使用如上。
END

来源地址:https://blog.csdn.net/lixinxiaos/article/details/132899394

--结束END--

本文标题: Android图像选择之 PictureSelector

本文链接: https://www.lsjlt.com/news/551458.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Android图像选择之 PictureSelector
    一款针对Android平台下的图片选择器,支持从相册获取图片、视频、音频&拍照,支持裁剪(单图or多图裁剪)、压缩、主题自定义配置等功能,支持动态获取权限&适配Android 5.0+系统的开源图片选择框架。 废话不多。开干!! 添加依赖:...
    99+
    2023-12-22
    android PictureSelector
  • Android选择与上传图片之PictureSelector教程
    效果图: 【注意】Demo已更新到最新版本,并稍作调整。 之前出过一篇 Android选择与上传图片之ImagePicker教程,这个是okgo作者出的,就一般需求来讲是够了...
    99+
    2024-04-02
  • Android选择与上传图片之Matisse教程
    效果图: 就目前效果图来看,好像也没什么毛病哈,其实我这个集成的过程是有点坎坷的。 而且,功能也不算是很齐全吧…主要体现在以下几个点 没有回调之后的预览 选择之...
    99+
    2024-04-02
  • Android选择与上传图片之ImagePicker教程
    效果图: 后来又出了两篇,也可以看一下 Android选择与上传图片之PictureSelector教程 Android选择与上传图片之Matisse教程 添加依赖:...
    99+
    2024-04-02
  • Android实现底部图片选择Dialog
    业务需要选择弹出对话框,然后点击选择图片。网上已经有了很多,不过感觉写的有点乱。自己这里总结一下,有需要开发者可以按照如下步骤直接使用即可。1.效果图如下点击选择照相后,弹出如下选择对话框: 2. Dialog实现布局<LinearL...
    99+
    2023-05-30
    android 图片选择 dialog
  • Android PicSelector图片选择器小功能
    本文实例为大家分享了Android实现图片选择器小功能的具体代码,供大家参考,具体内容如下 效果预览 实现 需要用到的库 compile 'com.squareup.picas...
    99+
    2024-04-02
  • Android 之 Button (按钮)与 ImageButton (图像按钮)
    本节引言: 今天给大家介绍的Android基本控件中的两个按钮控件,Button普通按钮和ImageButton图像按钮; 其实ImageButton和Button的用法基本类似,至于与图片相关的则和后面ImageView相同,所以本节...
    99+
    2023-09-24
    android ui
  • android选择系统映像时要注意什么
    选择Android系统映像时需要注意以下几点:1. 版本兼容性:确保选择的系统映像与你的设备兼容。不同设备通常需要不同的系统版本。如...
    99+
    2023-10-11
    android
  • Android入门之日历选择与时间选择组件的使用
    目录介绍DatePicker有两种StyleTimePicker同样也有两种Style课程目标主界面代码后台交互代码运行效果介绍 DatePicker日期选择器,自带spinner和...
    99+
    2022-11-16
    Android日历选择组件 Android 时间选择组件 Android 日历 时间组件
  • Android 开发中如何模仿一个微信拍摄和图像选择界面
    这篇文章将为大家详细讲解有关Android 开发中如何模仿一个微信拍摄和图像选择界面,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。 插件运行后的画面如下:下面这张图对图像进行筛选,...
    99+
    2023-05-31
    android roi
  • Android中如何使用ImageEditContainer图片选择器
    Android中如何使用ImageEditContainer图片选择器,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。1. 简介ImageEditButton 和 Image...
    99+
    2023-05-30
    android
  • Python图像处理之图像拼接
    目录一、前言二、特征点匹配三、匹配错误的特征点干扰四、消除干扰五、RANSAC进行图像匹配六、总结一、前言 图像拼接技术就是将数张有重叠部分的图像(可能是不同时间、不同视角或者不同传...
    99+
    2024-04-02
  • Android图像视图ImageView实现图像拉伸效果
    本文实例为大家分享了Android图像视图ImageView实现图像拉伸效果的具体代码,供大家参考,具体内容如下 在layout调整属性src指定图形来源。Activity中setS...
    99+
    2024-04-02
  • 排序算法图解之Java选择排序
    目录1.选择排序简介2.图解选择排序算法3.选择排序代码实现1.选择排序简介 选择排序(Selection sort)是一种简单直观的排序算法。它的工作原理是:第一次从待排序的数据元...
    99+
    2022-11-13
    Java选择排序 Java 排序
  • OpenCV图像处理之图像拼接详解
    目录图像拼接技术一、需求分析二、具体步骤三、代码实现图像拼接技术 一、需求分析 将下面两张图像进行拼接 拼接得到一张完整的图像 二、具体步骤 1.选择特征点 //1、选...
    99+
    2022-11-13
    OpenCV 图像处理 图像拼接 OpenCV 图像拼接 OpenCV 图像处理
  • Python图像处理之模糊图像判断
    目录上期回顾采用Laplace算子的原因实现的效果图片素材代码的展示与讲解效果展示项目资源上期回顾 上一次的图像清晰度评价没有成功,主要的原因是那几张图像清晰度评价函数都实际都采用了...
    99+
    2022-12-08
    Python模糊图像判断 Python模糊图像 Python模糊 判断
  • Android图像处理之绘制圆形、三角形及扇形的头像
    前言相信大家在Android日常开发中,绘制圆形和绘制图片都是很容易的事情,但是绘制圆形图片就有点难倒人了。以前为了偷懒就直接去github上找一个开源项目,后来才发现绘制圆形图片其实也是很简单的事。绘制圆形图片也需要两个步骤:绘制圆形和绘...
    99+
    2023-05-31
    android 扇形 圆形头像
  • android中怎么实现在相册中选择图片
    这期内容当中小编将会给大家带来有关android中怎么实现在相册中选择图片,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。首先在 activity_main.xml 文件中增加一个 Button,用来触发从...
    99+
    2023-05-30
  • 云服务器镜像选择
    云服务器镜像是一种在虚拟机上运行的文件,可以用于存储和备份数据。选择一个镜像时,你需要考虑以下几个因素: 性能:云服务器镜像通常具有更高的数据处理能力和容错能力,因此你需要确保它可以在需要的时候处理高负载。 扩展性:选择镜像时,你应该考...
    99+
    2023-10-27
    镜像 服务器
  • Android实现QQ图片说说照片选择效果
    本文实例为大家分享了Android实现QQ图片说说照片选择的具体代码,供大家参考,具体内容如下效果展示布局文件布局是很简单的,一个GridView,直接上布局:layout/activity_add_photo.xml<?xm...
    99+
    2023-05-30
    android 照片选择
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作