使用Retrofit+RxJava+MVP打造一款MaterialDesign风格的APP

为了熟悉使用一些开源框架,便决定利用业余时间写一个APP来熟悉这些框架的使用。提前踩一踩坑,方便以后在公司的项目中使用。使用的接口是聚合数据的和干货集中营的,非常感谢。

效果图

用到的主流框架

  • 首页侧滑栏使用DrawerLayout+NavigationView实现的
  • 使用Realm数据库实现本地收藏
  • 使用Retrofit+RxJava+RxAndroid实现网络请求,并对返回结果进行了简单的封装
  • 对RecyclerView的Adapter和ViewHolder进行封装,实现了上拉加载
  • 使用CoordinatorLayout+AppBarLayout+CollapsingToolbarLayout实现了炫酷的滑动动画
  • 使用Glide实现了图片的加载
  • 使用PhotoView实现了图片的缩放
  • 日历使用开源的material-calendarview
  • 实现了SwipeRefreshLayout首次进入自动刷新

一、使用DrawerLayout+NavigationView实现侧滑栏


  1. <?xml version="1.0" encoding="utf-8"?><android.support.v4.widget.DrawerLayout 
  2.     xmlns:android="http://schemas.android.com/apk/res/android" 
  3.     xmlns:app="http://schemas.android.com/apk/res-auto" 
  4.     android:id="@+id/drawerLayout" 
  5.     android:layout_width="match_parent" 
  6.     android:layout_height="match_parent"> 
  7.  
  8.     <LinearLayout 
  9.         android:layout_width="match_parent" 
  10.         android:layout_height="match_parent" 
  11.         android:orientation="vertical"> 
  12.  
  13.        <android.support.v7.widget.Toolbar  
  14.             android:id="@+id/toolbar" 
  15.             android:layout_width="match_parent" 
  16.             android:layout_height="wrap_content" 
  17.             app:titleTextColor="@android:color/white" /> 
  18.  
  19.         <FrameLayout 
  20.             android:id="@+id/fl_main" 
  21.             android:layout_width="match_parent" 
  22.             android:layout_height="match_parent"></FrameLayout> 
  23.     </LinearLayout> 
  24.  
  25.     <android.support.design.widget.NavigationView 
  26.         android:id="@+id/navigation" 
  27.         android:layout_width="match_parent" 
  28.         android:layout_height="match_parent" 
  29.         android:layout_gravity="start" 
  30.         android:fitsSystemWindows="true" 
  31.         app:headerLayout="@layout/drawer_header" 
  32.         app:menu="@menu/drawer_menu"> 
  33.     </android.support.design.widget.NavigationView></android.support.v4.widget.DrawerLayout>  

DrawerLayout是Androidv4包里自带的控件,支持左滑和右滑,android:layout_gravity="leftt"代表左滑界面(或者start),android:layout_gravity="right"代码右滑的界面(或者end),不加layout_gravity的就是主界面。代码里可以添加ActionBarDrawerToggle控制侧滑栏展示与隐藏。


  1. ActionBarDrawerToggle mDrawerToggle = new ActionBarDrawerToggle(this, mDrawerLayout, mToolBar, R.string.open, R.string.close); 
  2. mDrawerToggle.syncState(); 
  3. mDrawer.addDrawerListener(mDrawerToggle);  

NavigationView是Google在5.0之后推出的一个控件,主要作为菜单控件使用,分为上下部分,上面的部分为headerLayout,可以自定义布局,下面的部分为menu,作为导航菜单的菜单项


  1. <?xml version="1.0" encoding="utf-8"?><menu xmlns:android="http://schemas.android.com/apk/res/android"> 
  2.     <item 
  3.         android:id="@+id/drawer_todayInHistory" 
  4.         android:checkable="true" 
  5.         android:icon="@drawable/ic_history" 
  6.         android:title="历史上的今天" /> 
  7.     <item 
  8.         android:id="@+id/drawer_gril" 
  9.         android:checkable="true" 
  10.         android:icon="@drawable/icon_gril" 
  11.         android:title="妹纸" /> 
  12.     <item 
  13.         android:id="@+id/drawer_like" 
  14.         android:checkable="true" 
  15.         android:icon="@drawable/ic_unlike" 
  16.         android:title="收藏" /> 
  17.     <item 
  18.         android:id="@+id/drawer_about" 
  19.         android:checkable="true" 
  20.         android:icon="@drawable/ic_about" 
  21.         android:title="关于" /></menu>  

点击事件:


  1. navigationView.setNavigationItemSelectedListener(new NavigationView.OnNavigationItemSelectedListener() {   
  2.     @Override   
  3.     public boolean onNavigationItemSelected(MenuItem item) {   
  4.         //在这里处理item的点击事件   
  5.         return true;   
  6.     }   
  7. });  

获取头部(headerLayout)内控件:


  1. View headView=navigationView.getHeaderView(0); 

设置菜单列表图标颜色:

默认情况下,菜单图标颜色为灰色,可以通过一下设置图标颜色


  1. app:itemIconTint="" 

添加分割线:

只需将菜单分成多个Group,每个Group设置一个Id,那么Group之间就会有分割线:


  1. <menuxmlns:android="http://schemas.android.com/apk/res/android"> 
  2. <groupandroid:id="@+id/g1"> 
  3. <item 
  4. android:id="@+id/favorite" 
  5. android:icon="@mipmap/ic_launcher" 
  6. android:title="历史上的今天"/> 
  7. <item 
  8. android:id="@+id/wallet" 
  9. android:icon="@mipmap/ic_launcher" 
  10. android:title="收藏"/> 
  11. </group> 
  12. <groupandroid:id="@+id/g2"> 
  13. <item 
  14. android:id="@+id/photo" 
  15. android:icon="@mipmap/ic_launcher" 
  16. android:title="妹子"/> 
  17. </group> 
  18. <item 
  19. android:id="@+id/file" 
  20. android:icon="@mipmap/ic_launcher" 
  21. android:title="关于"/> 
  22. </menu>  

二、Glide加载图片

设置绑定生命周期


  1. Glide.with(Context context);// 绑定Context 
  2.   Glide.with(Activity activity);// 绑定Activity 
  3.   Glide.with(FragmentActivity activity);// 绑定FragmentActivity 
  4.   Glide.with(Fragment fragment);// 绑定Fragment  

常规用法:


  1. Glide.with(context) 
  2.                 .load(imageUrl)//图片路径 
  3.                 .placeholder(R.drawable.ic_launcher)//设置加载中图片 
  4.                 .error(R.drawable.ic_launcher)//设置加载失败图片 
  5.                 .skipMemoryCache(true)//设置跳过内存缓存 
  6.                 .diskCacheStrategy(DiskCacheStrategy.ALL)//设置缓存策略:all:缓存源资源和转换后的资源/none:不作任何磁盘缓存 /source:缓存源资源 /result:缓存转换后的资源 
  7.                 .priority(Priority.NORMAL)//设置下载优先级 
  8.                 .animate(R.anim.item_alpha_in)//设置加载动画 
  9.                 .thumbnail(0.1f)//设置缩略图支持(先加载缩略图,再加载全图) 
  10.                 .override(400,400)//设置加载尺寸 
  11.                 .centerCrop()//设置动态变换 
  12.                 .into(imageView);  

加载Git图片:


  1. Glide.with(this).load(imageUrl).asGif().into(imageView); 

动态缓存清理:


  1. Glide.get(this).clearDiskCache();//清理磁盘缓存 需要在子线程中执行 Glide.get(this).clearMemory();//清理内存缓存 可以在UI主线程中进行 

加载圆角图片或圆形图片:


  1. Glide.with(this).load(imageUrl).transform(new GlideRoundTransform(this)).into(imageView); 

需要自定义Transform,这里提供一个圆角和一个圆形的Transform:

圆角转换:


  1. public class GlideRoundTransform extends BitmapTransformation { 
  2.  
  3.     private static float radius = 0f; 
  4.  
  5.     public GlideRoundTransform(Context context) { 
  6.         this(context, 4); 
  7.     } 
  8.  
  9.     public GlideRoundTransform(Context context, int dp) { 
  10.         super(context); 
  11.         this.radius = Resources.getSystem().getDisplayMetrics().density * dp; 
  12.     } 
  13.  
  14.     @Override protected Bitmap transform(BitmapPool pool, Bitmap toTransform, int outWidth, int outHeight) { 
  15.         return roundCrop(pool, toTransform); 
  16.     } 
  17.  
  18.     private static Bitmap roundCrop(BitmapPool pool, Bitmap source) { 
  19.         if (source == null) return null; 
  20.  
  21.         Bitmap result = pool.get(source.getWidth(), source.getHeight(), Bitmap.Config.ARGB_8888); 
  22.         if (result == null) { 
  23.             result = Bitmap.createBitmap(source.getWidth(), source.getHeight(), Bitmap.Config.ARGB_8888); 
  24.         } 
  25.  
  26.         Canvas canvas = new Canvas(result); 
  27.         Paint paint = new Paint(); 
  28.         paint.setShader(new BitmapShader(source, BitmapShader.TileMode.CLAMP, BitmapShader.TileMode.CLAMP)); 
  29.         paint.setAntiAlias(true); 
  30.         RectF rectF = new RectF(0f, 0f, source.getWidth(), source.getHeight()); 
  31.         canvas.drawRoundRect(rectF, radius, radius, paint); 
  32.         return result; 
  33.     } 
  34.  
  35.     @Override public String getId() { 
  36.         return getClass().getName() + Math.round(radius); 
  37.     } 
  38. }  

圆形图片转换:


  1. public class GlideCircleTransform extends BitmapTransformation { 
  2.     public GlideCircleTransform(Context context) { 
  3.         super(context); 
  4.     } 
  5.  
  6.     @Override protected Bitmap transform(BitmapPool pool, Bitmap toTransform, int outWidth, int outHeight) { 
  7.         return circleCrop(pool, toTransform); 
  8.     } 
  9.  
  10.     private static Bitmap circleCrop(BitmapPool pool, Bitmap source) { 
  11.         if (source == null) return null; 
  12.  
  13.         int size = Math.min(source.getWidth(), source.getHeight()); 
  14.         int x = (source.getWidth() - size) / 2; 
  15.         int y = (source.getHeight() - size) / 2; 
  16.  
  17.         // TODO this could be acquired from the pool too 
  18.         Bitmap squared = Bitmap.createBitmap(source, x, y, size, size); 
  19.  
  20.         Bitmap result = pool.get(size, size, Bitmap.Config.ARGB_8888); 
  21.         if (result == null) { 
  22.             result = Bitmap.createBitmap(size, size, Bitmap.Config.ARGB_8888); 
  23.         } 
  24.  
  25.         Canvas canvas = new Canvas(result); 
  26.         Paint paint = new Paint(); 
  27.         paint.setShader(new BitmapShader(squared, BitmapShader.TileMode.CLAMP, BitmapShader.TileMode.CLAMP)); 
  28.         paint.setAntiAlias(true); 
  29.         float r = size / 2f; 
  30.         canvas.drawCircle(r, r, r, paint); 
  31.         return result; 
  32.     } 
  33.  
  34.     @Override public String getId() { 
  35.         return getClass().getName(); 
  36.     } 
  37. }  

获取Bitmap


  1. Glide.with(this) 
  2.                 .load(imageUrl) 
  3.                 .asBitmap() 
  4.                 .into(new SimpleTarget<Bitmap>() { 
  5.                     @Override 
  6.                     public void onResourceReady(Bitmap resource, GlideAnimation<? super Bitmap> glideAnimation) { 
  7.                         imageView.setImageBitmap(mBitmap); 
  8.                     } 
  9.                 });  

作者:RaphetS

来源:51CTO

时间: 2024-08-02 20:03:53

使用Retrofit+RxJava+MVP打造一款MaterialDesign风格的APP的相关文章

Rxjava + retrofit + dagger2 + mvp搭建Android框架

最近出去面试,总会被问到我们项目现在采用的什么开发框架,不过据我的经验网络框架(volley)+图片缓存(uIl)+数据库(orm)+mvp,不过现在这套框架比较好了,现在采用什么呢?Rxjava + retrofit + dagger2 + mvp.好,那我们接下来说说由这几种库来组成我们一个常用的开发框架(当然其他的还有一些什么推送什么的我们另当别论). 借用google的一个MVP的说明图 项目总览 与常规的 MVP 不同的是,在 modle 层和 view 层通过EventBus 连接,

电商网站产品优化:打造爆款的真实案例

文章描述:打造爆款的真实案例. 2010年11月底,接到一个门户网站老总的小舅子的电话,让我帮他把1万多件长袖T恤库存在处理掉,"大冬天让我帮你卖T恤,这不是傻B吗",但不敢得罪他,只能硬着头皮告诉他试试看吧.可这家伙又问我能不能打照爆款,气的我差点晕过去,"这个小子一定被傻子吻过".但去淘宝浏览一下T恤的数据与他店铺的T恤数据后,发现这些T恤还真有打照爆款的可能,对方提出的条件是,这款长袖T恤进价是18元,只要18元能处理了就可以,于是决定拟定一个计划试一下. 计

PS+C4D怎么打造一款高逼格包装

PS+C4D怎么打造一款高逼格包装   宝立食品有限公司要开发一款新的产品,我这边负责产品包装的设计.整体版面构思,LOGO设计,包装袋的选材,以及后期运用C4D建模渲染出效果图全部是我一人完成,这是对自己的一个挑战.这次产品包装的整体设计格调大概就这样定下来了,后期可能微调,现在出一教程分享下.教程最后有彩蛋,不要错过哦! 分类: PS入门教程

PS打造一款精美的苹果图标

PS打造一款精美的苹果图标 先看完成图. 创建文档1280x1024像素,建新层填充黑色,加上图层样式. 效果如下. 建新层,用钢笔画出苹果形状,填充黑色. 加上图层样式. 效果如下.  分类: PS入门教程 ps图标制作教程

Wecoach陈隽永:如何打造一款完美的智能硬件App?

入秋的杭州,天气变得和深圳一样深沉,天空忽的下起暴雨. 上周六下午,宅客君在阴沉的背景下和WeCoach创始团队见面的.是的,作为一家创业公司,他们目前还需要周六加一天班来加快进度. 这次拜访,是因为WeCoach的产品度过众筹期,准备要正式上市了.宅客君过来聊聊,在众筹后.发布前产品的那些事. WeCoach,运动捕捉器迈向智能健身助手 WeCoach是一款内置九轴陀螺仪的运动捕捉器,它的外形酷似哑铃,只有一块奥利奥饼干大小. 在4月份发起众筹时,WeCoach的宣传卖点是:装到哑铃上,通过A

如何打造一款可靠的WAF

之前写了一篇<WAF防御能力评测及工具> ,是站在安全运维人员选型WAF产品的角度来考虑的(优先从测试角度考虑是前职业病,毕竟当过3年游戏测试!).本篇文章从WAF产品研发的角度来YY如何实现 一款可靠的WAF,灵感来自ModSecurity等,感谢开源.本片文章包括三个主题(1) WAF实现WAF包括哪些组件,这些组件如何交互来实现WAF防御功能(2)WAF规则(策略)维护规则(策略)如何维护,包括 获取渠道,规则测试方法以及上线效果评测(3) WAF支撑WAF产品的完善需要哪些信息库的支撑

利用消息队列MQTT,打造一款属于自己的IM社交软件

MQTT 是一种基于发布订阅模型的即时通讯协议,由于轻巧,开源,易用,耗能少,支持 QOS/遗言(WILL)等特性,正被广泛应用于物联网和移动互联网. 消息队列 MQ 提供了对 MQTT 协议的支持,完全兼容 MQTT 标准协议,但是在使用 MQ MQTT 时,对比标准协议,需要注意两点:1. 父级 Topic 需要提前创建 根据标准 MQTT 协议,Topic 存在多级,且拥有动态的特性(不需要用户提前定义和创建),但是使用过消息队列 MQ 的用户都知道,MQ Topic是需要通过 MQ 控制

如何利用AngularJS打造一款简单Web应用_AngularJS

目前不同类型的Web开发人员都在广泛使用AngularJS,这套卓越的框架也充分证明了自身满足各类不同需求的能力.作为一名Web开发人员,无论大家是刚刚入门的新手还是已经拥有丰富的实践经验,选择一款优秀的框架都是必要的工作前提,而AngularJS正是这样一套理想的解决方案.在使用AnguarJS的过程中,大家可以同时学习到更多与应用程序开发相关的知识以及如何构建起更出色.更具吸引力的应用成果.如果大家希望在应用程序的创建工作中采取各类最佳实践,那么AngularJS也能够带来极大的助益.总而言

苹果发布最新设计指南,教你如何更方便地打造一款“优雅”的App

苹果发布了其最新的设计指南(Design Guides and Resources),试图帮助开发者更好地学习 iOS7 风格的应用和游戏设计.打开&http://www.aliyun.com/zixun/aggregation/37954.html">nbsp;Apple developer center就能看到,新版指南提供了大量的视频和文档,内容涵盖应用和游戏的 UI 设计.iOS7 更新概览.最佳案例.注意事项等等,关联内容之间的链接也做得比较完善. 在指南首页,我们看到苹