上次分析了点击短信产生对应号码和链接的主要方法,现在需要考虑的问题是,怎样才能将这些系统识别出来的号码和链接区分开,而不是一起显示。
分析过程:
1,首先焦点不能落在item上,这样通过item获取到的肯定是所有被autoLink的对象;
2,不能通过onItemClick来触发号码或者链接的菜单栏,甚至不能设置onItemClick事件,因为我们希望获得焦点的是item里面的link而不是item。
既然想到了焦点问题,我们不妨再深入一点,我不在onItemClick里面的onMessageListItemClick()里面来触发,只能通过link来触发,link又没法获取到焦点,这是怎么回事?
(1)查看xml不难发现,listView为了让item始终获得焦点将短信的textview焦点获取设置成了false(不仅仅是短信textview,item内的所有view的焦点获取都被设置成了false),那我们先把这个焦点获取设置成true,然后添加android:linksClickable="true" 再点击短信里面的链接,发现可以直接打开网页了,点击号码可以直接拨号了。
(2)但这不是我们想要的效果。我们还是先把上面两个属性改成原来的样子(为什么?因为考虑到短信的多选,item的焦点获取必须要保证)
既然不能在配置文件里面将短信内容TextView的焦点属性定为true,那我只能在短信会话ListView更新数据并列出的时候就让这些TextView可以获得焦点,这就要对adapter做处理了。
(3)上文我们分析到了MessageListView使用到的适配器是MessageListAdapter,分析MessageListAdapter我们惊奇的发现他居然连一个adapter最需要的getView()都没有!其实也是,他本意就不想我们去处理item里面的子view,何须getView。但是这里我们就是要获取子View,还要写他们的onClick事件,决不能少了getView()!
(4)开始重写getView():
public View getView(int position, View convertView, ViewGroup parent) {
View v = super.getView(position, convertView, parent);
TextView tv = (TextView) v.findViewById(R.id.text_view);
tv.setFocusableInTouchMode(true);
tv.setFocusable(true);
tv.setLinksClickable(true);
urls = tv.getUrls();
Log.i("Fred", "textView = " + tv.getText());
SpannableStringBuilder str = new SpannableStringBuilder(tv.getText());
Spannable sp = (Spannable) str;
SpannableStringBuilder style = new SpannableStringBuilder(tv.getText());
style.clearSpans();
for (URLSpan url : urls) {
MyURLSpan myURLSpan = new MyURLSpan(url.getURL());
style.setSpan(myURLSpan, sp.getSpanStart(url), sp.getSpanEnd(url),
Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
}
tv.setText(style);
return v;
}
getView()做的事情其实很简单,就是想法设法在短信内容TextView写入之前重新组合,变成链接和号码可点击的item中的子view。
(5)继续看看MyURLSpan这个类:
class MyURLSpan extends ClickableSpan {
private String mUrl;
private String mms;
private String phone;
MyURLSpan(String url) {
mUrl = url;
mms = "smsto:";
phone = "tel:";
}
@Override
public void onClick(View widget) {
// Toast.makeText(mContext," hello! "+mUrl,
// Toast.LENGTH_SHORT).show();
links = new ArrayList<String>();
mDefaultCountryIso = MmsApp.getApplication().getCurrentCountryIso();
if (mUrl.contains(phone)) {
links.add(mUrl);
links.add(mUrl.replaceAll(phone, mms));
} else
links.add(mUrl);
ArrayAdapter<String> adapter = new ArrayAdapter<String>(mContext,
android.R.layout.select_dialog_item, links) {
@Override
public View getView(int position, View convertView,
ViewGroup parent) {
View v = super.getView(position, convertView, parent);
TextView tv = (TextView) v;
String url = getItem(position).toString();
Uri uri = Uri.parse(url);
final String telPrefix = "tel:";
Drawable d = null;
try {
d = mContext.getPackageManager().getActivityIcon(
new Intent(Intent.ACTION_VIEW, uri));
} catch (android.content.pm.PackageManager.NameNotFoundException ex) {
}
if (d != null) {
d.setBounds(0, 0, d.getIntrinsicHeight(),
d.getIntrinsicHeight());
tv.setCompoundDrawablePadding(10);
tv.setCompoundDrawables(d, null, null, null);
} else {
if (url.startsWith(telPrefix)) {
d = mContext.getResources().getDrawable(
R.drawable.ic_launcher_phone);
d.setBounds(0, 0, d.getIntrinsicHeight(),
d.getIntrinsicHeight());
tv.setCompoundDrawablePadding(10);
tv.setCompoundDrawables(d, null, null, null);
} else {
tv.setCompoundDrawables(null, null, null, null);
}
}
final String smsPrefix = "smsto:";
final String mailPrefix = "mailto";
if (url.startsWith(telPrefix)) {
url = PhoneNumberUtils.formatNumber(
url.substring(telPrefix.length()),
mDefaultCountryIso);
if (url == null) {
url = getItem(position).toString().substring(
telPrefix.length());
}
} else if (url.startsWith(smsPrefix)) { url = PhoneNumberUtils.formatNumber(
url.substring(smsPrefix.length()),
mDefaultCountryIso);
if (url == null) {
url = getItem(position).toString().substring(
smsPrefix.length());
}
} else if (url.startsWith(mailPrefix)) {
MailTo mt = MailTo.parse(url);
url = mt.getTo();
}
tv.setText(url);
return v;
}
};
AlertDialog.Builder b = new AlertDialog.Builder(mContext);
DialogInterface.OnClickListener click = new DialogInterface.OnClickListener() {
@Override
public final void onClick(DialogInterface dialog, int which) {
if (which >= 0) {
Uri uri = Uri.parse(links.get(which));
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
intent.putExtra(Browser.EXTRA_APPLICATION_ID,
mContext.getPackageName());
if (links.get(which).startsWith("smsto:")) {
intent.setClassName(mContext,
"com.android.mms.ui.SendMessageToActivity");
}
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET);
mContext.startActivity(intent);
if (links.get(which).startsWith("smsto:")) {
intent.setClassName(mContext,
"com.android.mms.ui.SendMessageToActivity");
}
}
dialog.dismiss();
}
};
b.setTitle(R.string.select_link_title);
b.setCancelable(true);
b.setAdapter(adapter, click);
b.setNegativeButton(android.R.string.cancel,
new DialogInterface.OnClickListener() {
@Override
public final void onClick(DialogInterface dialog,
int which) {
dialog.dismiss();
}
});
b.show();
}
}
(6)好吧,不难看出,我这个类里面除了构造方法和一些数组处理,其他基本都是照抄的MessageListItem类的onMessageListItemClick()方法里面的内容。确实,他们获取链接或者号码然后生成菜单的方式确实就是一样的 =。=
(7)到了这里,基本的修改都已经完成了,在生成ListView的时候才修改TextView的焦点属性,让短信列表中号码和链接可以被点击。但总觉得还有不妥。我这样将MessageListAdapter类修改了,多了这个getView()方法,并且写入了onClick事件,会导致所有用到这个adapter的listView的ltem被注册这些事件,这不符合我们代码涉及的原则。所以我们新建一个类MessageListAdapterHistory继承MessageListAdapter,然后将关于getView()的部分移动到MessageListAdapterHistory中,然后只修改
mMsgListAdapter = new MessageListAdapterHistory();即可。也不会影响到其他列表(比如多选等)对item的操作。