《PPC中如何判断网络已经连接或者断开》中已经说明了如何判断PPC的网络是否连接和断开,本篇文章讲讲如何枚举出当前的网络状态,并找到正在使用中的
网络。一个比较简单的应用就是当接到系统连接到网络的通知之后,需要判断是免费网络还是收费网络,之后再作相应处理。一般免费网络就是WIFI和
ActiveSync,收费网络就是CMNET和CMWAP。下面的代码就来判断当前连接的网络究竟是什么网络,要做到这一点,需要使用一个函数
ConnMgrQueryDetailedStatus,这个函数可以得到网络的详细信息,从他的dwType中可得到网络的类别,dwSubtype可
以知道具体的网络类型。具体代码如下,写的已经是比较详细了,如有问题可以在17feixiang.com上留言。
CONNMGR_CONNECTION_DETAILED_STATUS *StatusBuffer;
CONNMGR_CONNECTION_DETAILED_STATUS *StatusBufferTemp;
DWORD pcbBufferSize = 0;
ConnMgrQueryDetailedStatus(NULL, &pcbBufferSize);
StatusBuffer = (CONNMGR_CONNECTION_DETAILED_STATUS *)new BYTE[pcbBufferSize];
ConnMgrQueryDetailedStatus(StatusBuffer, &pcbBufferSize);
StatusBufferTemp = StatusBuffer;
while(StatusBufferTemp != NULL)
{
if(StatusBufferTemp-〉dwConnectionStatus == CONNMGR_STATUS_CONNECTED)
{
if(StatusBufferTemp-〉dwType == CM_CONNTYPE_NIC)
{
switch(StatusBufferTemp-〉dwSubtype)
{
case CM_CONNSUBTYPE_NIC_WIFI :
MessageBox(_T("WIFI"));
break;
case CM_CONNSUBTYPE_NIC_ETHERNET :
break;
}
}
if(StatusBufferTemp-〉dwType == CM_CONNTYPE_CELLULAR)
{
switch(StatusBufferTemp-〉dwSubtype)
{
case CM_CONNSUBTYPE_CELLULAR_CSD :
break;
case CM_CONNSUBTYPE_CELLULAR_GPRS :
MessageBox(_T("GPRS"));
break;
case CM_CONNSUBTYPE_CELLULAR_1XRTT :
MessageBox(_T("1XRTT"));
break;
case CM_CONNSUBTYPE_CELLULAR_1XEVDO :
MessageBox(_T("1XEVDO"));
break;
case CM_CONNSUBTYPE_CELLULAR_1XEVDV :
break;
case CM_CONNSUBTYPE_CELLULAR_EDGE :
MessageBox(_T("EDGE"));
break;
case CM_CONNSUBTYPE_CELLULAR_UMTS :
break;
case CM_CONNSUBTYPE_CELLULAR_VOICE :
MessageBox(_T("VOICE"));
break;
case CM_CONNSUBTYPE_CELLULAR_PTT :
break;
//case CM_CONNSUBTYPE_CELLULAR_HSDPA :
// break;
}
}
if(StatusBufferTemp-〉dwType == CM_CONNTYPE_PROXY)
{
switch(StatusBufferTemp-〉dwSubtype)
{
case CM_CONNSUBTYPE_PROXY_WAP :
MessageBox(_T("WAP"));
break;
case CM_CONNSUBTYPE_PROXY_HTTP :
MessageBox(_T("HTTP"));
break;
}
}
if(StatusBufferTemp-〉dwType == CM_CONNTYPE_PC)
{
switch(StatusBufferTemp-〉dwSubtype)
{
case CM_CONNSUBTYPE_PC_DESKTOPPASSTHROUGH :
MessageBox(_T("ACTIVESYNC"));
break;
case CM_CONNSUBTYPE_PC_UNKNOWN :
break;
}
}
}
StatusBufferTemp = StatusBufferTemp-〉pNext;
}
delete StatusBuffer;
在PPC的开发中有时需要判断网络什么时候连接,什么时候断开,并且做出相应的处理。比如说TimeSyncPPC程序,就需要在网络连接之后得到消息然
后去下载时间同步信息,因此就必须要一种方法让TimeSyncPPC程序在PPC连接网络之后得到系统的消息。当然有人说可以不断去轮询看系统是否连接
到网络了,但这绝对不是一个好方法。其实Windows Mobile
5.0之后微软已经提供了一个专门的函数来得到网络状态变化的Notification,我们要做的就是等待这个消息,函数如下:
HRESULT WINAPI ConnMgrRegisterForStatusChangeNotification(
BOOL fEnable,
HWND hWnd
);
这个函数是通知系统将网络状态变化的Notification发给指定的窗口,返回值有三个:
S_OK The function call was successful.
E_HANDLE The supplied window handle is invalid.
E_ACCESSDENIED The calling process does not have sufficient privileges to use this function.
当
网络连接或者断开时,ConnMgrRegisterForStatusChangeNotification函数会发送
CONNMGR_STATUS_CHANGE_NOTIFICATION_MSG的消息给窗口,但是其实看
CONNMGR_STATUS_CHANGE_NOTIFICATION_MSG的定义,它并不是一个消息,只是一个字符串:
#define CONNMGR_STATUS_CHANGE_NOTIFICATION_MSG TEXT("CONNMGR_STATUS_CHANGE_NOTIFICATION_MSG")
所以使用这个消息之前还需要注册这个消息:
m_ConnectMsg = RegisterWindowMessage(CONNMGR_STATUS_CHANGE_NOTIFICATION_MSG);
返回值m_ConnectMsg就是传递到窗口的消息。
收
到CONNMGR_STATUS_CHANGE_NOTIFICATION_MSG
的消息之后可以根据wParam来判断是断开还是连接上网络。测试发现,连接ActiveSync、GPRS(无论是cmwap或是cmnet)、
WIFI甚至打电话挂电话都会触发这个消息,在实际使用时还需要注意。
另外,微软的文档中有一个错误,经常把函数
ConnMgrRegisterForStatusChangeNotification错写成
ConnMgrRegisterForStatusNotification,所以如果看到
ConnMgrRegisterForStatusNotification这个函数,其实就是指的
ConnMgrRegisterForStatusChangeNotification函数。
下面是一个实际的例子,如有疑问请在www.17feixiang.com留言:
#include "connmgr_status.h"
#include "connmgr.h"
#pragma comment(lib,"wininet.lib")
#pragma comment(lib,"cellcore.lib")
UINT m_ConnectMsg;
BOOL CXXXXDlg::OnInitDialog()
{
......
HRESULT ret = ConnMgrRegisterForStatusChangeNotification(TRUE, this-〉m_hWnd);
m_ConnectMsg = RegisterWindowMessage(CONNMGR_STATUS_CHANGE_NOTIFICATION_MSG);
return TRUE; // return TRUE unless you set the focus to a control
}
LRESULT CXXXXDlg::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
// TODO: Add your specialized code here and/or call the base class
if(message == m_ConnectMsg)
{
switch(wParam)
{
case CONNMGR_STATUS_CONNECTED :
MessageBox(_T("Connected"));
break;
case CONNMGR_STATUS_DISCONNECTED :
MessageBox(_T("DISCONNECTED"));
break;
default :
break;
}
}
return CDialog::WindowProc(message, wParam, lParam);
}
// to create live folder on "home" screen
if (LiveFolders.ACTION_CREATE_LIVE_FOLDER.equalsIgnoreCase(getIntent().getAction())) { // getIntent().getAction() can be null Intent intent = new Intent(); Uri LIVE_FOLDER_URI = Uri.parse("content://contacts/live_folders/people"); intent.setData(LIVE_FOLDER_URI); intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_BASE_INTENT, new Intent( Intent.ACTION_VIEW, Contacts.People.CONTENT_URI)); intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_NAME, "allnamefolder"); intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_ICON, Intent.ShortcutIconResource .fromContext(this, R.drawable.krec)); intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_DISPLAY_MODE, LiveFolders.DISPLAY_MODE_GRID); this.setResult(RESULT_OK, intent); } else { this.setResult(Activity.RESULT_CANCELED); }
// to create shortcut on "home" screen
// toPrint is a very simple apk.
Intent toPrint = new Intent(this, anSimplePrint.class); Intent addShortcut = new Intent("com.android.launcher.action.INSTALL_SHORTCUT"); addShortcut.putExtra(Intent.EXTRA_SHORTCUT_NAME, "SP"); addShortcut.putExtra(Intent.EXTRA_SHORTCUT_INTENT, toPrint); addShortcut.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE, Intent.ShortcutIconResource .fromContext(this, R.drawable.ksig));
//remember to fix manifest file
// add .
<intent-filter> <action android:name="android.intent.action.CREATE_SHORTCUT" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> <intent-filter> <action android:name="android.intent.action.CREATE_LIVE_FOLDER" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> <uses-permission android:name="com.android.launcher.permission.INSTALL_SHORTCUT"></uses-permission>
///////////////////////////////////////////////////////////////////////////////
// 系统的contacts 在桌面livefolder添加快捷方式的代码如下。想要在livefolder中添加// 其他app的快捷方式,也可以依法炮制
public class ContactsLiveFolders { public static class StarredContacts extends Activity { public static final Uri CONTENT_URI = Uri.parse("content://contacts/live_folders/favorites"); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); final Intent intent = getIntent(); final String action = intent.getAction(); if (LiveFolders.ACTION_CREATE_LIVE_FOLDER.equals(action)) { setResult(RESULT_OK, createLiveFolder(this, CONTENT_URI, getString(R.string.liveFolder_favorites_label), R.drawable.ic_launcher_folder_live_contacts_starred)); } else { setResult(RESULT_CANCELED); } finish(); } } public static class PhoneContacts extends Activity { public static final Uri CONTENT_URI = Uri.parse("content://contacts/live_folders/people_with_phones"); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); final Intent intent = getIntent(); final String action = intent.getAction(); if (LiveFolders.ACTION_CREATE_LIVE_FOLDER.equals(action)) { setResult(RESULT_OK, createLiveFolder(this, CONTENT_URI, getString(R.string.liveFolder_phones_label), R.drawable.ic_launcher_folder_live_contacts_phone)); } else { setResult(RESULT_CANCELED); } finish(); } } public static class AllContacts extends Activity { public static final Uri CONTENT_URI = Uri.parse("content://contacts/live_folders/people"); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); final Intent intent = getIntent(); final String action = intent.getAction(); if (LiveFolders.ACTION_CREATE_LIVE_FOLDER.equals(action)) { setResult(RESULT_OK, createLiveFolder(this, CONTENT_URI, getString(R.string.liveFolder_all_label), R.drawable.ic_launcher_folder_live_contacts)); } else { setResult(RESULT_CANCELED); } finish(); } } private static Intent createLiveFolder(Context context, Uri uri, String name, int icon) { final Intent intent = new Intent(); intent.setData(uri); intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_BASE_INTENT, new Intent(Intent.ACTION_VIEW, Contacts.CONTENT_URI)); intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_NAME, name); intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_ICON, Intent.ShortcutIconResource.fromContext(context, icon)); intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_DISPLAY_MODE, LiveFolders.DISPLAY_MODE_LIST); return intent; } }
////////////////////
//
intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_BASE_INTENT, new Intent(Intent.ACTION_VIEW, Contacts.CONTENT_URI)); 这个在用的时候。 如果你用的uri读出来没有一个叫name的表项, 那这个ACTION_VIEW会失败,所以需要有个叫name的,作为livefolder显示出来的东西。 比如系统的contactsprovider 里面 大概3790行 case LIVE_FOLDERS_CONTACTS_GROUP_NAME: qb.setTables(mDbHelper.getContactView()); qb.setProjectionMap(sLiveFoldersProjectionMap); qb.appendWhere(CONTACTS_IN_GROUP_SELECT); selectionArgs = insertSelectionArg(selectionArgs, uri.getLastPathSegment()); break; 这里qb.setProjectionMap(sLiveFoldersProjectionMap); 就是把有一个项作为name显示 //contactsProvider2.java line 854 sLiveFoldersProjectionMap = new HashMap<String, String>(); sLiveFoldersProjectionMap.put(LiveFolders._ID, Contacts._ID + " AS " + LiveFolders._ID); sLiveFoldersProjectionMap.put(LiveFolders.NAME, Contacts.DISPLAY_NAME + " AS " + LiveFolders.NAME);
自己实现一contentprovider来做这个事情的时候也需要实现一个 name作为显示,
即在cursor query的时候加入比如select xxx as name, 在projection里加。
说不清了。。。
//一般还是不要自己造contentprovider了。。 系统的差不多够用
//问题是怎么在系统里找到所有的uri? 嘿嘿。我不会- -
这里转个大大的博克
http://kuikui.iteye.com/blog/318627
刚起步的时候经常困扰我们的是一些本来容易解决的问题,往往我们会花掉很大的力气去找解决的办法,最后才知道原来这么简单,这就是英文世界造成的。
Intent在 Android应用开发中,占有很大的分量,关于Intent在Android中的作用在网络上已经有很多资料了,这里不再累赘,本人喜欢直来直去。在网上看到很多关于Intent的资料,说那么多,你也许还是一头雾水,到底如何使用Intent呢?这里总结一些重用的Intent使用,仅供参考。
下面直接给我学习的实例片段。
1,掉web浏览器
Uri myBlogUri = Uri.parse("http://kuikui.iteye.com");
returnIt = new Intent(Intent.ACTION_VIEW, myBlogUri);
2,地图
Uri mapUri = Uri.parse("geo:38.899533,-77.036476");
returnIt = new Intent(Intent.ACTION_VIEW, mapUri);
3,调拨打电话界面
Uri telUri = Uri.parse("tel:100861");
returnIt = new Intent(Intent.ACTION_DIAL, telUri);
4,直接拨打电话
Uri callUri = Uri.parse("tel:100861");
returnIt = new Intent(Intent.ACTION_CALL, callUri);
5,卸载
Uri uninstallUri = Uri.fromParts("package", "xxx", null);
returnIt = new Intent(Intent.ACTION_DELETE, uninstallUri);
6,安装
Uri installUri = Uri.fromParts("package", "xxx", null);
returnIt = new Intent(Intent.ACTION_PACKAGE_ADDED, installUri);
7,播放
Uri playUri = Uri.parse("file:///sdcard/download/everything.mp3");
returnIt = new Intent(Intent.ACTION_VIEW, playUri);
8,掉用发邮件
Uri emailUri = Uri.parse("mailto:shenrenkui@gmail.com");
returnIt = new Intent(Intent.ACTION_SENDTO, emailUri);
9,发邮件
returnIt = new Intent(Intent.ACTION_SEND);
String[] tos = { "shenrenkui@gmail.com" };
String[] ccs = { "shenrenkui@gmail.com" };
returnIt.putExtra(Intent.EXTRA_EMAIL, tos);
returnIt.putExtra(Intent.EXTRA_CC, ccs);
returnIt.putExtra(Intent.EXTRA_TEXT, "body");
returnIt.putExtra(Intent.EXTRA_SUBJECT, "subject");
returnIt.setType("message/rfc882");
Intent.createChooser(returnIt, "Choose Email Client");
10,发短信
Uri smsUri = Uri.parse("tel:100861");
returnIt = new Intent(Intent.ACTION_VIEW, smsUri);
returnIt.putExtra("sms_body", "shenrenkui");
returnIt.setType("vnd.android-dir/mms-sms");
11,直接发邮件
Uri smsToUri = Uri.parse("smsto://100861");
returnIt = new Intent(Intent.ACTION_SENDTO, smsToUri);
returnIt.putExtra("sms_body", "shenrenkui");
12,发彩信
Uri mmsUri = Uri.parse("content://media/external/images/media/23");
returnIt = new Intent(Intent.ACTION_SEND);
returnIt.putExtra("sms_body", "shenrenkui");
returnIt.putExtra(Intent.EXTRA_STREAM, mmsUri);
returnIt.setType("image/png");
用获取到的Intent直接调用startActivity(returnIt)就ok了。
今天就说这么多,有更大需求的可以看源代码。