Android RecyclerView,Android CardView示例教程

时间:2020-02-23 14:29:11  来源:igfitidea点击:

Android RecyclerView和Android CardView通过Material Design在Android Lollipop中引入。
对于那些不了解Material Design的用户,它提供了全面的UI Widgets教程,自Android 5.0以来就引入了该教程,它提高了应用程序的视觉吸引力。

Android RecyclerView

Android RecyclerView是ListView的更高级,强大和灵活的版本。
Android RecyclerView与ListView相似,除了它迫使我们使用RecyclerView.ViewHolder类来保存不是ListView中的强制项的元素。

顾名思义,通过循环使用列表中的项目,Android RecyclerView可以在上下滚动时重用单元格。
RecyclerView的另一个改进是,它允许我们在运行时动态设置LayoutManager,而ListView仅在垂直滚动列表中可用。
RecyclerView允许我们在运行时设置以下类型的布局。

  • LinearLayoutManager:它支持垂直和水平列表
  • StaggeredLayoutManager:它支持交错列表
  • GridLayoutManager:它支持显示网格,如之前的GalleryView所示

Android RecyclerView类

  • 与ListViews不同,RecyclerView.ItemAnimator类为动画视图提供了更好的支持。

  • RecyclerView.ItemDecorator类在添加边框和分隔线时提供了更好的支持,从而为我们提供了巨大的控制权

因此,与ListView相比,RecyclerView更可自定义,并为用户提供了更好的控制。
RecyclerView在支持库中可用。
因此,我们需要修改gradle脚本以添加以下依赖关系。

dependencies {
     compile 'com.android.support:recyclerview-v7:21.0.0-rc1'
 }

Android CardView

Android CardView UI组件显示卡内的信息。
此组件通常用于显示联系信息。
该组件在另一个支持库中可用,因此我们也必须添加其依赖项。

dependencies {
      compile 'com.android.support:cardview-v7:21.0.0-rc1'
      compile 'com.android.support:recyclerview-v7:21.0.0-rc1'
 }

Android CardView小部件允许我们控制背景颜色,阴影,角半径,高程等。
要在XML中使用自定义属性,我们需要在父布局中添加以下名称空间声明。
以下是名称空间声明,其中包含我们项目中的某些属性。

<android.support.v7.widget.CardView
      android:id="@+id/card_view"
      xmlns:card_view="https://schemas.android.com/apk/res-auto"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      card_view:cardBackgroundColor="@color/grey_300"
      card_view:cardCornerRadius="10dp"
      card_view:cardElevation="5dp"
      card_view:cardUseCompatPadding="true">

上面使用的重要属性是:

  • card_view:cardCornerRadius:用于设置布局中的拐角半径
  • card_view:cardBackgroundColor:用于设置视图的背景色

在示例项目中,我们将添加一个RecyclerView来显示CardViews列表,其中包含Android版本名称和编号以及示例徽标。
CardViewonclick已编程为从列表中删除该Card。
我们在操作列中添加了一个菜单选项,以便按顺序重新添加已删除的卡片。

注意:徽标图像是从Google随机获取的。
因此大小会有所不同。

Android RecyclerView和CardView示例

该项目由一个显示" RecyclerView"的" MainActivity"组成。
CardView从CustomAdapter类添加到RecyclerView。
DataModel用于通过getter检索每个CardView的数据。
MyData类保存textviews和drawable及其ID的数组。

Android RecyclerView和CardView示例代码

" activity_main.xml"将RecyclerView保留在RelativeLayout中,如下所示。

activity_main.xml代码:

<RelativeLayout xmlns:android="https://schemas.android.com/apk/res/android"
  xmlns:tools="https://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:paddingBottom="@dimen/activity_vertical_margin"
  android:paddingLeft="@dimen/activity_horizontal_margin"
  android:paddingRight="@dimen/activity_horizontal_margin"
  android:paddingTop="@dimen/activity_vertical_margin"
  tools:context=".MainActivity"
  android:background="@color/grey_300"
  >

  <android.support.v7.widget.RecyclerView
      android:id="@+id/my_recycler_view"
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:scrollbars="vertical"
      

</RelativeLayout>

Android CardView布局定义如下:

cards_layout.xml代码:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="https://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical"
  android:tag="cards main container">

  <android.support.v7.widget.CardView
      android:id="@+id/card_view"
      xmlns:card_view="https://schemas.android.com/apk/res-auto"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      card_view:cardBackgroundColor="@color/color_white"
      card_view:cardCornerRadius="10dp"
      card_view:cardElevation="5dp"
      card_view:cardUseCompatPadding="true">

      <LinearLayout
          android:layout_width="match_parent"
          android:layout_height="wrap_content"
          android:orientation="horizontal"
          >

          <ImageView
              android:id="@+id/imageView"
              android:tag="image_tag"
              android:layout_width="0dp"
              android:layout_height="wrap_content"
              android:layout_margin="5dp"
              android:layout_weight="1"
              android:src="@drawable/ic_launcher"

          <LinearLayout
              android:layout_width="0dp"
              android:layout_height="wrap_content"
              android:layout_marginTop="12dp"
              android:layout_weight="2"
              android:orientation="vertical"
              >

              <TextView
                  android:id="@+id/textViewName"
                  android:layout_width="wrap_content"
                  android:layout_height="wrap_content"
                  android:layout_gravity="center_horizontal"
                  android:layout_marginTop="10dp"
                  android:text="Android Name"
                  android:textAppearance="?android:attr/textAppearanceLarge"

              <TextView
                  android:id="@+id/textViewVersion"
                  android:layout_width="wrap_content"
                  android:layout_height="wrap_content"
                  android:layout_gravity="center_horizontal"
                  android:layout_marginTop="10dp"

                  android:text="Android Version"
                  android:textAppearance="?android:attr/textAppearanceMedium"

          </LinearLayout>
      </LinearLayout>

  </android.support.v7.widget.CardView>

</LinearLayout>

Android CardView在嵌套线性布局中包含一个ImageView和两个TextView。
menu_main.xml包含一个单项,用于添加回去的卡。

menu_main.xml代码:

<menu xmlns:android="https://schemas.android.com/apk/res/android"
  xmlns:app="https://schemas.android.com/apk/res-auto"
  xmlns:tools="https://schemas.android.com/tools"
  tools:context=".MainActivity">
  <item android:id="@+id/add_item"
      android:title="Add"
      android:orderInCategory="100"
      app:showAsAction="always"
</menu>

MainActivity.java类的定义如下:

package com.theitroad.recyclerviewcardview;

import android.content.Context;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.DefaultItemAnimator;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.TextView;
import android.widget.Toast;

import java.util.ArrayList;

public class MainActivity extends AppCompatActivity {

  private static RecyclerView.Adapter adapter;
  private RecyclerView.LayoutManager layoutManager;
  private static RecyclerView recyclerView;
  private static ArrayList<DataModel> data;
  static View.OnClickListener myOnClickListener;
  private static ArrayList<Integer> removedItems;

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

      myOnClickListener = new MyOnClickListener(this);

      recyclerView = (RecyclerView) findViewById(R.id.my_recycler_view);
      recyclerView.setHasFixedSize(true);

      layoutManager = new LinearLayoutManager(this);
      recyclerView.setLayoutManager(layoutManager);
      recyclerView.setItemAnimator(new DefaultItemAnimator());

      data = new ArrayList<DataModel>();
      for (int i = 0; i < MyData.nameArray.length; i++) {
          data.add(new DataModel(
                  MyData.nameArray[i],
                  MyData.versionArray[i],
                  MyData.id_[i],
                  MyData.drawableArray[i]
          ));
      }

      removedItems = new ArrayList<Integer>();

      adapter = new CustomAdapter(data);
      recyclerView.setAdapter(adapter);
  }

  private static class MyOnClickListener implements View.OnClickListener {

      private final Context context;

      private MyOnClickListener(Context context) {
          this.context = context;
      }

      @Override
      public void onClick(View v) {
          removeItem(v);
      }

      private void removeItem(View v) {
          int selectedItemPosition = recyclerView.getChildPosition(v);
          RecyclerView.ViewHolder viewHolder
                  = recyclerView.findViewHolderForPosition(selectedItemPosition);
          TextView textViewName
                  = (TextView) viewHolder.itemView.findViewById(R.id.textViewName);
          String selectedName = (String) textViewName.getText();
          int selectedItemId = -1;
          for (int i = 0; i < MyData.nameArray.length; i++) {
              if (selectedName.equals(MyData.nameArray[i])) {
                  selectedItemId = MyData.id_[i];
              }
          }
          removedItems.add(selectedItemId);
          data.remove(selectedItemPosition);
          adapter.notifyItemRemoved(selectedItemPosition);
      }
  }

  @Override
  public boolean onCreateOptionsMenu(Menu menu) {
      super.onCreateOptionsMenu(menu);
      getMenuInflater().inflate(R.menu.menu_main, menu);
      return true;
  }

  @Override
  public boolean onOptionsItemSelected(MenuItem item) {
      super.onOptionsItemSelected(item);
      if (item.getItemId() == R.id.add_item) {
         //check if any items to add
          if (removedItems.size() != 0) {
              addRemovedItemToList();
          } else {
              Toast.makeText(this, "Nothing to add", Toast.LENGTH_SHORT).show();
          }
      }
      return true;
  }

  private void addRemovedItemToList() {
      int addItemAtListPosition = 3;
      data.add(addItemAtListPosition, new DataModel(
              MyData.nameArray[removedItems.get(0)],
              MyData.versionArray[removedItems.get(0)],
              MyData.id_[removedItems.get(0)],
              MyData.drawableArray[removedItems.get(0)]
      ));
      adapter.notifyItemInserted(addItemAtListPosition);
      removedItems.remove(0);
  }
}

从监听器方法中调用removeItems()方法以删除所单击的CardView。
其各自的ID存储在数组中以供以后检索。
为了稍后添加视图,我们实现了另一个名为addRemovedItemToList()的方法。
在这种方法中,我们将该视图添加到列表中的预定义位置,然后从" removedItems"数组中删除其ID。
在两种情况下都会通知CustomAdapter。

CustomeAdapter.java类的定义如下:

package com.theitroad.recyclerviewcardview;

import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;

import java.util.ArrayList;

public class CustomAdapter extends RecyclerView.Adapter<CustomAdapter.MyViewHolder> {

  private ArrayList<DataModel> dataSet;

  public static class MyViewHolder extends RecyclerView.ViewHolder {

      TextView textViewName;
      TextView textViewVersion;
      ImageView imageViewIcon;

      public MyViewHolder(View itemView) {
          super(itemView);
          this.textViewName = (TextView) itemView.findViewById(R.id.textViewName);
          this.textViewVersion = (TextView) itemView.findViewById(R.id.textViewVersion);
          this.imageViewIcon = (ImageView) itemView.findViewById(R.id.imageView);
      }
  }

  public CustomAdapter(ArrayList<DataModel> data) {
      this.dataSet = data;
  }

  @Override
  public MyViewHolder onCreateViewHolder(ViewGroup parent,
                                         int viewType) {
      View view = LayoutInflater.from(parent.getContext())
              .inflate(R.layout.cards_layout, parent, false);

      view.setOnClickListener(MainActivity.myOnClickListener);

      MyViewHolder myViewHolder = new MyViewHolder(view);
      return myViewHolder;
  }

  @Override
  public void onBindViewHolder(final MyViewHolder holder, final int listPosition) {

      TextView textViewName = holder.textViewName;
      TextView textViewVersion = holder.textViewVersion;
      ImageView imageView = holder.imageViewIcon;

      textViewName.setText(dataSet.get(listPosition).getName());
      textViewVersion.setText(dataSet.get(listPosition).getVersion());
      imageView.setImageResource(dataSet.get(listPosition).getImage());
  }

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

在上面的代码中,我们通过扩展RecyclerView.ViewHolder实现了自己的ViewHolder。
该视图是从我们在layouts目录中定义的cards_layout.xml中放大的。
以下代码段将MainActivity中的onClickListener附加到此视图。

view.setOnClickListener(MainActivity.myOnClickListener);

ArrayList将所有数据以DataModel类对象的形式存储在ArrayList中,并将它们添加到列表中的各个卡中。

下面给出了包含该应用程序特定数据的DataModel.javaMyData.java类:

package com.theitroad.recyclerviewcardview;

public class DataModel {

  String name;
  String version;
  int id_;
  int image;

  public DataModel(String name, String version, int id_, int image) {
      this.name = name;
      this.version = version;
      this.id_ = id_;
      this.image=image;
  }

  public String getName() {
      return name;
  }

  public String getVersion() {
      return version;
  }

  public int getImage() {
      return image;
  }

  public int getId() {
      return id_;
  }
}
package com.theitroad.recyclerviewcardview;

public class MyData {

  static String[] nameArray = {"Cupcake", "Donut", "Eclair", "Froyo", "Gingerbread", "Honeycomb", "Ice Cream Sandwich","JellyBean", "Kitkat", "Lollipop", "Marshmallow"};
  static String[] versionArray = {"1.5", "1.6", "2.0-2.1", "2.2-2.2.3", "2.3-2.3.7", "3.0-3.2.6", "4.0-4.0.4", "4.1-4.3.1", "4.4-4.4.4", "5.0-5.1.1","6.0-6.0.1"};

  static Integer[] drawableArray = {R.drawable.cupcake, R.drawable.donut, R.drawable.eclair,
          R.drawable.froyo, R.drawable.gingerbread, R.drawable.honeycomb, R.drawable.ics,
          R.drawable.jellybean, R.drawable.kitkat, R.drawable.lollipop,R.drawable.marsh};

  static Integer[] id_ = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
}

以下是我们的android RecyclerView和CardView示例应用程序产生的输出。

如您所见,被删除的项目始终添加在第三个索引(列表中的第四个位置)上