使用 Google API 的 GTalk
关键技能 & 概念
● 执行一个 Google API 包装
● 为 Google 存取配置 XMPP 开发环境设置
● 执行 View.OnClickListener( ) 方法
第九章为你介绍了 Google API。你创建了一个影响 Google API 和 Google 地图
的活动。因为 API 的易用和灵活性,可以快速的在 Google 地图上显示用户的当
前位置。同样,你还学习到了如何使用很少量的相关代码来熟练操控地图。
Google API 包括了不仅是进入 Google 地图的功能。在上一章中,你使用了很大
的 API 中很小的一个部分。对于 Google API 的基本包装是 com.google。从这个
基础中,Google API 包含了允许你创建操控 GTalk(Google 的 聊 天 服 务 ),Google
日历,Google 文档,Google 电子表格和 Google 服务等等的活动的权力。
当我看是写这本书时,这个版本的 Android SDK 是 m3-rc22。写完书时,Google
不提倡这些包装中的一些,但是还是留在 SDK 中。有显示 Google 日历,Google
电子表格和 Google 服务仍旧需要升级,遗憾的是,在 m5-rc15 版本的 SDK 中还
处于未完成的状态。为了避免混乱,Google 还移除了任何与这些包装相关联的
帮助文件。因此,本章的重点是在最新发布的 Android SDK 中工作很好的 GTalk。
在本章中,将会构造一个小的,利用 Android SDK 的 GTalk 包装的活动。当活动
完成,你将可以利用手机发送并接受信息到/从另外一个 GTalk 用户。
注意
在第一个 Google API 的反复中,处理 GTalk 的包装是一个非常广泛的 XMPP 包装 。
(XMPP 是很多聊天平台的基础协议,包括 GTalk 和 Jabber)。使用最新版的 SDK,
初始的 XMPP 包装为反映 GTalk 的特性而加强并重命名。要开始,用 Eclipse 创
建一个新项目,并且命名 GoogleAPI。
为 GTalk 配置 Android 模拟器
在开始为本项目写代码之前,你需要在 Android 模拟器中调整开发环境设定,
XMPP 设定。
在项目打开的状态下,需要离开常规的程序一会儿。如果你熟悉 GTalk,你知道
只有登录 Google 帐户以后才可以使用这个产品。因此,你必须要采取特别的步
骤来确保你的设备(本例是 Android 模拟器)可以登录你的 Google 帐户,这样,
可以确保发送和接受信息。136
导航到 AndroidSDK/tools 文件夹并且启动模拟器。你可以从 Eclipse 开发环境
中启动它,但是那样会需要同时启动还没有写代码的活动。为了节约时间,手动
启动模拟器。
模拟器启动后,点击所有的快捷方式(All shortcut)。找到 Dev Tools 条目并
且启动它。你将会看到和下面类似的图(略)。
滚动 Dev Tools 菜单直到看见 XMPP 设置。选择 XMPP 设定。
注意
当你打开 XMPP 设定,活动的名称是 GTalk 设定。这个可能是个表象说明 Google
将在 Google API 包装中留下这个包装。命名显而易见的断开可能是从不同 SDK
版本之间改变而剩下的。
活动应当读取账户:<None>,如图所示(略)。这表明了设备中没有储存登录信息 。
你需要为 Google 帐户增加登录信息来允许活动来有权使用 Google 的服务器。
点击增加帐户来显示一个屏幕。输入用户名和密码之后,点击登录。Android 模
拟 器 应 当 试 着 去 验 证 你 的 信 息 。 当 模 拟 器 尝 试 验 证 信 息 时 , 它 显 示
“Authenticating”信息。
警告
取决于你的连接和你是否有一个调试器和模拟器相连接。你可能会看到
“Authenticating”信息一会儿。如果你的帐户几分钟后无法被验证,重新启动
模拟器再试试。
一旦你的信息被验证,你应当看到下图(略)。注意,这里没有返回按钮,点击
模拟器中的 Home 键返回到主屏幕。
现在模拟器配置好了,并且项目也被设置好,可以开始为活动写代码了。
在 Android 中执行 GTalk
在 Android Android Android Android 中执行 GTalk GTalk GTalk GTalk 第十章(2) (2) (2) (2)
在本节中将使用 Google API 来创建一个使用 GTalk 的活动。这个活动会从 GTalk
网络发送并接收信息,在屏幕上保存它们,并在通知条中显示。活动可以和其它
GTalk 用户通信,而不管他们是在使用 Android 手机或者电脑上的 GTalk。
下一部分将从如何创建应用程序的布局开始。第一步为活动的编码工作就是增加
布局到 GoogleAPI.xml 中。
在 GoogleAPI.xml 中创建活动的布局
本活动由一些 Views 组成。需要一个 ListView 来显示需要发送的文本信息。同
样需要两个 EditText,用于接收者的地址和信息,还有一个发送的 Button。
首先设置一个 id 是 messageList 的 ListView,如下所示。在本布局中,将使用
一个新的属性,android:scrollbars。设置这个属性为立式会让你可以在信息列
表中滚动。137
把 ListView 放到主要布局标签中。在 ListView 布局的下方,放置一个 EditText
的布局,如下。这个 EditText 保留你会发信息的收信人地址。
这个 EditText 视图应该没有什么特殊之处。
最后,再创建一个水平的布局来保留信息内容——EditText 和发送按钮:
这 个 布 局 会 以 线 性 的 方 式 放 置 视 图 , 所 以 它 们 相 互 排 成 一 排 。 把 这 个
LinearLayout 放置进主要的 LinearLayout。最后,GoogleAPI.xml 文件将会如
下:
<ListView
android:id="@+id/messageList"
android:layout_width="fill_parent"
android:layout_height="0dip"
android:scrollbars="vertical"
android:layout_weight="1"
android:drawSelectorOnTop="false" />
<EditText
android:id="@+id/messageTo"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="16sp"
android:minWidth="250dp"
android:scrollHorizontally="true" />
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="horizontal"
android:layout_width="fill_parent"
android:layout_height="wrap_content">
<EditText
android:id="@+id/messageText"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="16sp"
android:minWidth="250dp"
android:scrollHorizontally="true" />
<Button
android:id="@+id/btnSend"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Send Msg">
</Button>
</LinearLayout>138
为 GoogleAPI.java 增加包装
布局文件完成后,有一些新的包装需要增加到 GoogleAPI.java 文件中。第一个
<?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">
<ListView
android:id="@+id/messageList"
android:layout_width="fill_parent"
android:layout_height="0dip"
android:scrollbars="vertical"
android:layout_weight="1"
android:drawSelectorOnTop="false" />
<EditText
android:id="@+id/messageTo"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="16sp"
android:minWidth="250dp"
android:scrollHorizontally="true" />
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="horizontal"
android:layout_width="fill_parent"
246 Android: A Programmer’s Guide
android:layout_height="wrap_content">
<EditText
android:id="@+id/messageText"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="16sp"
android:minWidth="250dp"
android:scrollHorizontally="true" />
<Button
android:id="@+id/btnSend"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Send Msg">
</Button>
</LinearLayout>
</LinearLayout>139
必须导入的包装应该和增加到布局中的 Views 相对应。因此,必须为 EditText,
ListView,ListAdapter,和按钮导入包装:
同样需要导入的是 Google API 和 GTalk 打交道的包装:
其它需要的包装有 intent,ServiceConnection, Color, 和 Im. 完整的列表如
下:
如你所见,本活动所需的包装不多。而且,你会发现,发送和接受信息所需要的
代码非常的少。现在,需要执行一个 onClickListnerner 来允许代码。
执行 View.OnClickListener
需要修改 GoogleAPI 的类来执行 View.OnClickListener。当任何按钮被点击,
这将允许你从活动的主类来呼叫 onClick()方法。通常,这种执行 onClick()方
法是有效的:只有当你在一个活动中有一组按钮,并且以一种方式来出处理所有
的 onClick 呼叫。但是,我感觉你仍旧需要看看这个方式是如何工作的,那样你
就可以在将来用于自己的代码中。记住,展示这个方式是因为在很多情况下,它
可以是非常有用的工具
在活动中执行常规变量是本书中另外一个未曾谈到的内容。你需要在活动中建立
一些常规变量,这样就可以以很多方法来使用:
import android.widget.EditText;
import android.widget.ListView;
import android.widget.ListAdapter;
import android.widget.Button;
import com.google.android.gtalkservice.IGTalkSession;
import com.google.android.gtalkservice.IGTalkService;
import com.google.android.gtalkservice.GTalkServiceConstants;
import com.google.android.gtalkservice.IChatSession;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.database.Cursor;
Chapter 10: Using the Google API with GTalk 247
248 Android: A Programmer’s Guide
import android.os.Bundle;
import android.os.DeadObjectException;
import android.os.IBinder;
import android.provider.Im;
import android.graphics.Color;
import android.view.View;
import android.widget.SimpleCursorAdapter;
public class GoogleAPI extends Activity implements View.OnClickListener {
}140
在 onCreate()方法中,你将执行正常的初始化。应当赋值布局到活动并且设置
IGTalkSession 为 null。同样,在活动中增加点小乐趣,改变 ListView 的背景
色为灰色。
提示
因为你是从类中执行 View.OnClickListener 的,可以设置 Send Button 的
OnClickListener()方式到其中。最后在 onCreate()方法内执行是绑定你的服
务。这个过程创建创建需要使用的连接,由 Google 帐户使用,传递 GTalk 信息:
在上面的 bindService 声明中,传递到 setComponent()方法的一个参数就是连
接。这个变量表示一个 ServiceConnection 执行 onServiceConnected()和
onServiceDisconnected()方法。下面的代码构造约束前面 bindService 声明
的连接:
在 onServiceConnected()方法中,你建立了一个使用 IGTalkService.Stub 的
EditText messageText;
ListView messageList;
IGTalkSession myIGTalkSession;
EditText messageTo;
Button sendButton;
myIGTalkSession = null;
messageText = (EditText) findViewById(R.id.messageText);
messageList = (ListView) findViewById(R.id.messageList);
messageTo = (EditText) findViewById(R.id.messageTo);
sendButton = (Button) findViewById(R.id.btnSend);
sendButton.setOnClickListener(this);
messageList.setBackgroundColor(Color.GRAY );
this.bindService(new
Intent().setComponent(GTalkServiceConstants.GTALK_SERVICE_COMPONENT),
connection, 0);
private ServiceConnection connection = new ServiceConnection() {
public void onServiceConnected(ComponentName name, IBinder service) {
try {
myIGTalkSession =
IGTalkService.Stub.asInterface(service).getDefaultSession();
} catch (DeadObjectException e) {
myIGTalkSession = null;
}
}
public void onServiceDisconnected(ComponentName name) {
myIGTalkSession = null;
}
};141
片段。如果这个过程失败,你需要再次把这个片段设为 null。现在,可以为类
的 onClick 事件创建代码。在每一个 onClick 事件中你应当执行一些行动:
1. 为任何的信息检查数据库。
2. 从查询的结果中创建一个 ListAdapter 并显示到 ListView 中。
3. 创建一个 ChatSession 到 EditView 中的地址并发送你的信息文本。
注意
Android 服务器包括 SQLite 数据库,你可以使用来保留任何你认为需要放入的
活动相关条目和任何的定制数据。这个数据库在第十一章做深入的介绍。
下面的代码为你和接受者之间发生的信息查询数据库:
使用 下 面 的 代 码 来 从 查 询 结 果 创 建 一 个 ListAdapter 并且 赋 值 接 收 器 到
ListView。在前一个活动,你已经使用了一个类似的过程,所以,对你应该不陌
生。
信息可以显示了,最后的步骤是发送你的信息。下面的代码用定义的 messageTO
address 创建一个 IchatSession。这个信息文本然后被从这里传递到接受者。
所有的东西在一起,完成后的 GoogleAPI.java 文件如下:
Cursor cursor = managedQuery(Im.Messages.CONTENT_URI, null,
"contact=\'" + messageTo.getText().toString() + "\'", null, null);
ListAdapter adapter = new SimpleCursorAdapter(this,
android.R.layout.simple_list_item_1, cursor,
new String[]{Im.MessagesColumns.BODY},
new int[]{android.R.id.text1});
this.messageList.setAdapter(adapter);
try {
IChatSession chatSession;
chatSession =
myIGTalkSession.createChatSession(messageTo.getText().toString(););
chatSession.sendTextMessage(messageText.getText().toString());
} catch (DeadObjectException ex) {
myIGTalkSession = null;
}
package android_programmers_guide.GoogleAPI;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.database.Cursor;
import android.os.Bundle;142
import android.os.DeadObjectException;
import android.os.IBinder;
import android.provider.Im;
import android.graphics.Color;
import android.view.View;
import android.widget.EditText;
import android.widget.ListView;
import android.widget.ListAdapter;
import android.widget.Button;
import android.widget.SimpleCursorAdapter;
import com.google.android.gtalkservice.IGTalkSession;
import com.google.android.gtalkservice.IGTalkService;
import com.google.android.gtalkservice.GTalkServiceConstants;
import com.google.android.gtalkservice.IChatSession;
public class GoogleAPI extends Activity implements View.OnClickListener {
EditText messageText;
ListView messageList;
IGTalkSession myIGTalkSession;
EditText messageTo;
Button mSend;
@Override
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
setContentView(R.layout.main);
myIGTalkSession = null;
messageText = (EditText) findViewById(R.id.messageText);
messageList = (ListView) findViewById(R.id.messageList);
messageTo = (EditText) findViewById(R.id.messageTo);
mSend = (Button) findViewById(R.id.btnSend);
mSend.setOnClickListener(this);
messageList.setBackgroundColor(Color.GRAY );
Chapter 10: Using the Google API with GTalk 251
this.bindService(new
Intent().setComponent(GTalkServiceConstants.GTALK_SERVICE_COMPONENT),
connection, 0);
}
private ServiceConnection connection = new ServiceConnection() {
public void onServiceConnected(ComponentName name, IBinder service) {
try {
myIGTalkSession =
IGTalkService.Stub.asInterface(service).getDefaultSession();
} catch (DeadObjectException e) {
myIGTalkSession = null;
}143
编译并运行 GoogleAPI
编译并运行 GoogleAPI GoogleAPI GoogleAPI GoogleAPI 第十章(3) (3) (3) (3)
现在,在模拟器中运行 GoogleAPI。如何你的连接成功,你应当能看见下面的屏
幕。
要测试活动,我发送“hello”信息给
[email protected],
显示如下:
}
public void onServiceDisconnected(ComponentName name) {
myIGTalkSession = null;
}
};
public void onClick(View view) {
Cursor cursor = managedQuery(Im.Messages.CONTENT_URI, null,
"contact=\'" + messageTo.getText().toString() + "\'",
null, null);
ListAdapter adapter = new SimpleCursorAdapter(this,
android.R.layout.simple_list_item_1, cursor,
new String[]{Im.MessagesColumns.BODY},
new int[]{android.R.id.text1});
this.messageList.setAdapter(adapter);
try {
IChatSession chatSession;
chatSession =
myIGTalkSession.createChatSession(messageTo.getText().toString());
chatSession.sendTextMessage(messageText.getText().toString());
} catch (DeadObjectException ex) {
myIGTalkSession = null;
}
}
}144
下一个插图,你将看到点击发送按钮后,移动我发送的信息到 ListView。
当我作为 androidprogrammersguide 登录,我发现信息确实到了预想的收件人,
如下:
我回复“Greetings!”,要查看这个,看下面的两个图。注意活动屏幕顶部的信
息条。紧接的图,你可以看到发送者的信息被显示。下一章,将创建最后一个应用程序,你会在 Google 地图上使用 SQLite 数据库和
Google 地图 Overlays 来绘制数据记录。这些是非常有力的技术来提升 Android
超越其它的移动操作系统。
试试这个:为 GoogleAPI 活动增加设置特性
试试这个:为 GoogleAPI GoogleAPI GoogleAPI GoogleAPI 活动增加设置特性 第十章(4) (4) (4) (4)
编辑 GoogleAPI 活动来包括一个设置特性。使用第八章的 AndroidViews 活动作
为向导,增加一个可以改变应用程序布局属性按钮到 GoogleAPI 活动。这里是一
些提示关于如何设置按钮:
● 改变信息列表的字体
● 改变信息列表的字体颜色,使得发送和接受相反。
● 改变信息列表的背景色。
问专家
Q:GTalk API 可用于其它基于 XMPP 的聊天客户端吗?
A: 对于这个的回答还不清晰。m3-rc22 版本的 SDK 包括一个 XMPP API 而 M5-15
SDK 只含了 GTalk API。有可能这两者在将来发布的 Android SDK 绑在一起。那
样,GTalk API 可以被用来与其它基于 XMPP 聊天客户端的程序通信。