Axure RP已经成为成功设计Web原型的重要工具,也是产品经理使用的最多的原型工具,虽然Axure的优势在Web原型,但是用过导入一些 组件也可以方便的设计手机原型和交互效果,这对于使用Axure已经娴熟的人来说,基本不需要任何学习成本,所以也是一种比较推荐的工具。
http://www.axure.com/
SMS管理
[功能]
1. 收信箱:显示所有收到的信息 且实时显示 即:当有新信息收到 能自动刷新显示
2. 发信箱:显示所有已发信息 同上
3. 编写新信息: 鉴于一些问题 打算不自行定义 而只通过Intent调用系统的
[原理]
1. 通过目标Uri显示收信箱 发信箱 目标Uri:content://sms/inbox content://sms/sent
2. 实时刷新:一个办法是开辟thread 定时查询目标Uri 显示之 但会带来一些效能影响 所以决定使用ContentObserve监听目标Uri 当有变动 由ContentObserve通知注册方 该Uri:content://sms
3. 注意:ContentObserve不能监听: content://sms/inbox & content://sms/sent 而只能监听content://sms
[代码 步骤]
1. 定义SMSObserver 用于监听目标 并通过Handle通知注册方
public class SMSObserver extends ContentObserver { public final static int SMS_CHANGE = 0; Handler handle; public SMSObserver(Handler h) { super(h); // TODO Auto-generated constructor stub handle = h; } public void onChange(boolean selfChange) { // TODO Auto-generated method stub super.onChange(selfChange); //notify SMSInbox & SMSSent handle.sendEmptyMessage(SMS_CHANGE); } }
2. 定义注册方:SMSInbox 鉴于SMSSent与其原理类似 故打算以SMSInbox为例
> 2.1. 显示当前所有收信箱 并与ListView适配
lv = (ListView)findViewById(R.id.list); cursor = getContentResolver().query(Uri.parse("content://sms/inbox"), null, null, null, null); adapter = new ItemAdapter(this); lv.setAdapter(adapter);
> 2.2. 定义Handle 用于接受变动 并注册与ContentObserve 当接到通知后 查询目标Uri 并刷新显示
handler = new Handler(){ public void handleMessage(Message msg){ if(msg.what == SMSObserver.SMS_CHANGE){ cursor = getContentResolver().query(Uri.parse("content://sms/inbox"), null, null, null, null); adapter.notifyDataSetChanged(); } } }; sObserver = new SMSObserver(handler); this.getContentResolver().registerContentObserver(Uri.parse("content://sms"), true, sObserver);
> 2.3. SMSInbox 仅用于显示 收信箱 故定义 SMSDetails extends Activity 用于详细显示 sms信息
- 2.3.1. 定义布局:details.xml
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <TextView android:id="@+id/detailsNumber" android:layout_width="fill_parent" android:layout_height="wrap_content" /> <TextView android:id="@+id/detailsBody" android:layout_width="fill_parent" android:layout_height="200dip" /> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="horizontal" android:layout_width="300dip" android:layout_height="wrap_content" > <Button android:id="@+id/detailsReply" android:layout_width="100dip" android:layout_height="wrap_content" android:layout_gravity="left" android:text="回复" /> <Button android:id="@+id/detailsOK" android:layout_width="100dip" android:layout_height="wrap_content" android:layout_gravity="right" android:text="确定" /> </LinearLayout> </LinearLayout>
- 2.3.2. 其中2个TextView 分别显示信息地址和正文 2个Button 一个用于关闭当前窗口 一个用于短信回复 且自动填充 收信人地址
public class SMSDetails extends Activity { TextView textNumber,textBody; Button btnOK,btnReply; OnClickListener cl; String address,body; /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.details); textNumber = (TextView)findViewById(R.id.detailsNumber); textBody = (TextView)findViewById(R.id.detailsBody); btnReply = (Button)findViewById(R.id.detailsReply); btnOK = (Button)findViewById(R.id.detailsOK); Intent i = this.getIntent(); Bundle bundle = i.getExtras(); address = bundle.getString("address"); body = bundle.getString("body"); textNumber.setText("from:"+address); textBody.setText("message body:\n"+body); cl = new OnClickListener(){ @Override public void onClick(View arg0) { // TODO Auto-generated method stub switch(arg0.getId()){ case R.id.detailsReply: sendGoNativeNew(address); break; case R.id.detailsOK: sendBack(); break; } } }; btnReply.setOnClickListener(cl); btnOK.setOnClickListener(cl); } public void sendGoNativeNew(String address){ //native send sms app Intent sendIntent = new Intent(Intent.ACTION_SENDTO, Uri.parse("sms://")); //auto fill "address" sendIntent.putExtra("address", address); startActivity(sendIntent); } public void sendBack(){ Intent i = new Intent(); this.setResult(RESULT_OK, i); this.finish(); } }
- 2.3.3. 点击SMSInbox 某项 跳转到SMSDetails
lv.setOnItemClickListener(new OnItemClickListener(){ @Override public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) { // TODO Auto-generated method stub cursor.moveToPosition(arg2); String body = cursor.getString(cursor.getColumnIndexOrThrow("body")).toString(); String address = cursor.getString(cursor.getColumnIndexOrThrow("address")).toString(); Bundle b = new Bundle(); b.putString("body", body); b.putString("address", address); Intent intent = new Intent(SMSInbox.this,SMSDetails.class); intent.putExtras(b); startActivity(intent); } });
- 2.3.4. 其中 item.xml 用于定义子项布局
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="horizontal" android:layout_width="wrap_content" android:layout_height="wrap_content" > <ImageView android:id="@+id/body" android:layout_width="wrap_content" android:layout_height="wrap_content" /> <TextView android:id="@+id/num" android:layout_width="wrap_content" android:layout_height="wrap_content" android:paddingLeft="5dip" /> </LinearLayout> <TextView android:id="@+id/body" android:layout_width="fill_parent" android:layout_height="wrap_content" android:paddingLeft="10dip" /> </LinearLayout>
3. 鉴于SMSSent与SMSInbox大同小异 故不再细说 仅补上代码
public class SMSSent extends Activity { ListView lv; Cursor cursor; ItemAdapter adapter; SMSObserver sObserver; Handler handler; /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.list); setTitle(R.string.sent); lv = (ListView)findViewById(R.id.list); cursor = getContentResolver().query(Uri.parse("content://sms/sent"), null, null, null, null); adapter = new ItemAdapter(this); lv.setAdapter(adapter); lv.setOnItemClickListener(new OnItemClickListener(){ @Override public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) { // TODO Auto-generated method stub cursor.moveToPosition(arg2); String body = cursor.getString(cursor.getColumnIndexOrThrow("body")).toString(); String address = cursor.getString(cursor.getColumnIndexOrThrow("address")).toString(); Bundle b = new Bundle(); b.putString("body", body); b.putString("address", address); Intent intent = new Intent(SMSSent.this,SMSDetails.class); intent.putExtras(b); startActivity(intent); } }); //register SMSObserve handler = new Handler(){ public void handleMessage(Message msg){ if(msg.what == SMSObserver.SMS_CHANGE){ cursor = getContentResolver().query(Uri.parse("content://sms/sent"), null, null, null, null); adapter.notifyDataSetChanged(); } } }; sObserver = new SMSObserver(handler); this.getContentResolver().registerContentObserver(Uri.parse("content://sms"), true, sObserver); } public class ItemAdapter extends BaseAdapter { Activity activity; public ItemAdapter(Activity a){ activity = a; } @Override public int getCount() { // TODO Auto-generated method stub return cursor.getCount(); } @Override public Object getItem(int arg0) { // TODO Auto-generated method stub return cursor.getString(arg0); } @Override public long getItemId(int arg0) { // TODO Auto-generated method stub return arg0; } @Override public View getView(int arg0, View arg1, ViewGroup arg2) { // TODO Auto-generated method stub return composeItem(arg0); } private View composeItem(int position){ cursor.moveToPosition(position); String body = cursor.getString(cursor.getColumnIndexOrThrow("body")).toString(); String number = cursor.getString(cursor.getColumnIndexOrThrow("address")).toString(); LinearLayout item = (LinearLayout)activity.getLayoutInflater().inflate(R.layout.item, null); LinearLayout l1 = (LinearLayout)item.getChildAt(0); TextView tbody = (TextView)item.getChildAt(1); if(tbody != null){ tbody.setText(body); } TextView tnum = (TextView)l1.getChildAt(1); if(tnum != null){ tnum.setText(number); } ImageView image = (ImageView)l1.getChildAt(0); image.setImageResource(R.drawable.message); return item; } } }
4. emulator 运行截图
> 4.1. SMSInbox:
- 4.1.1. 通过telnet localhost 5554 登录emulator 通过 sms send 123 hello to 123 模拟发送短信
- 4.1.2. 短信发送记录为:
Android Console: type 'help' for a list of commands OK sms send 12 ds OK sms send 23 hi to 23 OK sms send 34 i am griddinshi OK
- 4.1.3. SMSInbox:
> 4.2. SMSSent
- 4.2.1. 已发短信记录:
- 4.2.2. SMSSent:
5. 未解决问题:
> 5.1. ContentObserver 只能监听content://sms 而不支持content://sms/inbox content://sms/sent 个人猜测是因为:android 在写sms数据库 insert(...) 没有通过ContentResolver通知content://sms/inbox content://sms/sent 所致 即:没有以下代码:
getContext().getContentResolver().notifyChange(noteUri, null);
6. 如果写的有问题的 欢迎讨论 否则 请回帖支持一些
群发?抱歉 因为我没有真机做测试 而emulator又不支持
不过那要看我怎么做了吧
很多书上只讲到了从数据库中获得联系人,并显示在一个ListView中,却没有讲到单击联系人需要拨号,经过本人努力,实现了这个功能。
主要是需要知道并获得CursorWrapper 这个类,它是指针的封装类,可以用于获得当前指针指向的数据。代码公布如下:
import android.app.Activity; import android.app.PendingIntent; import android.content.ContentUris; import android.content.Intent; import android.database.Cursor; import android.database.CursorWrapper; import android.graphics.Color; import android.net.Uri; import android.os.Bundle; import android.provider.Contacts.People; import android.telephony.PhoneNumberUtils; import android.telephony.gsm.SmsManager; import android.util.Log; import android.view.View; import android.widget.AdapterView; import android.widget.LinearLayout; import android.widget.ListAdapter; import android.widget.ListView; import android.widget.SimpleCursorAdapter; import android.widget.Toast; public class App extends Activity { private static final String TAG="App"; ListView listView; ListAdapter adapter; /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // setContentView(R.layout.main); LinearLayout linearLayout=new LinearLayout(this); linearLayout.setOrientation(LinearLayout.VERTICAL); linearLayout.setBackgroundColor(Color.BLACK); LinearLayout.LayoutParams param=new LinearLayout.LayoutParams(LinearLayout.LayoutParams.FILL_PARENT,LinearLayout.LayoutParams.WRAP_CONTENT); listView=new ListView(this); listView.setBackgroundColor(Color.BLACK); linearLayout.addView(listView,param); this.setContentView(linearLayout); //从数据库获取联系人姓名和电话号码 Cursor cur=this.getContentResolver().query(People.CONTENT_URI,null, null,null,null); adapter=new SimpleCursorAdapter(this,android.R.layout.simple_list_item_2,cur,new String[]{People.NAME,People.NUMBER},new int[]{android.R.id.text1,android.R.id.text2}); this.startManagingCursor(cur); listView.setAdapter(adapter); //listView.setEmptyView(findViewById(R.id.empty)); listView.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener(){ public void onItemSelected(AdapterView<?> arg0, View arg1, int arg2, long arg3) { // TODO Auto-generated method stub //openToast("滚动到:"+arg0.getSelectedItemId()); //短信发送 // PendingIntent pi = PendingIntent.getActivity(App.this,0,new Intent(App.this,App.class),0); // SmsManager sms = SmsManager.getDefault(); // sms.sendTextMessage("5554", null, "message", pi, null); } 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 // String[] names=((CursorWrapper)listView.getItemAtPosition(position)).getColumnNames(); //从指针的封装类中获得选中项的电话号码并拨号 CursorWrapper wrapper=(CursorWrapper)listView.getItemAtPosition(position); int columnIndex=wrapper.getColumnIndex(People.NUMBER); if(!wrapper.isNull(columnIndex)){ String number=wrapper.getString(columnIndex); Log.d(TAG,"number="+number); // //判断电话号码的有效性 if(PhoneNumberUtils.isGlobalPhoneNumber(number)){ Intent intent = new Intent(Intent.ACTION_DIAL,Uri.parse("tel://"+ number)); startActivity(intent); } } } }); } private void openToast(String str){ Toast.makeText(this,str,Toast.LENGTH_SHORT).show(); } }
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.wt.app" android:versionCode="1" android:versionName="1.0"> <application android:icon="@drawable/icon" android:label="@string/app_name"> <activity android:name=".App" android:label="@string/app_name"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> <!-- 点击拨号时,询问调用默认的程序还是调用本程序拨号 --> <intent-filter> <action android:name="android.Intent.Action.CALL_BUTTON" /> <category android:name="android.Intent.Category.DEFAULT" /> </intent-filter> </activity> </application> <uses-permission android:name="android.permission.READ_CONTACTS" /> <uses-sdk android:minSdkVersion="3" /> </manifest>
ANDROID初级历险之spinner篇
listview和spinner都算是adapter容器,在显示时都需要一个adapter去装载需要显示的数据。上一篇已经介绍了怎样用adapter去做数据绑定,这节我们去看看下试图部分的模板。
回忆下,在listview中的ArrayAdapter部分,我们用过simple_list_item_1做视图模板。这个视图模板实际上就是一个textview组件而已。你到对应sdk的layout目录下可以找到答案。同时你也可以发现还有很多可用的模板:simple_list_item_2,simple_list_item_checked,simple_list_item_multiple_choice,simple_list_item_single_choice…这些模板理论上都可以替换原来的simple_list_item_1。具体看进去:simple_list_item_2就是2个textview构成的。simple_list_item_multiple_choice和simple_list_item_single_choice实际上就是个CheckedTextView,checkMark属性是个reference类型,引用单选或多选按钮图片就成了上面2个视图的区别。
同理,spinner也有2种模板。simple_spinner_item,这个是textview的。simple_spinner_dropdown_item,这个是带单选按钮的。
所以熟悉了系统自带的默认模板,有时候我们会发现大部分情况下,我们并不需要自定义视图模板了…
稍微仔细观察下listview和spinner的不同,你会发现listview是全屏的,而spinner的是模态的圆角非全屏的。显然spinner的表现要好得多,那它和listview有什么不同呢?
spinner有一个属性android:prompt,提示这样描述的,The prompt to display when the spinner’s dialog is shown。哦,原来弹出的是个dialog。那弹出的dialog中这么显示列表呢?源代码进去找,发现个AlertDialog有setItems方法或者setAdapter方法,可以让我们轻松的显示一个集合列表。在仔细看进去,你还会发现几个类似的方法:setMultiChoiceItems,setSingleChoiceItems …
基本上到这里,常用的listview和spinner之间的区别就这些了。