Android 实现类似于QQ空间选择并展示本地图片

       新人刚刚接触android,最近由于项目需求,需要实现类似于QQ空间里的发说说功能,大家都知道说说能发本地图片,对于刚刚入坑的新人自己实现还是花点时间的,于是上网查了下参考了许多网上大神的思路,当然实战中还是碰到了很多问题,自己做了些修改,终于做出了一个基本功能的版本。放在这里权当做个笔记,这里只实现了最简单的功能,要用到实际还需要添加很多东西哈,这里只做了读取并展示,上传以后再加。这里都是自己新入安卓的探索过程,漏洞百出,不喜勿喷哈~

先放个效果图吧


选择前
选择后

好了下面记录自己的思路。

首先需要实现图片的展示,这里用Gridview实现,图片写死了每行放四张。。初始demo以后再优化。防止图片选择过多无限往下占用屏幕影响体验,Gridview最多放20张图片,显示两行。下面是activity里Gridview的布局

    android:id="@+id/gridview"

    android:layout_width="match_parent"

    android:layout_height="wrap_content"

    android:numColumns="4"

    />

然后再activity代码里设置gridview的大小(xml里wrap_content实际只显示一行,最简单的思路是调整高度实现显示行数)

//设置gridview跟屏幕等宽,高度为两个图片的长度,这样就显示两行

WindowManager wm = (WindowManager)getSystemService(Context.WINDOW_SERVICE);

DisplayMetrics dm = new DisplayMetrics();

wm.getDefaultDisplay().getMetrics(dm);

int width = dm.widthPixels;

int widthSingle = (width-50)/4;

ViewGroup.LayoutParams lp = new LinearLayout.LayoutParams(width, widthSingle*2 + 20);

gridview.setLayoutParams(lp);

先写Gridview的adapter

public class ItemAdapter extends BaseAdapter {

    private List<String> list = new ArrayList<String>();//图片url列表

    private Context context;

    private static int MAX_SINGLE_LINE = 4;//每行显示4张图片

    public ItemAdapter (List<String> list,Context context){

        this.list = list;

        this.context = context;

    }

    @Override

    public int getCount() {

        return list.size();

    }

    @Override

    public Object getItem(int position) {

        return null;

    }

    @Override

    public long getItemId(int position) {

        return 0;

    }

    @Override

    public View getView(int position, View convertView, ViewGroup parent) {

        ViewHolder viewholder = null;

        if (convertView == null) {

            convertView = View.inflate(context, R.layout.layout_image_item, null);

            viewholder = new ViewHolder();

            viewholder.iamge = (ImageView) convertView.findViewById(R.id.photo);

            convertView.setTag(viewholder);

        } else {

            viewholder = (ViewHolder) convertView.getTag();

        }

        //设置图片长宽为屏幕宽度四分之一

        WindowManager wm = (WindowManager) ActivityUtil.findActivity(context).getSystemService(Context.WINDOW_SERVICE);

        DisplayMetrics dm = new DisplayMetrics();

        wm.getDefaultDisplay().getMetrics(dm);

        int width = dm.widthPixels;

        int widthSingle = (width-MAX_SINGLE_LINE;

        LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(widthSingle, widthSingle);

        lp.setMargins(5, 0, 0, 0);

        if (list.get(position).equals("default")) {  //添加默认图片

            viewholder.iamge.setLayoutParams(lp);

            viewholder.iamge.setImageResource(R.mipmap.add_photo);

        }

        else {

            Bitmap loacalBitmap = LoadUrl(list.get(position));

            if (loacalBitmap != null) {

                viewholder.iamge.setLayoutParams(lp);

                viewholder.iamge.setImageBitmap(loacalBitmap);

            }

        }

        return convertView;

    }

    //读取图片url

    public static Bitmap LoadUrl(String url) {

        try {

            FileInputStream fis = new FileInputStream(url);

            return BitmapFactory.decodeStream(fis);

        } catch (FileNotFoundException e) {

            e.printStackTrace();

            return null;

        }

    }

    class ViewHolder {

        ImageView iamge;

    }

}

下面是activity里Gridview的布局

  android:id="@+id/gridview"

    android:layout_width="match_parent"

    android:layout_height="wrap_content"

    android:numColumns="4"

    />

然后再activity代码里设置gridview,findViewById就省略了

private GridView gridview;

private ItemAdapter itemAdapter;

 private List<String> list = new ArrayList<String>();//存储选中图片的url

private final int MAX_PHOTO = 21;//最多20张图

private static final int MY_PERMISSIONS_WRITE_EXTERNAL_STORAGE = 1; private static final int MY_PERMISSIONS_READ_EXTERNAL_STORAGE = 2;

onCreate里

// xml里设置wrap_content实际只显示一行,最简单的思路是调整高度实现显示行数

//设置gridview跟屏幕等宽,高度为两个图片的长度,这样就显示两行

WindowManager wm = (WindowManager)getSystemService(Context.WINDOW_SERVICE);

DisplayMetrics dm = new DisplayMetrics();

wm.getDefaultDisplay().getMetrics(dm);

int width = dm.widthPixels;

int widthSingle = (width-50)/4;

ViewGroup.LayoutParams lp = new LinearLayout.LayoutParams(width, widthSingle*2 + 20);

gridview.setLayoutParams(lp);

//添加默认图片

list.add("default");

itemAdapter = new ItemAdapter(list,context);

gridview.setAdapter(photoAdapter);

然后设置监听,点击默认图片进入相册选择图片,进入相册获取图片url的方法:

/**

* 进入相册

*/

private void JumpToDCIM(){

Intent intent =new Intent(Intent.ACTION_PICK,

            android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);

    startActivityForResult(intent, 0);

}

选择图片后的回调,获取url

/**

    * 进入相册后选择图片的回调

    * @param requestCode

    * @param resultCode

    * @param data

    */

    @Override

    protected void onActivityResult(int requestCode, int resultCode, Intent data) {

        super.onActivityResult(requestCode, resultCode, data);

        if(data == null){

            return;

        }

        if (requestCode == 0) {

            try {

                Uri originalUri = data.getData();

                ContentResolver testcr = getContentResolver();

                Cursor cursor = testcr.query(originalUri, new String[] { MediaStore.Images.Media.DATA }, null, null, null);

                if(infoList.size() == MAX_PHOTO){

                    removeItem();

                    refreshAdapter();

                    return;

                }

                if (null == cur) {

                    return;

                }

                removeItem();

                for ( cursor.moveToFirst(); ! cursor.isAfterLast(); cursor.moveToNext()) {

                    int dataColumn =  curso.getColumnIndex(MediaStore.Images.Media.DATA);

                    String image_path = cursor.getString(dataColumn);

                    infoList.add(image_path);

                }

                infoList.add("default");

                refreshAdapter();

            } catch (Exception e) {

                System.out.println(e.getMessage());

            }

        }

    }

refreshAdapter方法:

/**

    * 刷新视图

    */

    private void refreshAdapter(){

        if(infoList == null){

            infoList = new ArrayList<String>();

        }

        if(photoAdapter == null){

            photoAdapter = new PhotoAdapter(infoList,mContext);

        }

        if(infoList.size() == MAX_PIC){

            infoList.remove(infoList.size() - 1);

        }

        photoAdapter.notifyDataSetChanged();

    }

removeItem方法:

/**

* 达到最大图片数删除默认图片,不能添加

*/

private void removeItem() {

if(infoList.size() -1 !=MAX_PIC){

if(infoList.size() !=0){//删除默认图片

            infoList.remove(infoList.size() -1);

        }

}

}

好了这样获取相册里图片url的方法写好了,我们在Gridview的监听里判断只要点击的是默认图片就调用这个JumpToDCIM方法。

然后发现并不好使,选择完图片后啥也没有增加。。。。。。

好吧继续调试,打断点试试,发现JumpToDCIM确实获取到了图片的url,问题出现在adapter的LoadUrl(String url)方法里,读取url时报了permission denied异常。上网查了下这是什么东东,发现是没有读取权限,好吧那就添加权限。

我们在进入相册之前先设置读写权限,在AndroidManifest.xml里添加权限:

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

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

实战中发现Android6.0以后AndroidManifest.xml里的设置并不管用,坑。。。那就写个方法申请权限好了。

先写检查是否有权限的方法:

public static boolean checkPermission(Context context, String permissionStr){

        boolean result = true;

        if(Build.VERSION.SDK_INT >= 23){

            result = (context.checkSelfPermission(permissionStr) == PackageManager.PERMISSION_GRANTED);

        }

        else{

            result = (PermissionChecker.checkSelfPermission(context,permissionStr) == PackageManager.PERMISSION_GRANTED);

        }

        return result;

    }

然后是请求权限的方法,调用该方法后系统会弹出弹窗确认:

@RequiresApi(api = 23)

    public static void requestPermission(Activity activity, String[] permissionStrs, int requestCode){

        activity.requestPermissions(permissionStrs,requestCode);

    }

请求权限的回调,这里需要开头对permissions,grantResults进行非空判断,不然请求权限的弹窗出现时在用户选择之前就会回调,permissions,grantResults此时都为空就会崩溃。

@Override

    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults){

        if( permissions.length > 0 && grantResults.length > 0

            if (requestCode == MY_PERMISSIONS_READ_EXTERNAL_STORAGE) {

                if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {

                    JumpToDCIM();

                } else {

                    // Permission Denied

                    Toast.makeText(mContext, "用户拒绝添加读取权限", Toast.LENGTH_SHORT).show();

                }

            }

            if (requestCode == MY_PERMISSIONS_WRITE_EXTERNAL_STORAGE) {

                if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {

                    JumpToDCIM ();

                } else {

                    // Permission Denied

                    Toast.makeText(mContext, "用户拒绝添加写入权限", Toast.LENGTH_SHORT).show();

                }

            }

          }

        super.onRequestPermissionsResult(requestCode, permissions, grantResults);

    }

好了现在可以在Gridview里添加监听读取相册了,代码如下。

/**

        * 如果是默认图片则点击进入相册添加,进入前判断是否有读写权限,没有的话申请权限(Android 6.0/API 23以上时AndroidManifest里添加权限无效需要申请权限)

        */

        gridview.setOnItemClickListener(new AdapterView.OnItemClickListener() {

            @Override

            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {

                if(infoList.get(position).equals("default")){

                    if(checkPermission(mContext, Manifest.permission.READ_EXTERNAL_STORAGE) && checkPermission(mContext,Manifest.permission.WRITE_EXTERNAL_STORAGE)) {

                        JumpToDCIM ();

                    }

                    else {

                        if(Build.VERSION.SDK_INT >= 23) {

                            requestPermission(AddDiaryActivity.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, MY_PERMISSIONS_READ_EXTERNAL_STORAGE);

                            requestPermission(AddDiaryActivity.this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, MY_PERMISSIONS_WRITE_EXTERNAL_STORAGE);

                        }

                    }

                }

            }

        });

OK,一个最最简单功能的demo就实现了,收工。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,035评论 25 707
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,711评论 2 59
  • 听说简书不错,来看看~
    隔壁家的烧烤味很nice阅读 271评论 0 4
  • 01 生在职场, 命不由己啊!尤其碰上傻逼老板,简直心塞到生无可恋! 最近的小娜就陷入要不要辞职的困扰中........
    苏唐苏阅读 394评论 2 4
  • 山脚下,石径旁,朵朵小花在开放 淡淡飘幽香 你上山,我下山,过客匆匆肩擦肩 花开笑依然 注:读秋水饮马陌路有感
    细雨蒙蒙A阅读 254评论 15 14