点击短信内容,获取短信内的号码和链接(2)
2015-12-03 17:09
295 查看
[转]点击短信内容,获取短信内的号码和链接(2)
转自:MrDemigod的专栏http://m.blog.csdn.net/blog/MrDemigod/12911659上次分析了点击短信产生对应号码和链接的主要方法,现在需要考虑的问题是,怎样才能将这些系统识别出来的号码和链接区分开,而不是一起显示。
分析过程:
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的操作。
转自:MrDemigod的专栏http://m.blog.csdn.net/blog/MrDemigod/12911659
相关文章推荐
- Struts2-ActionMethod
- Codeblocks 安装
- POST和GET
- android studio安装出现两个图标的问题
- 一个ios手势密码功能实现
- vnc远程桌面-桌面空白解决方案
- Git的基本原理 及 常用命令情景模拟
- C++ vector用法
- 关于http get和post的相关资料
- 利用shell脚本监控redis的使用内存
- 保存方法
- (五十七)指针、数组、指针算数
- (五十六)写了一个小的对战程序
- 正则表达式30分钟入门教程
- ubuntu 14.04 use Kinect v1
- IOS--UIProcessView
- Design Pattern ——Builder
- OC 实例变量(instance var)与属性(@property)的关系 isa指针
- Struts2-Path
- 软件工程与计算机科学