Xamarin.Android开发实践(十四)

原文:Xamarin.Android开发实践(十四)

Xamarin.Android之ListView和Adapter

一、前言

如今不管任何应用都能够看到列表的存在,而本章我们将学习如何使用Xamarin去实现它,以及如何使用适配器和自定义适配器(本文中的适配器的主要内容就是将原始的数据转换成了能够供列表控件显示的项)。

 

二、简介适配器

在开始之前我们需要先了解下适配器,首先是提供给我们使用的适配器之间的关系:

下面我们将上面的适配器进行简单的介绍:

BaseAdapter:顾名思义,就是所以适配器的基类,但是我们不能将其实例化,因为它是一个虚类,一般我们都是继承该类并实现其中的方法,形成形成自定义的列表(大多数情况下我们都会使用到它)。

ArrayAdapterArrayAdapter<T>:就是专门用于将列表数据的适配器,该适配器内部已经实现了BaseAdapter的方法,所以我们只需要指定对应的数据项以及列表项资源即可。

CursorAdapter:上面的适配器只是用于列表数据,而该适配器则可以用在将数据库返回的结果集显示到列表中去,当然在难度上也要比上面的适配器高。

 

三、正文

 

1.简单列表

下面我们将利用ListActivityArrayAdapater<T>去实现一个简单的列表界面,下面为该示例的代码(MainActivity.cs):

 1     public class MainActivity : ListActivity
 2     {
 3         protected string[] items;
 4
 5         protected override void OnCreate(Bundle bundle)
 6         {
 7             base.OnCreate(bundle);
 8             items = new string[]{
 9                 "First","Second","Third"
10             };
11             ListAdapter = new ArrayAdapter<string>(this, Android.Resource.Layout.SimpleListItem1, items);
12         }
13
14         protected override void OnListItemClick(ListView l, View v, int position, long id)
15         {
16             var t = items[position];
17             Toast.MakeText(this, t, ToastLength.Short).Show();
18         }
19 }

这里有一个重要的关键词就是ListActivity,如果存在一个界面整个界面都是列表,那么我们就可以继承这个特殊的活动,并且可以通过ListView属性和ListAdapter去控制,同时还可以响应关于列表的事件。其中我们利用了ArrayAdapter给列表指定了一个适配器,而这个适配器的第一个参数是当前的上下文,第二个是列表中的项的界面,最后一个就是对应的数据了。

 

最后将显示如下的界面:

 

当我们点击不同的项后,还能看到底部显示了当前我们选择的项。这个功能就是在我们重写了OnListItemClick实现了,正如代码中所示,我们根据position获取指定的数据,然后通过Toast将其显示出来。

 

 

2.自定义一个适配器

简单介绍过如何使用适配器后,我们将开始学习如何利用BaseAdapter自定义一个适配器,能够促使我们理解适配器内部的工作原理,首先我们来看下笔者的写的代码:

 1         public class MyCustomeAdapter : BaseAdapter<string>
 2         {
 3             string[] items;
 4             Activity activity;
 5
 6             public MyCustomeAdapter(Activity context, string[] values)
 7                 : base()
 8             {
 9                 activity = context;
10                 items = values;
11             }
12
13             public override string this[int position]
14             {
15                 get { return items[position]; }
16             }
17
18             public override int Count
19             {
20                 get { return items.Length; }
21             }
22
23             public override long GetItemId(int position)
24             {
25                 return position;
26             }
27
28             public override View GetView(int position, View convertView, ViewGroup parent)
29             {
30                 View v = convertView;
31                 if (v == null)
32                     v = activity.LayoutInflater.Inflate(Android.Resource.Layout.SimpleListItem1, null);
33                 v.FindViewById<TextView>(Android.Resource.Id.Text1).Text = items[position];
34                 return v;
35             }
36         }

其中最主要的是GetView方法,ListView和ListActivity主要是通过调用该方法获取对应项的视图,然后将其添加为子控件,从而显示,如果我们需要显示例如淘宝等app那种复杂的列表就需要通过重写该方法达到,细心的读者可以发现笔者是先判断convertView是否为null,如果为null才通过Inflate方法重新实例化一个视图,关于这部分我们可以参考该文章

 

关于其他的方法相信大家靠名称就能够明白了,这里就不多做解释,具体的使用方法如下所示:

1         protected override void OnCreate(Bundle bundle)
2         {
3             base.OnCreate(bundle);
4             string[] items = new string[]{
5                 "First","Second","Third"
6             };
7             ListAdapter = new MyCustomeAdapter(this, items);
8         }

 

 

3.关键字索引

在手机的使用中,大家一定经常使用着电话簿,当我们点击右边的滚动条进行滑动的时候还会显示a-z的字母并且列表的内容也会根据这些字母的变化发生变化,将快速定位到联系人姓名为指定字母开头的位置,本节我们将会学习如何实现该功能。

 

提供这个功能我们需要实现一个接口,这个接口就是ISectionIndexer,下面是关于该接口的代码:

1     public interface ISectionIndexer : IJavaObject, IDisposable
2     {
3         int GetPositionForSection(int section);
4         int GetSectionForPosition(int position);
5         Object[] GetSections();
6     }

关于这些接口简单的介绍下:

GetPositionForSection:根据关键字的索引获取该关键字的起始数据索引。

GetSectionForPosition:根据数据索引获取关键字索引。

GetSections:返回关键字数组。

 

笔者为了能够节约时间,所以利用上节的示例代码,在MyCustomeAdapter中实现了ISectionIndexer接口,下面是笔者的代码:

 1 public class MyCustomeAdapter : BaseAdapter<string> , ISectionIndexer
 2         {
 3             string[] items;
 4             Activity activity;
 5
 6             Dictionary<string, int> alphaindex;
 7             Java.Lang.Object[] sectionsObjects;
 8             string[] sections;
 9
10             public MyCustomeAdapter(Activity context, string[] values)
11                 : base()
12             {
13                 activity = context;
14                 items = values;
15
16                 alphaindex = new Dictionary<string, int>();
17                 //获取每种关键字的起始数据索引
18                 for (int i = 0; i < items.Length; i++)
19                 {
20                     string key = items[i][0].ToString();
21                     if (!alphaindex.ContainsKey(key))
22                         alphaindex.Add(key, i);
23                 }
24
25                 //将关键字转换成数据
26                 sections = new string[alphaindex.Keys.Count];
27                 alphaindex.Keys.CopyTo(sections, 0);
28
29                 //将关键字转换成Java.Lang.String类型
30                 sectionsObjects = new Java.Lang.Object[alphaindex.Keys.Count];
31                 for (int i = 0; i < sections.Length; i++)
32                 {
33                     sectionsObjects[i] = new Java.Lang.String(sections[i]);
34                 }
35             }
36
37             public override string this[int position]
38             {
39                 get { return items[position]; }
40             }
41
42             public override int Count
43             {
44                 get { return items.Length; }
45             }
46
47             public override long GetItemId(int position)
48             {
49                 return position;
50             }
51
52             public override View GetView(int position, View convertView, ViewGroup parent)
53             {
54                 View v = convertView;
55                 if (v == null)
56                     v = activity.LayoutInflater.Inflate(Android.Resource.Layout.SimpleListItem1, null);
57                 v.FindViewById<TextView>(Android.Resource.Id.Text1).Text = items[position];
58                 return v;
59             }
60
61             public int GetPositionForSection(int section)
62             {
63                 //根据关键字索引获取关键字,然后在根据关键字从alphaindex获取对应的value,即该关键字的起始数据索引
64                 return alphaindex[sections[section]];
65             }
66
67             public int GetSectionForPosition(int position)
68             {
69                 int preposition = 0;
70                 //循环关键字
71                 for (int i = 0; i < sections.Length; i++)
72                 {
73                     //判断当前的索引是否在i所在关键字的范围内
74                     if (GetPositionForSection(i) > position)
75                         break;
76                     preposition = i;
77                 }
78                 return preposition;
79             }
80
81             public Java.Lang.Object[] GetSections()
82             {
83                 return sectionsObjects;
84             }
85         }

 

关于该接口方法中的实现可以看代码中的注释,在使用的时候也要写如下代码:

 1         protected override void OnCreate(Bundle bundle)
 2         {
 3             base.OnCreate(bundle);
 4             string[] items = new string[]{
 5                 "Apple","As","Banner",
 6                 "BMW","Beatf","Bad",
 7                 "Day","Dazy","Eat","Else",
 8                 "Feel","Fly","Go","Hourse"
 9             };
10             ListAdapter = new MyCustomeAdapter(this, items);
11             ListView.FastScrollEnabled = true;
12             ListView.FastScrollAlwaysVisible = true;
13         }

 

按照官方的说法其实只要将FastScrollEnabled设置为true即可,但是笔者在虚拟机中测试的时候并不会显示,所以笔者还要再将FastScrollAlwaysVisible一起设置为true才可以看到效果,下面为笔者的虚拟机上的截图:

 

 

4.Cursor列表

前面我们都是通过数组来填充列表的,但是实际情况下数据大多数都是来自于数据库,而数据库通过查询后将会得到ICursor类型的返回值,当然我们也可以通过将ICursor转换成数组或者键值对,但这些都是没有必要的,因为有自带的适配器(CursorAdapter)。

 

首先我们需要创建一个数据库,下面是笔者的代码,想学习数据库的可以参考本人的以下随笔:

《Xamarin.Android之SQLiteOpenHelper》

《Xamarin.Android之ContentProvider》

《Xamarin.Android之SQLite.NET ORM》

 

代码如下所示:

 1     public class TestDatabase : SQLiteOpenHelper
 2     {
 3         public TestDatabase(Context context):base(context,"testdatabase.db",null,1)
 4         {}
 5
 6         public override void OnCreate(SQLiteDatabase db)
 7         {
 8             db.ExecSQL("CREATE TABLE TestTable ( _id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,name TEXT NOT NULL)");
 9             db.ExecSQL("INSERT INTO TestTable (name) values('Vegetables')");
10             db.ExecSQL("INSERT INTO TestTable (name) values('Fruits')");
11             db.ExecSQL("INSERT INTO TestTable (name) values('Flower Buds')");
12             db.ExecSQL("INSERT INTO TestTable (name) values('Legumes')");
13             db.ExecSQL("INSERT INTO TestTable (name) values('Bulbs')");
14             db.ExecSQL("INSERT INTO TestTable (name) values('Tubers')");
15         }
16
17         public override void OnUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)
18         {
19
20         }
21 }

 

数据库创建部分比较简单,主要的重点是下面的代码。我们将实例化SimpleCursorAdapter并赋值给ListAdapter中去,代码如下所示:

 1     [Activity(Label = "CursorAdapter", MainLauncher = true, Icon = "@drawable/icon")]
 2     public class MainActivity : ListActivity
 3     {
 4         TestDatabase vdb;
 5         ICursor cursor;
 6
 7         protected override void OnCreate(Bundle bundle)
 8         {
 9             base.OnCreate(bundle);
10             vdb = new TestDatabase(this);
11             //通过SQL查询数据库
12             cursor = vdb.ReadableDatabase.RawQuery("SELECT * FROM TestTable", null);
13             StartManagingCursor(cursor);
14             //创建适配器
15             ListAdapter = new SimpleCursorAdapter(this, Android.Resource.Layout.SimpleListItem1, cursor,
16                 new string[] { "name" },  //指定使用结果集中的哪个字段数据
17                 new int[] { Android.Resource.Id.Text1 }); //填充到哪个控件中
18         }
19
20         protected override void OnDestroy()
21         {
22             base.OnDestroy();
23             StopManagingCursor(cursor);
24             cursor.Close();
25         }
26 }

通过上面的代码我们可以看到SimpleCursorAdapter相比之前的适配器,需要的参数更多。同时还需要对Cursor这个重要资源进行管理,对应的可以看到StartManagingCursorStopManagingCursor,希望读者在实际开发中一定要注意资源的释放,以上的结果笔者就不截图了。

 

5.自定义CursorAdapter

为了某些效果我们不一定非要继承BaseAdapter从而实现很多的方法,如果 数据是来自于数据库的我们还可以继承自CursorAdapter,这样我们能够节省实现很多的方法,从而只需要关注重要部分,下面我们继承 CursorAdapter来查看需要实现哪些方法:

 1     public class CustomeCursorAdapter : Android.Widget.CursorAdapter
 2     {
 3
 4         public override void BindView(Android.Views.View view, Android.Content.Context context, Android.Database.ICursor cursor)
 5         {
 6             throw new NotImplementedException();
 7         }
 8
 9         public override Android.Views.View NewView(Android.Content.Context context, Android.Database.ICursor cursor, Android.Views.ViewGroup parent)
10         {
11             throw new NotImplementedException();
12         }
13 }

可以看到我们只需要实现两个方法,相比BaseAdapter而言减少了很多,其中BindView就是将ICursor中的数据与View进行绑定,而NewView则是创建所需的界面。下面我们自定一个适配器实现和上一节一样的效果:

 

 1     public class CustomeCursorAdapter : Android.Widget.CursorAdapter
 2     {
 3         Activity context;
 4
 5         public CustomeCursorAdapter(Activity context, ICursor cursor)
 6             : base(context, cursor)
 7         {
 8             this.context = context;
 9         }
10
11         public override void BindView(Android.Views.View view, Android.Content.Context context, Android.Database.ICursor cursor)
12         {
13             TextView tv = view.FindViewById<TextView>(Android.Resource.Id.Text1);
14             tv.Text = cursor.GetString(1);
15         }
16
17         public override Android.Views.View NewView(Android.Content.Context context, Android.Database.ICursor cursor, Android.Views.ViewGroup parent)
18         {
19             return this.context.LayoutInflater.Inflate(Android.Resource.Layout.SimpleListItem1, parent, false);
20         }
21 }

 

 6.注意点

 需要读者需要使用系统自带的SimpleAdapter时,传递数据时候不能以Dictionary类型添加,必须以JavaDictionary,否则将会出现无法将其转换为java.util.map的错误。

时间: 2025-01-30 14:19:30

Xamarin.Android开发实践(十四)的相关文章

Xamarin.Android开发实践(十二)

原文:Xamarin.Android开发实践(十二) Xamarin.Android之ContentProvider 一.前言 掌握了如何使用SQLiteOpenHelper之后,我们就可以进行下一步的学习.本章我们将会学习如何使用ContentProvider来将数据库方面的操作封装起来,同时它还可以供其他应用访问并操作数据库.   二.概念 首先我们不会急于写代码,而是要搞懂如何利用ContentProvider对数 据库进行操作,因为我们不会直接操作数据库对象,而是通过URI来操作数据库.

Xamarin.Android开发实践(十八)

原文:Xamarin.Android开发实践(十八) Xamarin.Android之SlidingMenu 一.前言 有位网友在评论中希望能够出个在Xamarin.Android下实现SlidingMenu效果的随笔,刚好昨天在观看官网示例项目的时候也看到这个SlidingMenu,但是最终的效果并不是我们所期待的,至此笔者就在官方的论坛中寻找,最后也成功的寻找到的答案,下面笔者将带领带领大家实现SlidingMenu.   二.准备工作 实现SlidingMenu重点是需要一个第三方的类库,

Xamarin.Android开发实践(十六)

原文:Xamarin.Android开发实践(十六) Xamarin.Android之Fragment Walkthrough 利用Fragment设计能够兼容不同屏幕的应用 这里我们先围观下最后的成果图,给读者打打气:   普通手机上显示的结果:   在平板上显示的结果:   笔者要郑重声明下,虽然看似是两种不同的显示效果,但是同一个应用,而下面笔者将逐步教会大家如何利用Fragment制作出能够兼容不同屏幕的应用.   准备工作 创建一个项目是必不可少的,并且Android SDK的版本要在

Xamarin.Android开发实践(十)

原文:Xamarin.Android开发实践(十) Xamarin.Android之SQLiteOpenHelper 一.前言 在手机中进行网络连接不仅是耗时也是耗电的,而耗电却是致命的.所以我们就需要数 据库帮助我们存储离线数据,以便在用户未使用网络的情况下也可以能够使用应用的部分功能,而在需要网络连接的功能上采用提示方式,让用户决定是否打开网 络.而本节我们将会学习如何访问数据库以及提供基本的增删改查功能,并且使他们尽量的解耦.   二.数据库 Xamarin.Android下创建本地数据库

Xamarin.Android开发实践(十五)

原文:Xamarin.Android开发实践(十五) Xamarin.Android学习之应用程序首选项 一.前言 任何App都会存在设置界面,如果开发者利用普通控件并绑定监听事件保存设置,这 一过程会非常的枯燥,而且耗时.我们可以看到Android系统的设置界面里面的选项如此之多,是不是都是这样开发的呢?其实不然,Android已经给 我们提供了专门设计这一功能的技术,叫应用程序首选项,今天我们将学习如何使用他们来开发配置界面以及功能.   二.准备工作 首先需要理解的就是我们设置界面还是需要

Xamarin.Android开发实践(六)

原文:Xamarin.Android开发实践(六) Xamarin.Android通知详解 一.发送通知的机制 在日常的app应用中经常需要使用通知,因为服务.广播后台活动如果有事件需要通知用户,则需要通过通知栏显示,而在Xamarin.Android下的通知需要获取NotificationManager服务,而该服务需要通过GetSystemService获取,同时还要传递一个标识符.获取了通知管理器后我们就可以实例化Notification,然后再由NotificationManager发送

Xamarin.Android开发实践(十七)

原文:Xamarin.Android开发实践(十七) Xamarin.Android之定位 一.前言 打开我们手中的应用,可以发现越来越多的应用使用了定位,从而使我们的生活更加方便,所以本章我们将学习如何在Xamarin中进行定位的开发.     二.准备工作 因为我们的虚拟机是运行在电脑本地的,自然就没法进行定位了,但是我们可以借助DDMS这个工具帮助我们去调试.   首先要确定你的Android SDK所在的目录,读者可以通过以下方式找到: 工具->选项   然后读者打开该文件夹下的tool

Xamarin.Android开发实践(二)

原文:Xamarin.Android开发实践(二) 一.准备 开始学习本教程前必须先完成该教程http://www.cnblogs.com/yaozhenfa/p/xamarin_android_quickstart.html 否则将无法继续.   二.界面 1.打开Resources/layout/Main.axml文件,并在Call Button下方继续加入一个按钮,并设置其id为@+id/CallHistoryButton同时设置Text为@string /callHistory(这个其实

Xamarin.Android开发实践(五)

原文:Xamarin.Android开发实践(五) 一.服务的生命周期 服务与活动一样,在它的整个生命周期中存在着一些事件,下图可以很好解释整个过程以及涉及到的方法: 在真实的使用中,Service来还包含一个OnBind方法,并且必须要使用该方法,但是只要返回NULL即可,除非当前服务是一个绑定服务,那么就要返回实现了IBinder的实例.   二.回调方法的总结 上图中涉及到了几个方法,下面将做简单的介绍: OnCreate:只会在服务第一次开启的时候调用,主要负责一些初始化代码 OnSta