写一写虎扑 APP 首页效果,jrs 都进来看一眼!

   2016-12-01 0
核心提示:首先先上效果图:首先我们先考虑这个界面有什么难点:1.日期悬停。2.今日和其他日期在悬停布局上的处理3.首次进入时,自动滑动到今日比赛4.向上滑动向下滑动一定距离后,出现向上向下的图标,并滚动到今日的比赛界面5.向上加载更多,向下加载更多,当然这个并

首先先上效果图:

首先我们先考虑这个界面有什么难点:

1.日期悬停。

2.今日和其他日期在悬停布局上的处理

3.首次进入时,自动滑动到今日比赛

4.向上滑动向下滑动一定距离后,出现向上向下的图标,并滚动到今日的比赛界面

5.向上加载更多,向下加载更多,当然这个并不算是难点,只是比较新颖~。

好了,首先我们一项一项的开始解决这几个难点,首先这样的列表布局,强烈推荐大家使用RecyclerView,谷歌粑粑真的是赋予了它无穷的潜力。观看本文前依然推荐一波博客地址:点击打开链接一个头像狂魔的博客。

如果你已经看过了上篇博客,你可能会对这个界面已经有一定的了解,那么首先我们实现ItemDecoration的编写,首先我们应该考虑后台与我们交互时给出来的数据,一般为List或者Map格式,所以我们在编写时应充分考虑两种格式写法上的差异(其实也没什么差异的地方)。

public class HuPuItemDecoration<T extends HuPuBaseBean> extends RecyclerView.ItemDecoration {

    private int todayPosition;
    private List<T> mData;

    private Map<String, List<T>> mapData;

    private Paint mPaint;


    private int itemHeight;

    private int background;

    private int textSize;

    private int textColor;

    private Context context;

    private Rect outBounds = new Rect();


    public HuPuItemDecoration(Context context, int todayPosition, List<T> mData) {
        this.context = context;
        this.todayPosition = todayPosition;
        this.mData = mData;
        mPaint = new Paint();
        itemHeight = dip2px(context, 25);
        background = ContextCompat.getColor(context, android.R.color.darker_gray);
        textSize = sp2px(context, 15);
        textColor = ContextCompat.getColor(context, android.R.color.black);
    }

    public HuPuItemDecoration(Context context, int todayPosition, Map<String, List<T>> mapData) {
        this.context = context;
        this.todayPosition = todayPosition;
        this.mapData = mapData;
        mPaint = new Paint();
        itemHeight = dip2px(context, 25);
        background = ContextCompat.getColor(context, android.R.color.darker_gray);
        textSize = sp2px(context, 15);
        textColor = ContextCompat.getColor(context, android.R.color.black);
        for (Map.Entry<String, List<T>> entry : mapData.entrySet()) {
            mData.addAll(entry.getValue());
        }
    }

这里的HuPuBaseBean是一个bean类,内部实现了getDayType的方法,这里是基于封装而写出的基类,如果在实际项目中可以直接引入我们所需要的实体类来进行赋值。

初始化完成后,我们开始一步一步实现:代码很简单,一梭子给出来一部分先:

关于这部分方法的意义,推荐大家去鸿洋sama的博客去充值信仰!!!:点击打开链接

@Override
    public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
        super.getItemOffsets(outRect, view, parent, state);
        int position = ((RecyclerView.LayoutParams) view.getLayoutParams()).getViewLayoutPosition();
        if (null != mData && null != mapData) {
            throw new IllegalArgumentException("这里不会走到的,只是判断一下");
        }

        if (position > -1) {
            if (position == 0) {
                outRect.set(0, itemHeight, 0, 0);
            } else {
                if (!mData.get(position).getDayType().equals(mData.get(position - 1).getDayType())) {
                    outRect.set(0, itemHeight, 0, 0);
                } else {
                    outRect.set(0, 0, 0, 0);
                }
            }
        }
    }

这里的代码逻辑比较简单,就是当position==0时,我们必然会有一个divider去标识日期,而后当前的type不等于它所在位置的上一个position的type时,这时两个position所在的view中间应有一个divider来进行标识!这部分代码非常的简单,我们继续一梭子撸下面的代码:

@Override
    public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) {
        super.onDraw(c, parent, state);
        int left = parent.getPaddingLeft();
        int right = parent.getRight() - parent.getPaddingRight();
        int count = parent.getChildCount();
        for (int i = 0; i < count; i++) {
            View child = parent.getChildAt(i);
            RecyclerView.LayoutParams lp = (RecyclerView.LayoutParams) child.getLayoutParams();
            int position = lp.getViewLayoutPosition();
            if (position == 0) {
                mPaint.setColor(background);
                c.drawRect(left, child.getTop() - itemHeight - lp.topMargin, right, child.getTop() - lp.topMargin, mPaint);
                mPaint.setColor(textColor);
                mPaint.setTextSize(textSize);
                String type = mData.get(position).getDayType();
                mPaint.getTextBounds(type, 0, type.length(), outBounds);
                c.drawText(type, child.getRight() / 2 - outBounds.width() / 2, child.getTop() - lp.topMargin -
	 (itemHeight / 2 - outBounds.height() / 2), mPaint);
            } else {
                if (!mData.get(position).getDayType().equals(mData.get(position - 1).getDayType())) {
                    mPaint.setColor(background);
                    c.drawRect(left, child.getTop() - itemHeight - lp.topMargin, right, child.getTop() - lp.topMargin, mPaint);
                    if (position == todayPosition) {
                        mPaint.setColor(Color.RED);
                        c.drawRect(left, child.getTop() - itemHeight - lp.topMargin, 15, child.getTop() - lp.topMargin, mPaint);
                    }
                    mPaint.setColor(background);
                    mPaint.setTextSize(textSize);
                    mPaint.setColor(textColor);
                    String type = mData.get(position).getDayType();
                    mPaint.getTextBounds(type, 0, type.length(), outBounds);
                    c.drawText(type, child.getRight() / 2 - outBounds.width() / 2, child.getTop() -
	 lp.topMargin - (itemHeight / 2 - outBounds.height() / 2), mPaint);
                } else {

                }
            }
        }

    }

这部分代码其实跟上面的代码在逻辑上非常接近,如果你已经看了最上面的博客链接,这里的逻辑你应该也能看得懂,这里和推荐博客在这里并没有什么不同,如果这里有任何疑问,我强烈大家去推荐博客那里去看详细的解释,(什么是甩锅,这就叫甩锅)。

最后一段代码:

@Override
    public void onDrawOver(Canvas c, RecyclerView parent, RecyclerView.State state) {
        super.onDrawOver(c, parent, state);
        int position = ((LinearLayoutManager) parent.getLayoutManager()).findFirstVisibleItemPosition();
        View itemView = parent.findViewHolderForAdapterPosition(position).itemView;
        if (mData.size() - 1 >= position) {
            if (!mData.get(position).getDayType().equals(mData.get(position + 1).getDayType())) {
                if (itemView.getHeight() + itemView.getTop() < itemHeight) {
                    c.translate(0, itemView.getHeight() + itemView.getTop() - itemHeight);

                }
            }

        }

        mPaint.setColor(background);
        c.drawRect(parent.getPaddingLeft(), parent.getPaddingTop(), parent.getRight() - parent.getPaddingRight(), 
parent.getPaddingTop() + itemHeight, mPaint);
        if (position >= todayPosition && mData.get(todayPosition).getDayType().equals(mData.get(position).getDayType())) {
            mPaint.setColor(Color.RED);
            c.drawRect(parent.getPaddingLeft(), parent.getPaddingTop(), 15, parent.getPaddingTop() + itemHeight, mPaint);
        }
        mPaint.setColor(textColor);
        mPaint.setTextSize(textSize);
        String dayType = mData.get(position).getDayType();
        mPaint.getTextBounds(dayType, 0, dayType.length(), outBounds);
        c.drawText(dayType, (parent.getRight() - parent.getPaddingRight() - parent.getPaddingLeft() - outBounds.width()) / 2,
parent.getPaddingTop() + itemHeight - (itemHeight / 2 - outBounds.height() / 2), mPaint);

    }

这里就是做悬停部分的代码,这里唯一需要注意的就是我们对于今日比赛悬停界面的处理。

if (position >= todayPosition && mData.get(todayPosition).getDayType().equals(mData.get(position).getDayType())) {
            mPaint.setColor(Color.RED);
            c.drawRect(parent.getPaddingLeft(), parent.getPaddingTop(), 15, parent.getPaddingTop() + itemHeight, mPaint);
        }

如果当前的position大于等于我们传过来的todayPos,然后type相等,既判定为今日比赛。其他则无视之。

写完了这部分代码后,我们就简单写一个RecyclerView出来:

rv = (RecyclerView) findViewById(R.id.rv);
        top= (ImageView) findViewById(R.id.top);
        bottom= (ImageView) findViewById(R.id.bottom);

        for (int i = 0; i < 10; i++) {
            HuPuBean bean = new HuPuBean("雷霆", "火箭", "早上8:00", 124, 100, "昨日", false);
            mData.add(bean);
        }
        for (int i = 0; i < 10; i++) {
            HuPuBean bean = new HuPuBean("雷霆", "火箭", "早上8:00", 124, 100, "今日", true);
            mData.add(bean);
        }
        for (int i = 0; i < 10; i++) {
            HuPuBean bean = new HuPuBean("雷霆", "火箭", "早上8:00", 124, 100, "明日", true);
            mData.add(bean);
        }

        rv.addItemDecoration(new HuPuItemDecoration<HuPuBean>(MainActivity.this, Utils.getTodayPosition("今日", mData), mData));
        final LinearLayoutManager manger = new LinearLayoutManager(MainActivity.this, LinearLayoutManager.VERTICAL, false);
        rv.setLayoutManager(manger);
        rv.setAdapter(new RecyclerView.Adapter<Holder>() {
            @Override
            public Holder onCreateViewHolder(ViewGroup parent, int viewType) {
                View itemView = LayoutInflater.from(MainActivity.this).inflate(R.layout.item, parent, false);
                return new Holder(itemView);
            }

            @Override
            public void onBindViewHolder(Holder holder, int position) {
                HuPuBean bean = mData.get(position);
                holder.teamOne.setText(bean.getTeamOne());
                holder.teamTwo.setText(bean.getTeamTwo());
                holder.teamOnePoint.setText("" + bean.getTeamOnePoint());
                holder.teamOnePoint.setText("" + bean.getTeamTwoPoint());
            }

            @Override
            public int getItemCount() {
                return mData.size();
            }
        });

这里代码写得比较粗糙,勿怪。

然后我们来看一下效果图。

我们来看一下utils方法,也非常简单,因为涉及到加载刷新的逻辑,todayPostion的位置会发生改变,不建议使用赋值,

public class Utils {


    public static int getTodayPosition(String type, List<? extends HuPuBaseBean> mData) {
        for (int i = 0; i < mData.size(); i++) {
            if (mData.get(i).getDayType().equals(type)) {
                return i;
            }
        }
        return 0;
    }


}

最后我们来处理上滑下滑时top bottom图片的显示隐藏:

首先我们看MianActivity的布局:

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    >

    <android.support.v7.widget.RecyclerView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/rv"
        ></android.support.v7.widget.RecyclerView>

    <ImageView
        android:layout_width="50dp"
        android:layout_height="50dp"
        android:layout_gravity="right"
        android:layout_marginTop="20dp"
        android:layout_marginRight="20dp"
        android:id="@+id/bottom"
        android:background="@mipmap/ic_launcher"
        android:visibility="gone"
        />
    <ImageView
        android:layout_width="50dp"
        android:layout_height="50dp"
        android:layout_gravity="bottom|right"
        android:layout_marginBottom="20dp"
        android:layout_marginRight="20dp"
        android:id="@+id/top"
        android:background="@mipmap/ic_launcher"
        android:visibility="gone"
        />
</FrameLayout>

然后通过RecyclerView的对滑动监听来实现两张图片的显示和隐藏:

rv.addOnScrollListener(new RecyclerView.OnScrollListener() {

            @Override
            public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
                super.onScrolled(recyclerView, dx, dy);
                int position = manger.findFirstVisibleItemPosition();
                if(position-Utils.getTodayPosition("今日",mData)>=10){
                    bottom.setVisibility(View.GONE);
                    top.setVisibility(View.VISIBLE);
                }else if(position-Utils.getTodayPosition("今日",mData)<=-10){
                    bottom.setVisibility(View.VISIBLE);
                    top.setVisibility(View.GONE);
                }else {
                    bottom.setVisibility(View.GONE);
                    top.setVisibility(View.GONE);
                }
            }
        });

当然这种实现方式是最常见的,不过我们还可以通过自定义behavior这种方式来实现滑动的联动效果,但是本篇就不在这里实现了。

最后首次加载滑动到今日比赛,和点击图片滑动到今日比赛,其实非常简单,一行代码搞定:

manger.scrollToPositionWithOffset(Utils.getTodayPosition("今日", mData), 0);

至此,基本的虎扑看球页面的效果就已经实现了,是不是非常简单,当然虎扑的页面更加复杂,itemType有数种,这里就不再实现了。最后去愉快的水群了。临走前在发一张我老婆的美图:

 
标签: 安卓开发
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • 安卓中通知功能的具体实现
    安卓中通知功能的具体实现
    通知[Notification]是Android中比较有特色的功能,当某个应用程序希望给用户发出一些提示信息,而该应用程序又不在前台运行时,就可以借助通知实现。使用通知的步骤1、需要一个NotificationManager来获得NotificationManager manager = (NotificationManager
    02-05 安卓开发
  • Android view系统分析-setContentView
    Android view系统分析-setContentView
    第一天上班,列了一下今年要学习的东西。主要就是深入学习Android相关的系统源代码,夯实基础。对于学习Android系统源代码,也没什么大概,就从我们平常使用最基础的东西学起,也就是从view这个切入点开始学习Android的源码,在没分析源码之前,我们有的时候
    02-05 安卓开发
  • 如何进行网络视频截图/获取视频的缩略图
    如何进行网络视频截图/获取视频的缩略图
    小编导读:获取视频的缩略图,截图正在播放的视频某一帧,是在音视频开发中,常遇到的问题。本文是主要用于点播中截图视频,同时还可以获取点播视频的缩略图进行显示,留下一个问题,如下图所示, 如果要获取直播中节目视频缩略图,该怎么做呢?(ps:直播是直
  • Android NDK 层发起 HTTP 请求的问题及解决
    Android NDK 层发起 HTTP 请求的问题及解决
    前言新的一年,大家新年快乐~~鸡年大吉!本次给大家带来何老师的最新文章~虽然何老师还在过节,但依然放心不下广大开发者,在此佳节还未结束之际,给大家带来最新的技术分享~ 事件的起因不说了,总之是需要实现一个 NDK 层的网络请求。为了多端适用,还是选择
  • Android插件化(六): OpenAtlasの改写aapt以防止资源ID冲突
    Android插件化(六): OpenAtlasの改写aapt以防
    引言Android应用程序的编译中,负责资源打包的是aapt,如果不对打包后的资源ID进行控制,就会导致插件中的资源ID冲突。所以,我们需要改写aapt的源码,以达到通过某种方式传递资源ID的Package ID,通过aapt打包时获取到这个Package ID并且应用才插件资源的命名
    02-05 安卓开发
  • Android架构(一)MVP架构在Android中的实践
    Android架构(一)MVP架构在Android中的实践
    为什么要重视程序的架构设计 对程序进行架构设计的原因,归根结底是为了 提高生产力 。通过设计是程序模块化,做到模块内部的 高聚合 和模块之间的 低耦合 (如依赖注入就是低耦合的集中体现)。 这样做的好处是使得程序开发过程中,开发人员主需要专注于一点,
    02-05 安卓开发
  • 安卓逆向系列教程 4.2 分析锁机软件
    安卓逆向系列教程 4.2 分析锁机软件
    安卓逆向系列教程 4.2 分析锁机软件 作者: 飞龙 这个教程中我们要分析一个锁机软件。像这种软件都比较简单,完全可以顺着入口看下去,但我这里还是用关键点来定位。首先这个软件的截图是这样,进入这个界面之后,除非退出模拟器,否则没办法回到桌面。上面那
    02-05 安卓开发
  • Android插件化(二):OpenAtlas插件安装过程分析
    Android插件化(二):OpenAtlas插件安装过程分析
    在前一篇博客 Android插件化(一):OpenAtlas架构以及实现原理概要 中,我们对应Android插件化存在的问题,实现原理,以及目前的实现方案进行了简单的叙述。从这篇开始,我们要深入到OpenAtlas的源码中进行插件安装过程的分析。 插件的安装分为3种:宿主启动时立
    02-05 安卓开发
  • [译] Android API 指南
    [译] Android API 指南
    众所周知,Android开发者有中文网站了,API 指南一眼看去最左侧的菜单都是中文,然而点进去内容还是很多是英文,并没有全部翻译,我这里整理了API 指南的目录,便于查看,如果之前还没有通读,现在可以好好看一遍。注意,如果标题带有英文,说明官方还没有翻
  • 使用FileProvider解决file:// URI引起的FileUriExposedException
    使用FileProvider解决file:// URI引起的FileUri
    问题以下是一段简单的代码,它调用系统的相机app来拍摄照片:void takePhoto(String cameraPhotoPath) {File cameraPhoto = new File(cameraPhotoPath);Intent takePhotoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);takePhotoIntent.putExtra(Medi
    02-05 安卓开发
点击排行