声明:学习的书籍《Android应用开发揭秘》,这里记录学习该书籍的日志,引用的相关代码与总结描述,没有商业的用途,完全是自我学习的一个记录,刚刚学习不可避免会出现很多问题,若是有错误还请大家多多批评。
继续前面的学习
一、TextView文本框(省略)。
二、列表ListView
针对列表的事件:
(1)当鼠标滚动时会触发:setOnItemSelectedListener事件;
(2)点击时触发:setOnItemClickListener事件
实例分析:对电话本联系人的信息进行操作。 布局格式:LinearLayout中显示ListView
【注意】是同个ListAdapter将获得的电话本数据与ListView关联起来,将ListAdapter添加到ListView中。
【遇到问题困难】:该实例遇到很多困难,因为才是学习Android的第2天,很多内容还不了解,比如Android下数据获取与绑定,相关参数与方法的使用,很多需要学习与掌握,查看API以及,在网上查找了很多资料。
1. 联系人与电话的获取,1.5以后API发生变化,所以获取方式就要修改,参考代码。
2. 启动模拟器时,弹出SDL_app:emulator.exe 应用程序错误:
解决方法:删除该模拟器的SD卡的大小。
具体是什么原因,暂时不是太清楚,后面深入学习会继续关注该问题。
3. 在用android日志LogCat的时候老是弹出一个窗口,内容为:"Copy" did not complete normally. Please see the log for more information.
Argument not valid
在网上也找到了答案:退出有道词典,或者划词功能就可以了。如果不是有道词典关闭翻译软件的划词功能。(的确很郁闷,存在这个问题)
以下是该例子的实例代码
public class Activity01 extends Activity {
private static final String TAG = "TestListView";
ListView listView;
ListAdapter adapter;
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
LinearLayout linearLayout = new LinearLayout(this);
linearLayout.setOrientation(LinearLayout.VERTICAL); //设置布局LinearLayout的属性
linearLayout.setBackgroundColor(Color.BLACK);
LinearLayout.LayoutParams param = new LinearLayout.LayoutParams(
LinearLayout.LayoutParams.FILL_PARENT,
LinearLayout.LayoutParams.WRAP_CONTENT); //设置该LinearLayout下的子控件的布局样式
listView = new ListView(this);
listView.setBackgroundColor(Color.BLACK);
linearLayout.addView(listView, param); // 添加listView到linearLayout布局
this.setContentView(linearLayout);
ArrayList<HashMap<String, Object>> listItem = new ArrayList<HashMap<String, Object>>();
/*Cursor是个Interface,API中介绍:A mock Cursor class that isolates the test code from real Cursor implementation.
SQLiteCursor A Cursor implementation that exposes results from a query on a SQLiteDatabase.
【注意】:这里获取联系人的信息的光标对象,注意sdk2.0以后,API发生调整,People接口由ContactsContract.Contacts代替,
自己参考《Android应用开发揭秘》实例4-4运行报错,书中实例代码:
// ListAdapter是ListView和后台数据的桥梁
ListAdapter adapter = new SimpleCursorAdapter(this,
// 定义List中每一行的显示模板
// 表示每一行包含两个数据项
android.R.layout.simple_list_item_2,
// 数据库的Cursor对象
cur,
// 从数据库的NAME和NUMBER两列中取数据
new String[] { PhoneLookup.DISPLAY_NAME, PhoneLookup.NUMBER },
// 与NAME和NUMBER对应的Views
new int[] { android.R.id.text1, android.R.id.text2 });
运行上述提取方式会报'number'列不存在,说明在2.2的版本下PhoneLookup.NUMBER这个获取方式不对
*/
Cursor cursor = getContentResolver().query(
ContactsContract.Contacts.CONTENT_URI, null, null, null, null);
while (cursor.moveToNext()) {
HashMap<String, Object> map = new HashMap<String, Object>();
String phoneName = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME));
map.put("ItemTitle", phoneName);// 联系人姓名
String contactId = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts._ID));//获取联系人行数据ID
String hasPhone = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts.HAS_PHONE_NUMBER));
//HAS_PHONE_NUMBER:一个标示用来说明:是否至少有一个电话. "1" 为真, "0" 假.
if (hasPhone.compareTo("1") == 0) {
Cursor phones = getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI,null,
ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = " + contactId, null, null);
/*方法android.content.ContentResolver.query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)
其中第三个参数:String selection,作用如API:A filter declaring which rows to return, formatted as an SQL WHERE clause (excluding the WHERE itself). Passing null will return all rows for the given URI
作为条件过滤器,类似于一个SQL语句中where子句,若为空,则返回所有行
*/
while (phones.moveToNext()) {
//注意:ContactsContract.CommonDataKinds.Phone.NUMBER与之前sdk版本的获取方式
String phoneNumber = phones.getString(phones.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
String phoneTpye = phones.getString(phones.getColumnIndex(ContactsContract.CommonDataKinds.Phone.TYPE));
map.put("ItemText", phoneNumber); // 多个号码如何处理
Log.d(TAG, "testNum=" + phoneNumber + " type:" + phoneTpye);
}
phones.close();
}
Cursor emails = getContentResolver().query(ContactsContract.CommonDataKinds.Email.CONTENT_URI,null,
ContactsContract.CommonDataKinds.Email.CONTACT_ID + " = "+ contactId, null, null);
while (emails.moveToNext()) {
String emailAddress = emails.getString(emails.getColumnIndex(ContactsContract.CommonDataKinds.Email.DATA));
String emailType = emails.getString(emails.getColumnIndex(ContactsContract.CommonDataKinds.Email.TYPE));
Log.d(TAG, "testEmail=" + emailAddress + " type:" + emailType);
}
emails.close();
listItem.add(map);
}
// 生成适配器的Item和动态数组对应的元素
SimpleAdapter listItemAdapter = new SimpleAdapter(this, listItem,// 数据源
android.R.layout.simple_list_item_2,// ListItem的XML实现
// 动态数组与ImageItem对应的子项
new String[] { "ItemTitle", "ItemText" },
// ImageItem的XML文件里面的一个ImageView,两个TextView ID
new int[] { android.R.id.text1, android.R.id.text2 });
listView.setAdapter(listItemAdapter);
cursor.close();
/* 为m_ListView视图添加setOnItemSelectedListener监听 */
listView.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
public void onItemSelected(AdapterView<?> arg0, View arg1, int arg2, long arg3){
DisplayToast("滚动到第"+Long.toString(arg0.getSelectedItemId())+"项");
}
public void onNothingSelected(AdapterView<?> arg0){
//没有选中
}
});
/* 为m_ListView视图添加setOnItemClickListener监听 */
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3){
//于对选中的项进行处理
DisplayToast("选中了第"+Integer.toString(arg2+1)+"项");
}
});
}
private void DisplayToast(String str) {
Toast.makeText(this, str, Toast.LENGTH_SHORT).show();
}
}
效果:(左图为选择事件,右图为点击事件)
当然也可以把选择事件setOnItemSelectedListener与点击事件setOnItemClickListener自己修改,如网上例子给出,选择事件为发送短信,点击事件为拨打电话。
listView.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
public void onItemSelected(AdapterView<?> arg0, View arg1,int arg2, long arg3) {
// TODO Auto-generated method stub
// openToast("滚动到:"+arg0.getSelectedItemId());
// 短信发送
setTitle("选择" + arg2 + "项目");
openToast("选择" + arg0.getSelectedItemId() + "项目");
RelativeLayout lr = (RelativeLayout) arg1;
TextView mText = (TextView) lr.getChildAt(1);
openToast(mText.getText().toString());
String number = mText.getText().toString();
Log.d(TAG, "number=" + number);
// 判断电话号码的有效性
if (PhoneNumberUtils.isGlobalPhoneNumber(number)) {
Intent intent = new Intent(Intent.ACTION_SENDTO, Uri.parse("smsto://" + number));
intent.putExtra("sms_body", "The SMS text");
startActivity(intent);
}
}
public void onNothingSelected(AdapterView<?> arg0) {
// TODO Auto-generated method stub
}
});
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
public void onItemClick(AdapterView<?> arg0, View arg1,
int position, long arg3) {
// TODO Auto-generated method stub
// openToast("Click"+Integer.toString(position+1)+"项目");
RelativeLayout lr = (RelativeLayout) arg1;
TextView mText = (TextView) lr.getChildAt(1);
openToast(mText.getText().toString());
String number = mText.getText().toString();
Log.d(TAG, "number=" + number);
// 判断电话号码的有效性
if (PhoneNumberUtils.isGlobalPhoneNumber(number)) {
Intent intent = new Intent(Intent.ACTION_DIAL, Uri.parse("tel://" + number));
startActivity(intent);
}
}
});
通过两天的学习才真正解决了ListView控件的学习,虽然时间比较长,还是学到很多东西,继续加油。
学习到P63页