热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android控件RecyclerView实现混排效果仿网易云音乐

这篇文章主要为大家详细介绍了Android控件RecyclerView实现混排效果,仿网易云音乐,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

前言

最近在使用网易云音乐的时候,看到如下图的排版效果图,自己也想实现一个

这里采用网上用法最多的方式,而且是比较简单的方式实现的,想要做项目的同学也可以快速入手搞定首页界面,可以在最快的时间内模仿出来,且效果达到90%以上的相似

效果演示

至于图片的加载你们可以根据网上的Api获取相应的图片加载到对应的位置,这里只是采用本地图片来演示

这里写图片描述

实现分析

这里是采用RecyclerView的GridLayoutManager的一个SpanSize这么一个东西,从下图很容易知道其意思

这里写图片描述

项目结构

项目结构可能对初学者感觉很庞大,不用担心,这里我会按照下面的包名划分,从最简单的开始分析

这里写图片描述

引入依赖

首先是在Gradle中引入对RecyclerView的依赖

compile 'com.android.support:recyclerview-v7:25.3.1'

View包

由于项目用到的图片是有规格限定的,所以需要对ImageView覆写,得到我们想要尺寸的图片

SquareImageView:正方形图片
RectImageView:长方形图片

public class SquareImageView extends android.support.v7.widget.AppCompatImageView {

  public SquareImageView(Context context) {
    this(context,null);
  }

  public SquareImageView(Context context, @Nullable AttributeSet attrs) {
    this(context, attrs,0);
  }

  public SquareImageView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);

    setScaleType(ScaleType.FIT_XY);
  }

  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, widthMeasureSpec);
  }
}

public class RectImageView extends android.support.v7.widget.AppCompatImageView {

  public RectImageView(Context context) {
    this(context, null);
  }

  public RectImageView(Context context, @Nullable AttributeSet attrs) {
    this(context, attrs, 0);
  }

  public RectImageView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);

    setScaleType(ScaleType.FIT_XY);
  }

  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int widthMode = MeasureSpec.getMode(widthMeasureSpec);
    int width = MeasureSpec.getSize(widthMeasureSpec);
    // 设置大小为宽度的三分之二
    int halfWidthMeasureSpec = MeasureSpec.makeMeasureSpec(width / 3 * 2, widthMode);
    super.onMeasure(widthMeasureSpec, halfWidthMeasureSpec);
  }
}

Music包

这里是我们存储实体的地方,其中四种类型的划分,分别对应项目展示中的前三个模块的划分,其中还有一个标题也算是一种类型,所以共四种

public class Music {

  public int type;
  public String title;
  // 后期可加入Glide加载网络图片Url
  public int imageId;

  public interface TYPE {
    int TYPE_GRID_THREE = 0x01;
    int TYPE_GRID_TWO = 0x02;
    int TYPE_LIST = 0x03;
    int TYPE_TITLE = 0x04;
  }
}

Listener包

由于RecyclerView自身是没有点击事件的,所以这个包是RecyclerView的点击事件接口

public interface OnItemClickListener {
  void OnItemClick(int position);
}

Decoration包

由于RecyclerView是不提供分割线的,所以这个包是自定义的分割线

public class SpacesItemDecoration extends RecyclerView.ItemDecoration {

  private int space;

  public SpacesItemDecoration(int space) {
    this.space = space;
  }

  @Override
  public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
    outRect.left = space;
    outRect.right = space;
    outRect.bottom = space;
    outRect.top = space;
  }
}

ViewHolder

这里存储的是我们混排效果的控件,标题可能会有点区别,其他是一样的效果,为了后期方便拓展,我们就把他们分开,不代码复用

public class GridThreeViewHolder extends RecyclerView.ViewHolder {

  public SquareImageView iv_icon;
  public TextView tv_content;

  public GridThreeViewHolder(View itemView) {
    super(itemView);
    iv_icon = (SquareImageView) itemView.findViewById(R.id.iv_icon);
    tv_cOntent= (TextView) itemView.findViewById(R.id.tv_content);
  }
}

public class GridTwoViewHolder extends RecyclerView.ViewHolder {

  public RectImageView iv_icon;
  public TextView tv_content;

  public GridTwoViewHolder(View itemView) {
    super(itemView);
    iv_icon = (RectImageView) itemView.findViewById(R.id.iv_icon);
    tv_cOntent= (TextView) itemView.findViewById(R.id.tv_content);
  }
}

public class ListViewHolder extends RecyclerView.ViewHolder {

  public RectImageView iv_icon;
  public TextView tv_content;

  public ListViewHolder(View itemView) {
    super(itemView);
    iv_icon = (RectImageView) itemView.findViewById(R.id.iv_icon);
    tv_cOntent= (TextView) itemView.findViewById(R.id.tv_content);
  }
}

public class TitleViewHolder extends RecyclerView.ViewHolder {

  public TextView tv_content;

  public TitleViewHolder(View itemView) {
    super(itemView);
    tv_cOntent= (TextView) itemView.findViewById(R.id.tv_content);
  }
}

Adapter包

这里就是对所有ViewHolder的控制器,然而这里并不是混排效果实现的最终地方,只不过是填充数据的地方

public class RecyclerAdapter extends RecyclerView.Adapter implements View.OnClickListener {

  private List mList;
  private Context mContext;
  private LayoutInflater mInflater;

  /**
   * 点击事件
   */
  private OnItemClickListener mOnItemClickListener;

  public void setOnItemClickListener(OnItemClickListener onItemClickListener) {
    this.mOnItemClickListener= onItemClickListener;
  }

  public RecyclerAdapter(Context context, List list) {
    this.mList = list;
    this.mCOntext= context;
    this.mInflater = LayoutInflater.from(context);
  }

  @Override
  public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View view;
    RecyclerView.ViewHolder mViewHolder = null;
    if (viewType == Music.TYPE.TYPE_GRID_THREE) {
      view = mInflater.inflate(R.layout.item_grid_three, parent, false);
      mViewHolder = new GridThreeViewHolder(view);
    } else if (viewType == Music.TYPE.TYPE_GRID_TWO) {
      view = mInflater.inflate(R.layout.item_grid_two, parent, false);
      mViewHolder = new GridTwoViewHolder(view);
    } else if (viewType == Music.TYPE.TYPE_LIST) {
      view = mInflater.inflate(R.layout.item_list, parent, false);
      mViewHolder = new ListViewHolder(view);
    } else if (viewType == Music.TYPE.TYPE_TITLE) {
      view = mInflater.inflate(R.layout.item_title, parent, false);
      mViewHolder = new TitleViewHolder(view);
    }
    return mViewHolder;
  }

  @Override
  public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
    switch (getItemViewType(position)) {
      case Music.TYPE.TYPE_GRID_THREE:
        GridThreeViewHolder gHolder_three = (GridThreeViewHolder) holder;
        gHolder_three.tv_content.setText(mList.get(position).title);
        gHolder_three.iv_icon.setImageResource(mList.get(position).imageId);
        //点击事件
        gHolder_three.itemView.setOnClickListener(this);
        gHolder_three.itemView.setTag(position);
        break;
      case Music.TYPE.TYPE_GRID_TWO:
        GridTwoViewHolder gHolder_two = (GridTwoViewHolder) holder;
        gHolder_two.tv_content.setText(mList.get(position).title);
        gHolder_two.iv_icon.setImageResource(mList.get(position).imageId);
        //点击事件
        gHolder_two.itemView.setOnClickListener(this);
        gHolder_two.itemView.setTag(position);
        break;
      case Music.TYPE.TYPE_LIST:
        ListViewHolder lHolder = (ListViewHolder) holder;
        lHolder.tv_content.setText(mList.get(position).title);
        lHolder.iv_icon.setImageResource(mList.get(position).imageId);
        //点击事件
        lHolder.itemView.setOnClickListener(this);
        lHolder.itemView.setTag(position);
        break;
      case Music.TYPE.TYPE_TITLE:
        TitleViewHolder tHolder = (TitleViewHolder) holder;
        tHolder.tv_content.setText(mList.get(position).title);
        //点击事件
        tHolder.itemView.setOnClickListener(this);
        tHolder.itemView.setTag(position);
        break;
    }
  }

  @Override
  public int getItemViewType(int position) {
    return mList.get(position).type;
  }

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

  @Override
  public void onClick(View v) {
    if (mOnItemClickListener != null) {
      int position = (int) v.getTag();
      mOnItemClickListener.OnItemClick(position);
    }
  }
}

Activity

这里就是我们实现混排效果的关键,我们会根据不同类型的数据,对RecyclerView的SpanSize的进行设置

public class MainActivity extends AppCompatActivity implements OnItemClickListener {

  private RecyclerView ry;
  private GridLayoutManager layoutManager;
  private RecyclerAdapter mAdapter;
  private static List mList;

  /**
   * 模拟本地数据
   */
  static {
    mList = new ArrayList<>();

    for (int i = 0; i <1; i++) {
      Music music = new Music();
      music.type = Music.TYPE.TYPE_TITLE;
      music.imageId = R.drawable.ic_cover;
      music.title = "推荐歌单";
      mList.add(music);
    }

    for (int i = 0; i <6; i++) {
      Music music = new Music();
      music.type = Music.TYPE.TYPE_GRID_THREE;
      music.imageId = R.drawable.ic_cover;
      music.title = "先不要降温!我没钱买衣服";
      mList.add(music);
    }

    for (int i = 0; i <1; i++) {
      Music music = new Music();
      music.type = Music.TYPE.TYPE_TITLE;
      music.imageId = R.drawable.ic_cover;
      music.title = "推荐MV";
      mList.add(music);
    }

    for (int i = 0; i <4; i++) {
      Music music = new Music();
      music.type = Music.TYPE.TYPE_GRID_TWO;
      music.imageId = R.drawable.ic_cover;
      music.title = "Perfect Day";
      mList.add(music);
    }

    for (int i = 0; i <1; i++) {
      Music music = new Music();
      music.type = Music.TYPE.TYPE_TITLE;
      music.imageId = R.drawable.ic_cover;
      music.title = "精选专栏";
      mList.add(music);
    }

    for (int i = 0; i <3; i++) {
      Music music = new Music();
      music.type = Music.TYPE.TYPE_LIST;
      music.imageId = R.drawable.ic_cover;
      music.title = "去看《银翼杀手2049》前,你应该知道的三件事";
      mList.add(music);
    }

    for (int i = 0; i <1; i++) {
      Music music = new Music();
      music.type = Music.TYPE.TYPE_TITLE;
      music.imageId = R.drawable.ic_cover;
      music.title = "最新音乐";
      mList.add(music);
    }

    for (int i = 0; i <6; i++) {
      Music music = new Music();
      music.type = Music.TYPE.TYPE_GRID_THREE;
      music.imageId = R.drawable.ic_cover;
      music.title = "[BGM]一定听过的神级背景配乐";
      mList.add(music);
    }
  }

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    ry = (RecyclerView) findViewById(R.id.ry);
    layoutManager = new GridLayoutManager(this, 6);
    layoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
      @Override
      public int getSpanSize(int position) {
        int type = mList.get(position).type;
        if (type == Music.TYPE.TYPE_GRID_THREE) {
          return 2;
        } else if (type == Music.TYPE.TYPE_GRID_TWO) {
          return 3;
        } else if (type == Music.TYPE.TYPE_LIST) {
          return 6;
        } else if (type == Music.TYPE.TYPE_TITLE) {
          return 6;
        }
        return 0;
      }
    });
    ry.setLayoutManager(layoutManager);
    ry.addItemDecoration(new SpacesItemDecoration(2));

    // 填充数据
    mAdapter = new RecyclerAdapter(this, mList);
    mAdapter.setOnItemClickListener(this);
    ry.setAdapter(mAdapter);

  }

  @Override
  public void OnItemClick(int position) {
    String title = mList.get(position).title;
    Toast.makeText(this, title, Toast.LENGTH_SHORT).show();
  }
}

layout布局文件

这里的布局很简单,比如用到我们的正方形图片,长方形图片等,这里就不做代码贴出,详细可以查看源码

源码下载

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 本文旨在提供一套高效的面试方法,帮助企业在短时间内找到合适的产品经理。虽然观点较为直接,但其方法已被实践证明有效,尤其适用于初创公司和新项目的需求。 ... [详细]
  • 本文介绍了如何通过设置背景形状来轻松地为 Android 的 TextView 添加圆形边框。我们将详细讲解 XML 代码的配置,包括圆角、描边和填充等属性。 ... [详细]
  • 在使用STM32Cube进行定时器配置时,有时会遇到延时不准的问题。本文探讨了可能导致延时不准确的原因,并提供了解决方法和预防措施。 ... [详细]
  • Netflix利用Druid实现高效实时数据分析
    本文探讨了全球领先的在线娱乐公司Netflix如何通过采用Apache Druid,实现了高效的数据采集、处理和实时分析,从而显著提升了用户体验和业务决策的准确性。文章详细介绍了Netflix在系统架构、数据摄取、管理和查询方面的实践,并展示了Druid在大规模数据处理中的卓越性能。 ... [详细]
  • 深入理解Lucene搜索机制
    本文旨在帮助读者全面掌握Lucene搜索的编写步骤、核心API及其应用。通过详细解析Lucene的基本查询和查询解析器的使用方法,结合架构图和代码示例,带领读者深入了解Lucene搜索的工作流程。 ... [详细]
  • 本文介绍如何使用 Android 的 Canvas 和 View 组件创建一个简单的绘图板应用程序,支持触摸绘画和保存图片功能。 ... [详细]
  • DeepMind迁入谷歌伦敦新总部:人工智能研究的新里程碑
    谷歌旗下的人工智能研究机构DeepMind已正式入驻位于伦敦国王十字车站潘克拉斯广场6号的新总部。这座现代化的办公大楼不仅为DeepMind提供了宽敞的研究空间,也象征着谷歌对AI技术发展的高度重视。 ... [详细]
  • 在项目部署后,Node.js 进程可能会遇到不可预见的错误并崩溃。为了及时通知开发人员进行问题排查,我们可以利用 nodemailer 插件来发送邮件提醒。本文将详细介绍如何配置和使用 nodemailer 实现这一功能。 ... [详细]
  • 本文将探讨Java编程语言中对象和类的核心概念,帮助读者更好地理解和应用面向对象编程的思想。通过实际例子和代码演示,我们将揭示如何在Java中定义、创建和使用对象。 ... [详细]
  • 本文详细探讨了JavaScript中的作用域链和闭包机制,解释了它们的工作原理及其在实际编程中的应用。通过具体的代码示例,帮助读者更好地理解和掌握这些概念。 ... [详细]
  • 本文详细介绍了如何在 Android 开发中高效地管理和使用资源,包括本地资源和系统资源的访问方法。通过实例和代码片段,帮助开发者更好地理解和应用资源管理的最佳实践。 ... [详细]
  • Python 内存管理机制详解
    本文深入探讨了Python的内存管理机制,涵盖了垃圾回收、引用计数和内存池机制。通过具体示例和专业解释,帮助读者理解Python如何高效地管理和释放内存资源。 ... [详细]
  • C#设计模式学习笔记:观察者模式解析
    本文将探讨观察者模式的基本概念、应用场景及其在C#中的实现方法。通过借鉴《Head First Design Patterns》和维基百科等资源,详细介绍该模式的工作原理,并提供具体代码示例。 ... [详细]
  • 本文详细介绍了划分树这一数据结构,重点探讨了其在子树和中值计算中的应用及优化方法。 ... [详细]
  • Android Studio 安装与配置指南
    本教程详细介绍了如何下载并安装 Android Studio,包括设置 SDK 路径和优化启动性能的方法。通过这些步骤,您可以顺利地开始开发 Android 应用。 ... [详细]
author-avatar
月芽2502915393
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有