最近需要去实现应用内跳转至微博的个人信息页面(用户个人主页),其本质就是打开并跳转至第三方App指定的页面。显然,我们可以通过调用 startActivity(intent)
来实现该功能,其关键就是 intent
中需要携带哪些信息。
反编译 AndroidManifest.xml
将新浪微博的apk文件中的 AndroidManifest.xml
文件解压提取出来,然后使用 AXMLPrinter2.jar 对清单文件进行反编译:
1
| java -jar AXMLPrinter2.jar AndroidManifest.xml > weibo.xml
|
其中 weibo.xml
即反编译成功的清单文件。
获取微博个人页 Activity
因为之前 weibo.xml
中的 Activity
过多,我们需要知道微博的用户个人信息页面对应的 Activity
全称。这里有两种方法:
- 使用辅助App来获取当前的 Activity ,例如 当前界面(当前Activity) ,具体用法这里就不介绍了。
- 使用 adb 命令获取当前 Activity:
1
| adb shell dumpsys activity activities | grep mFocusedActivity
|
当我们把新浪微博的个人信息页面切至前台时, adb 命令的结果为:
1
| mFocusedActivity: ActivityRecord{9063f6f u0 com.sina.weibo/.page.ProfileInfoActivity t1578}
|
也就是说,新浪微博的个人信息页对应的 Activity 为 com.sina.weibo.page.ProfileInfoActivity
, 那我们看 weibo.xml
中与该 Activity 对应的 xml 节点:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| <activity android:theme="@7F0B0011" android:name="com.sina.weibo.page.ProfileInfoActivity" android:exported="true" android:configChanges="0x000004A0" android:windowSoftInputMode="0x00000020"> <intent-filter> <action android:name="android.intent.action.VIEW"></action> <category android:name="android.intent.category.DEFAULT"></category> <category android:name="android.intent.category.BROWSABLE"></category> <data android:scheme="sinaweibo" android:host="userinfo"></data> </intent-filter> <intent-filter> <action android:name="android.intent.action.VIEW"></action> <category android:name="android.intent.category.DEFAULT"></category> <category android:name="android.intent.category.BROWSABLE"></category> <data android:scheme="http" android:host="weibo.cn" android:path="/qr/userinfo"></data> </intent-filter> <intent-filter> <action android:name="android.intent.action.VIEW"></action> <category android:name="android.intent.category.DEFAULT"></category> <data android:mimeType="vnd.android.cursor.item/vnd.com.sina.weibo.profile"></data> </intent-filter> </activity>
|
分析
上面的 xml 节点中,可以看到 android:exported="true"
,也就是说 ProfileInfoActivity
可以被外部程序调用打开。有三个 intent-filter
节点,其中第三个 intent-filter
与 ContentProvider
有关,可以忽略第三个只考虑前两个。
第一个 intent-filter
接受的 data 格式为 sinaweibo:userinfo
, 第二个 intent-filter
接受的 data 格式为 http://weibo.cn/qr/userinfo
。 我们通过 data 节点的数据可以判断除,第二个除了可以被新浪微博响应,也可以被浏览器之类的应用响应,但第一个只能被新浪微博响应,所以在使用第一个之前,需要判断新浪微博是否已经安装,否则会崩溃。
实现
通过查阅相关文档,ProfileInfoActivity
可以接受的参数为 uid=xxx
, 也就是 sinaweibo://userinfo?uid=xxx
和 http://weibo.cn/qr/userinfo?uid=xxx
,所以最终代码实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
| public void jumpToWeiboProfileInfo(Context context, String uid) { Intent intent = new Intent(Intent.ACTION_VIEW); intent.addCategory(Intent.CATEGORY_DEFAULT); intent.addCategory(Intent.CATEGORY_BROWSABLE); boolean weiboInstalled = PackageUtils.isSinaWeiboInstalled(context); if (weiboInstalled) { intent.setData(Uri.parse("sinaweibo://userinfo?uid=" + uid)); } else { intent.setData(Uri.parse("http://weibo.cn/qr/userinfo?uid=" + uid)); } context.startActivity(intent); }
public class PackageUtils { private PackageUtils(){}
public static boolean isSinaWeiboInstalled(Context context) { return isPackageInstalled(context, "com.sina.weibo"); }
public static boolean isPackageInstalled(Context context, String packageName) { PackageManager packageManager = context.getPackageManager(); if (packageManager == null) return false; List<PackageInfo> packageInfoList = packageManager.getInstalledPackages(0); for(PackageInfo info : packageInfoList) { if (info.packageName.equals(packageName)) return true; } return false; } }
|
参考
调用新浪微博显示用户信息