RxJava+Retrofit+Material Design极简新闻App

快速完成一个新闻APP

本Demo主要使用的技术:

  • 看标题就知道了
  • Material Design
  • 聚合数据

效果

直接点吧,先看下效果


这里写图片描述

Demo架构

老司机们一看就知道界面是由ViewPager+Fragment组成,还是比较简单的。新闻详情页面主要是采用了design包下的CoordinatorLayout作为父布局,因为要做出那个下拉折叠效果嘛。然后点击新闻列表时会有一个转场动画,不知道细心的朋友们有木有看出来,上拉刷新是采用的官方的SwipeRefreshLayout
一切都追求原滋原味。
整个Demo的网络请求是通过RxJava+Retrofit来实现的,为什么用这对基友组合呢?
三个字 “太爽了”
OK,后面会有相关的介绍。
聚合数据的key请自己申请,我的已经过期了,在Config里面配置下就好了。

代码分析

封装BaseActivity 
虽然整个Demo就两个Activity,那我们还是封装一下,因为我喜欢追求代码简洁(与后面可能会有些出入)额额,,
先上波代码吧

public abstract class BaseActivity extends AppCompatActivity {

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        initContentView(savedInstanceState);
        initStatusBar();
    }

    protected abstract void initContentView(Bundle savedInstanceState);
    /**
     * 初始化沉浸式状态栏
     */
    private void initStatusBar(){
        if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT){//4.4 全透明状态栏
            getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        }
        if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {//5.0 全透明实现
            Window window = getWindow();
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
                    | View.SYSTEM_UI_FLAG_LAYOUT_STABLE);
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            window.setStatusBarColor(Color.TRANSPARENT);//calculateStatusColor(Color.WHITE, (int) alphaValue)
        }
    }
    
    @SuppressWarnings("unchecked")
    public final  E findView(int id){
        try {
            return (E) findViewById(id);
        }catch (ClassCastException e){
            throw  e;
        }
    }
}

其实也没什么亮点,就是封装下共同的方法,一个是沉浸式状态栏,一个是我为了偷懒,不想在findviewById的时候加个强制类型转换。当然也可以通过框架注入,和databinding来解决这个,但这次的重点不是他们。

创建实体类
对了,这些数据都是从聚合数据获取下来的,具体细节就不多说了,就是申请个key,填写一些参数,然后它会返回一串json数据。我们就通过这些json数据去生成对应的实体类,用一个良心之作的工具 GsonFormat,具体操作可以自行Google或者百度。
 使用这个工具一是为了偷懒,二是为了配合gson来对json解析。

创建配置文件
项目中可能会遇到一些很多地方都会用到的常量,比如说聚合数据的key等等,我们可以创建一个接口
将这些数据写到这个接口里面,这样的话,哪里要用就直接继承这个接口就OK了。

public interface Config {
     String[] ARRYTITLES ={"头条","社会","科技","国内","国际","娱乐","时尚","军事","体育","财经"};
     String KEY_POSTION="key_postion";
     String[] ARRYTYPE={"top","shehui","keji","guonei","guoji","yule","shishang","junshi","tiyu","caijing"};
     String KEY_IMG_URL="imgurl";
     String KEY_CONTENT_URL="contenturl";
     String KEY_TYPE="type";
     String KEY_JUHE="0489bcea378ce792facda791d0f1e188";

}

因为请求不同的类型的新闻,参数不一样,所以弄个数组,把参数存入进去,记得与类型对应。

主Activity编写
这里因为创建项目的时候手贱了下,点了那个有侧滑的activity,所以一些生成了很多没什么卵用的代码(至少这个项目里面没什么用)

public class MainActivity extends BaseActivity
        implements NavigationView.OnNavigationItemSelectedListener,Config {
    private TabLayout mTabLayout;
    private ViewPager mViewPager;
    private ViewPagerAdapter mAdapter;
    private List mTitles=new ArrayList<>();

    @Override
    protected void initContentView(Bundle savedInstanceState) {
        setContentView(R.layout.activity_main);
        Toolbar toolbar = findView(R.id.toolbar);
        setSupportActionBar(toolbar);
        mTabLayout=findView(R.id.tab_layout);
        mViewPager=findView(R.id.viewpager);

        DrawerLayout drawer = findView(R.id.drawer_layout);
        ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(
                this, drawer, toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
        drawer.setDrawerListener(toggle);
        toggle.syncState();

        NavigationView navigationView = findView(R.id.nav_view);
        navigationView.setNavigationItemSelectedListener(this);

        initTitle();
        mAdapter=new ViewPagerAdapter(getSupportFragmentManager(),mTitles);
        mViewPager.setAdapter(mAdapter);
        mTabLayout.setupWithViewPager(mViewPager);
        mTabLayout.setTabMode(TabLayout.MODE_SCROLLABLE);
    }


    private void initTitle(){
        for (int i=0;i

这里就是进行初始化一些view,将fragment添加进去,viewpager+tablayout基友组合。

创建Fragment的适配器
先说说适配器吧,因为这里要用的fragment比较多,所以继承FragmentStatePagerAdapter,Why?

内存优化

因为一个Fragment占的内存还是比较大,一旦fragment数量比较多了,后果你懂的。当页面不可见时, 对应的Fragment实例可能会被销毁,但是Fragment的状态会被保存,所以一些提高了我们的app性能。

public class ViewPagerAdapter extends FragmentStatePagerAdapter {
    private List mTitles;


    public ViewPagerAdapter(FragmentManager fm, List mTitles) {
        super(fm);
        this.mTitles = mTitles;

    }

    @Override
    public Fragment getItem(int position) {

        return ContentFragment.instance(position);
    }

    @Override
    public int getCount(){
        return mTitles.size();
    }

    @Override
    public CharSequence getPageTitle(int position) {
        return mTitles.get(position);
    }
}

还是比较简单的,可能最后一个方法或许有些陌生,用过TabLayout的朋友应该懂,就是设置Tab的标题,因为我们的ViewPager是要与TabLayout进行关联的。

编写网络工具类
好了重头戏来了,也是本项目唯一的特色,RxJava+Retrofit。
记得别忘了引入这些框架

    compile 'io.reactivex:rxjava:1.0.14'
    compile 'io.reactivex:rxandroid:1.0.1'
    compile 'com.squareup.retrofit2:retrofit:2.1.0'
    compile 'com.squareup.retrofit2:converter-gson:2.1.0'
    compile 'com.google.code.gson:gson:2.6.2'
    compile 'com.squareup.retrofit2:adapter-rxjava:2.0.0'

这里我不做过多关于RxJava和Retrofit的描述,因为相关资料网上一堆堆。RxJava说到底就是异步,这是它整个流程非常简洁明了,而且方便线程切换。Retrofit是讲OKHttp更好的封装下,简化我们的网络请求。
首先编写Retrofit接口

public interface NewService {
    String BASE_URL="http://v.juhe.cn/";

    @GET("toutiao/index?")
    Observable getNews(@QueryMap Map map);
}

好像News 少了个s

接下来编写我们的请求工具类

 private static final int DEFAULT_TIMEOUT = 5;
    private Retrofit retrofit;
    private NewService newService;

    private RetrofitUtil(){
        //手动创建一个OkHttpClient并设置超时时间
        OkHttpClient.Builder httpClientBuilder = new OkHttpClient.Builder();
        httpClientBuilder.connectTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS);
        Gson gson = new GsonBuilder()
                //配置Gson
                .setDateFormat("yyyy-MM-dd hh:mm:ss")
                .create();
        retrofit=new Retrofit.Builder()
                .baseUrl(NewService.BASE_URL)
                .addConverterFactory(GsonConverterFactory.create(gson))
                .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
                .build();
        newService=retrofit.create(NewService.class);

    }

    //在访问HttpMethods时创建单例
    private static class SingletonHolder{
        private static final RetrofitUtil INSTANCE = new RetrofitUtil();
    }

    public static RetrofitUtil getInstance(){
        return SingletonHolder.INSTANCE;
    }

    public void getNews(Subscriber newsSubscriber,int type){
        newService.getNews(getParams(type))
                .subscribeOn(Schedulers.io())
                .unsubscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(newsSubscriber);

    }

    private Map getParams(int type){
        Map map=new HashMap<>();
        map.put("type",ARRYTYPE[type]);
        map.put("key",KEY_JUHE);
        return map;
    }
}

我们在外面只用调用getNews就行了,传一个Subscriber和类型就行了。逻辑都不是很复杂吧。里面的精髓就是线程切换 .subscribeOn(Schedulers.io()) ,RxJava给我吗提供了五个选择,这里因为我们是请求网络,所以就用io的,最后切换到主线程 .observeOn(AndroidSchedulers.mainThread())

编写Fragment

既然网络工具类写好了,那么就写个fragment来把这些数据展示出来吧!

public class ContentFragment extends Fragment implements Config {

    private int mType;
    private List mData;
    private SwipeRefreshLayout mRefreshLayout;
    private RecyclerView mShowNews;
    private NewsAdapter mAdapter;
    private int mSpacingInPixels;//
    private int mCount=0;


    public static Fragment instance(int postion){
        ContentFragment fragment=new ContentFragment();
        Bundle bundle = new Bundle() ;
        bundle.putInt(KEY_POSTION,postion);

        fragment.setArguments(bundle);
        return fragment;
    }

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view=inflater.inflate(R.layout.fragment_content,container,false);
        return view;
    }

    @Override
    public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        Bundle bundle = getArguments() ;
        Log.e("dandy","pos "+bundle.getInt(KEY_POSTION));
        initViews(view);
        mType=bundle.getInt(KEY_POSTION);

        getData();
    }

    private void initViews(View view) {
        mRefreshLayout= (SwipeRefreshLayout) view.findViewById(R.id.refresh_layout);
        mShowNews= (RecyclerView) view.findViewById(R.id.news_recyclerview);
        mRefreshLayout.setColorSchemeResources(R.color.colorPrimary,R.color.tab_select_text_color,R.color.refresh_color,R.color.colorAccent);
        mRefreshLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() {
            @Override
            public void onRefresh() {
                getData();
            }
        });
        mSpacingInPixels= getResources().getDimensionPixelSize(R.dimen.item_space);
        mShowNews.setHasFixedSize(true);
    }

    private void getData(){
        mRefreshLayout.setRefreshing(true);
        Subscriber subscriber=new Subscriber() {
            @Override
            public void onCompleted() {

            }
            //出现异常回调
            @Override
            public void onError(Throwable e) {
                Log.e("smile","获取失败");
            }
            //获取数据成功后回调
            @Override
            public void onNext(News news) {
                Log.e("smile","获取出来的"+news.getResult().getData().size());
                //mData=news.getResult().getData();
                setData(news);
            }
        };
        RetrofitUtil.getInstance().getNews(subscriber,mType);
    }

    private void setData(News data){

        mData=data.getResult().getData();
        mRefreshLayout.setRefreshing(false);
        mAdapter=new NewsAdapter(getContext(),data);
        mShowNews.setLayoutManager(new LinearLayoutManager(getContext()));
        //避免重复添加间距
        if (mCount==0){
            mShowNews.addItemDecoration(new SpacesItemDecoration(mSpacingInPixels));
        }

        mShowNews.setAdapter(mAdapter);
       // mAdapter.setOnScrollListener(mShowNews);
        mAdapter.setOnItemClickListener(new NewsAdapter.OnItemClickListener() {

            @Override
            public void onItemClick(View view, int position) {
                startDetailActivity(view,mData.get(position));
            }

            @Override
            public void onItemLongClick(View view, int position) {

            }
        }) ;
        mCount++;

    }

    @TargetApi(Build.VERSION_CODES.JELLY_BEAN)
    private void startDetailActivity(View view, News.ResultBean.DataBean bean){
        Intent intent=new Intent(getActivity(), NewsDetailActivity.class);
        Bundle bundle=new Bundle();
        bundle.putString(KEY_IMG_URL,bean.getThumbnail_pic_s());
        bundle.putString(KEY_CONTENT_URL,bean.getUrl());
        bundle.putString(KEY_TYPE,ARRYTITLES[mType]);
        intent.putExtras(bundle);
        ActivityOptionsCompat options = ActivityOptionsCompat.makeSceneTransitionAnimation(getActivity(),view.findViewById(R.id.item_news_img),"photos");
        getContext().startActivity( intent, options.toBundle());
    }
}

其实仔细一看也不是很复杂,就是调用我们开始写的getNews而已,请求成功后会回调onNext方法。对了这里有个转场动画,就是最后一个方法,这里的动画效果是共享元素,所以指定你要共享的元素就行,然后设置下它的 android:transitionName="photos"。

编写RecyclerView适配器
再见ListView,你好RecyclerView
RecyclerView的优点就不多说了,就是自由,任性
适配器,我就不贴代码了,累,而且没什么特色。

编写详情页面
这里主要都是用了Design里面的一些控件



    
        
            
            
        
    

    
        
    

折叠式效果就是通过编写xml就能实现了,还是简单的描述下那些属性的含义
CollapsingToolbarLayout

//折叠后的背景色  -> setContentScrim(Drawable)
  app:contentScrim="?attr/colorPrimary"   
  // 必须设置透明状态栏才有效  -> setStatusBarScrim(Drawable)     
  app:statusBarScrim="?attr/colorAccent"    
  // 标题  
  app:title="title"
  // 折叠后的标题位置
  app:collapsedTitleGravity="right"
  // 打开时的标题位置
  app:expandedTitleGravity="center_horizontal|bottom"

折叠效果

app:layout_collapseMode      
  有两个可选:
       parallax ——  视差模式,就是上面的图片的变化效果
       pin     —— 固定模式,在折叠的时候最后固定在顶端

  // 视差效果
  app:layout_collapseParallaxMultiplier   
  范围[0.0,1.0],值越大视差越大

下面的MyNestedScrollView是我自定义的一个view,继承的NestedScrollView,主要是为了解决事件冲突导致滑动卡顿,就是不让拦截子view的触摸事件。

class MyNestedScrollView extends NestedScrollView {
    private GestureDetector mGestureDetector;
    View.OnTouchListener mGestureListener;
    public MyNestedScrollView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    public MyNestedScrollView(Context context) {
        super(context);
    }

    public MyNestedScrollView(Context context, AttributeSet attrs) {
        super(context, attrs);
        mGestureDetector = new GestureDetector(context, new YScrollDetector());
        setFadingEdgeLength(0);
    }

    class YScrollDetector extends GestureDetector.SimpleOnGestureListener {
        @Override
        public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
            if (Math.abs(distanceY) > Math.abs(distanceX)) {
                return true;
            }
            return false;
        }
    }

}

到这里差不多也要完工了,就可以看到开头的效果了,貌似第一次写这么长的博客,,,
最后附上源码 传送门
喜欢就给个星星吧!

你可能感兴趣的:(RxJava+Retrofit+Material Design极简新闻App)