美文网首页
Android 实现类似于QQ空间选择并展示本地图片

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

作者: 风二叽 | 来源:发表于2018-10-11 17:20 被阅读0次

           新人刚刚接触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就实现了,收工。

    相关文章

      网友评论

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

          本文链接:https://www.haomeiwen.com/subject/gombaftx.html