【Android开发】找乐,一个笑话App的制作过程记录

22 篇文章 1 订阅
7 篇文章 0 订阅

缘起

想做一个笑话App的原因是因为在知乎上看过一个帖子,做Android可以有哪些数据可以练手,里面推荐了几个数据开放平台。在这些平台中无一不是有公共的笑话接口,当时心想这个可以拿来练手啊,还挺有意思的,估计还能积累一点用户。

碰巧(真的好巧)在Github中遇到了一个MVP设计模式的框架Beam,作者Jude95有一个笑话仓库————Joy(豆逼),就是一个做笑话的!更巧的是用到的接口也是我在关注的接口,心想不如改造一下吧,做个升级版,自己也可以在这个中学到别人是怎么写App的。后来发现这是一个非常正确的决定。

雏形

因为是基于别人的改进,所以在写之前就已经有雏形,当然这个雏形不是很完善,这恰恰给了我修改的空间。在获得作者的修改同意后,我就进一步研究这个利用MVP框架书写的App。未修改之前:

首先,豆逼只能查看段子和查看图片,我认为基本的复制文本和查看大图以及下载图片,这些都没有。作者只是用这个仓库来说明MVP模式的,所以只做了最基本的功能。作者也说,笑话连个id都没有,点赞、评论什么的根本没法做。那好,我就把我认为的文本复制和图片相关的做一下吧。

研究

MVP模式在这个项目之前我研究很少,只是听说,但是这个项目完全给我耳目一新的感觉,MVP对Android来说实在是太有用了!关于MVP我以后想仔细写个帖子研究一下,这里只想说明MVP使Android项目层次分明,代码结构简单,复用性高。参考作者的Beam

这个项目用了很棒的一个开源控件,也是项目作者自己的控件EasyRecyclerView,这个控件对我来说相见恨晚。线性布局仿EasyRecyclerView已经实现了下拉刷新,上拉加载更多,错误提示等,简直把项目开发中可能遇到的坑都给做好了,我之前只能一个一个的去实现这些功能!为什么没有早早的用上这个控件!

其他的没有重大的惊喜,但是项目总体感觉代码量很少,很精简。如果是我完成相同的功能的App,可能需要3倍的代码才能实现。

改进

查看大图

首先实现点击查看大图的功能。

PhotoView这个控件也是之前不久在Github中遇到的,使用的时候没想到竟然这么容易!只需要在xml中声明一个PhotoView,基本的放大、缩小、手势识别都有了!太方便!可能也是北邮人论坛官方客户端采用的一个查看大图的工具。

在java文件加载图片时则与ImageView完全相同,这个不在赘述。

还有一个拓展的地方是,单击图片返回(= = 一般都有吧?)。这个需要根据PhotoView的官方说明,使用Attacher来管理点击事件,经过我测试,貌似直接声明ImageView的点击是不会有效果的。

图片下载

这个App采用的是Glide加载网络图片,而Glide并没有直接的下载存储的方法,只有自己拓展,耽误了些功夫。

直接分享一段图片下载和通知图库的代码吧。

    public void saveImage(String imageUrl) {

        String[] names = new String[0];
        if (imageUrl != null) {
            names = imageUrl.split("/");
        }
        String imageName = names[names.length - 1];
        Glide
                .with(getView())
                .load(imageUrl)
                .asBitmap()
                .toBytes(Bitmap.CompressFormat.JPEG, 100)
                .into(new SimpleTarget<byte[]>() {
                    @Override
                    public void onResourceReady(final byte[] resource, GlideAnimation<? super byte[]> glideAnimation) {
                        new AsyncTask<Void, Void, Void>() {
                            @Override
                            protected Void doInBackground(Void... params) {

                                if (ImageStorage.checkifImageExists(imageName)) {
                                    Snackbar.make(getView().fab, "图片已存在", Snackbar
                                            .LENGTH_LONG)
                                            .setAction("Action", null).show();
                                    return null;
                                }
                                String path = Environment.getExternalStorageDirectory().toString();
                                JUtils.Log("path", path);

                                Bitmap bitmap = BitmapFactory.decodeByteArray(resource, 0, resource.length);
                                JUtils.Log("imageName", imageName);

                                ImageStorage.saveToSdCard(getView(), bitmap, imageName);

                                Snackbar.make(getView().fab, "图片已下载", Snackbar.LENGTH_LONG)
                                        .setAction("Action", null).show();

                                return null;
                            }
                        }.execute();
                    }
                });
    }

其中ImageStorage.java:

    public class ImageStorage {

        public static String saveToSdCard(Context context, Bitmap bitmap, String filename) {

            String stored = null;

            File sdcard = Environment.getExternalStorageDirectory();

            File folder = new File(sdcard.getAbsoluteFile(), "FindJoy");//the dot makes this directory hidden to
            // the
            // user
            folder.mkdir();
            File file = new File(folder.getAbsoluteFile(), filename + ".jpg");
            if (file.exists())
                return stored;

            try {
                FileOutputStream out = new FileOutputStream(file);
                bitmap.compress(Bitmap.CompressFormat.JPEG, 100, out);
                out.flush();
                out.close();
                stored = "success";
                JUtils.Log("stored", stored);
            } catch (Exception e) {
                e.printStackTrace();
            }
            // 其次把文件插入到系统图库
            try {
                MediaStore.Images.Media.insertImage(context.getContentResolver(),
                        file.getAbsolutePath(), filename, null);
            } catch (FileNotFoundException e) {
                e.printStackTrace();
            }
            // 最后通知图库更新
            context.sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.parse("file://" + file
                    .getAbsolutePath())));
            return stored;
        }

        public static File getImage(String imagename) {

            File mediaImage = null;
            try {
                String root = Environment.getExternalStorageDirectory().toString();
                File myDir = new File(root);
                if (!myDir.exists())
                    return null;

                mediaImage = new File(myDir.getPath() + "/FindJoy/" + imagename);
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            return mediaImage;
        }

        public static boolean checkifImageExists(String imagename) {
            Bitmap b = null;
            File file = ImageStorage.getImage("/" +
                    imagename + "" +
                    ".jpg");
            String path = file.getAbsolutePath();

            if (path != null)
                b = BitmapFactory.decodeFile(path);

            if (b == null || b.equals("")) {
                return false;
            }
            return true;
        }
    }

为什么之前我试了很久但是一直发现图库没有图片呢?一直以为是自己的图片没有存储下来,后来用图库的查看文件夹的方式发现了FindJoy目录。

原来是需要通知图库更新,否则图片不会再图库中显示。具体请看上面代码。

复制段子

这个本身是不麻烦的,出现问题的地方在于,这个MVP框架中怎么对这个List加上OnItemClilkListner。本身我就不很熟,这个地方犯了不少错误,我怎么没想到看EasyRecyclerView的官方说明呢?

解决方法是在TextViewHolder中的itemView加上:

    itemView.setOnClickListener(view ->
            new MaterialDialog.Builder(getContext())
                    .title(R.string.select)
                    .content(R.string.copy)
                    .positiveText(R.string.agree)
                    .negativeText(R.string.disagree)
                    .onPositive((dialog, which) -> {
                        // Gets a handle to the clipboard service.
                        ClipboardManager clipboard = (ClipboardManager) getContext().
                                getSystemService(Context.CLIPBOARD_SERVICE);
                        // Creates a new text clip to put on the clipboard
                        ClipData clip = ClipData.newPlainText("joy", data.getText());
                        // Set the clipboard's primary clip.
                        clipboard.setPrimaryClip(clip);
                        Snackbar.make(itemView, "已将该段子复制到粘贴板", Snackbar.LENGTH_SHORT).show();
                    })
                    .show()
    );

官方库还有可以设置EasyRecyclerView的监听的方法,效果是一样的。

友盟统计

友盟统计可能是我自己往外发包的一个必选的项了,因为要知道App的使用情况啊。这次发现友盟统计比以前好用多了,jar包也放到了jCenter()仓库,非常方便了。

这里要赞一下这个MVP库的好处了,竟然可以让所有的Activity的生命周期都调用同一段代码来实现友盟统计中要求的所有Actvity的OnResume()和OnPause()方法中都调用统计方法。

实现是通过一个顶级管理类MyActivityLifeCycleDelegate继承ActivityLifeCycleDelegate,在里面设置友盟统计的方法。

    public class MyActivityLifeCycleDelegate extends ActivityLifeCycleDelegate {
        public MyActivityLifeCycleDelegate(Activity act) {
            super(act);
        }

        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            JUtils.Log("onCreate" + getActivity().getClass().getName());
        }

        @Override
        protected void onPause() {
            super.onPause();
            JUtils.Log("onPause");
            MobclickAgent.onPause(getActivity());
        }

        @Override
        protected void onResume() {
            super.onResume();
            JUtils.Log("onResume");
            MobclickAgent.onResume(getActivity());
        }
    }

然后在App的Application中

Beam.setActivityLifeCycleDelegateProvider(MyActivityLifeCycleDelegate::new);

上面这行代码是IDE自己简化的,好高端啊,竟然有点不明白是怎么回事了。。)

哦,对了,不要忘记在Manifest中声明友盟的appkey。嗯。统计就集成好了。

自动更新

同样是友盟的服务,我也以为只是几分钟的事情就搞定了,可是因为自己的问题,耽误了一段时间,竟然还想着把这个锅扔给友盟。好吧,我错了。

这个和统计不一样的是需要手动下载包放到项目当中,其中包括了一个.so文件。

因为在app的gradle中声明了这句:

    compile fileTree(include: ['*.jar'], dir: 'libs')

我就以为万事大吉了。事实上我开启了友盟的debug模式才看了出来是我的.so没有加载进去。

嗯,jni应该这么声明,我给忘了:

    sourceSets {
        main {
            jniLibs.srcDirs = ['libs']
        }
    }

这样.so文件就能加载进去了。而友盟自动更新只需要在MainActivity中写一句代码:

    UmengUpdateAgent.update(this);

很酷对不对?

自动更新是根据app versionCode来判断的,更新的时候注意修改。

App截图

这些功能做完之后我修改了一下配色,最终效果大体如图,部分功能未截图。

应用市场

嗯,这些都实现了之后就上线应用商店了,主要有这几个:

可以扫码下载:

应用宝下载:

豌豆荚下载:

Fir.im下载:

尽量不要用Fir,因为Fir没有直观的下载数目统计,嗯。尽量通过正规应用商店吧。

下载这事还得大家捧个场。

结语

虽然是一个非常简单的App,但是却包含着非常多的心思在里面,而且尝试新的东西的时候可以学到不少东西,这个是值得肯定的。毕竟我现在有种想把之前的App都揉碎重新来写的冲动,毕竟抵挡不住 MVP + Material Design的双重诱惑啊!

Android 开发还有很长的路要走。

本项目已经完全开源,代码在:https://github.com/fuxuemingzhu/FindJoy,欢迎Star和Fork.

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值