Google Map和桌面组件 Android开发教程

By admin in 天文台 on 2018年10月12日

本文节选于机械工业出版社推出的《Android应用开发揭秘》一 书,作者为杨丰盛。本书内容全面,详细讲解了Android框架、Android组件、用户界面开发、游戏开发、数据存储、多媒体开发和网络开发等基础知 识,而且还深入阐述了传感器、语音识别、桌面组件开发、Android游戏引擎设计、Android应用优化、OpenGL等高级知识。另外,本书还全面 介绍了如何利用原生的C/C++(NDK)和Python、Lua等脚本语言(Android Scripting Environment)来开发Android应用,并以迭代的方式重现了各种常用的Android应用和经典Android游戏的开发全过程。

 

 

下的情节包括9.3 Google Map和9.4 桌面组件两独组成部分:

9.3.1  Google Map概述

9.3.2  准备干活

9.3.3  Google Map API的使用

9.3.4  定位系统

9.4.1  快捷方式

9.4.2  实时文件夹

9.4.3  Widget开发

第9节  Android特色开发

Android
是一个面向应用程序开发之丰富平台,它具有不少享有吸引力的用户界面元素、数据管理和网利用等得天独厚的功能。Android
还提供了成百上千格外有特点的接口。本章我们以分别介绍这些吸引开发者眼球的性状开发,主要不外乎:传感器系统(Sensor)、语音识别术
(RecognizerIntent)、Google
Map和用来支付桌面的插件(Widget)。通过本章的读,读者以对Android有一个复深入之了解,可以开出有有风味、有新意的应用程序。

9.3  Google Map

提起Google Map(Google地图),大家一概想到其姊妹产品Google
Earth(Google地球)。全新的免费地图服务让Google在2005年震了整整互联网界。此后,各大门户纷纷出好的地图服务,不少宗还
和Google一样提供了二次开发的API。目前,基于地图服务之各种以都使恒河沙数一般到处萌发了。当然,对于Google的Android系统来说,
地图肯定吗是必要的特征。

9.3.1  Google Map概述

Google Map是 Google
公司提供的电子地图服务,包括有详细的卫星照片。它能够提供三栽视图:一凡是矢量地图(传统地图),可提供政区和通以及商业信息;二是见仁见智分辨率的卫星照
片(俯视地图,与Google Earth
上之卫星照片基本一样);三是新兴丰富的形势视图,可以用来显示地形和相当高线。它的姊妹产品是Google
Earth——一个桌面应用程序,在三维模型上提供街景与还多之卫星视图及GPS定位的法力。

Google公司给2004年11月买断了美国Keyhole公司,推出了
http://maps.google.com,令人耳目一新。但Google
并未就此止步,在2005年6月之推出了桌面工具Google
Earth,把“地球”放到了每个人之桌面上,让你因在计算机面前,就足以于名川大山里漫步,在高楼大厦群被鸟瞰。

本来,随着Google Map和Google
Earth的生,也应运而生了很多分外幽默之运,比如下面两个突出的因Google
Earth和Google Maps的小游戏非常有创意,吸引了不少玩家。

只要图9-5所显示,我们得在海内外任何地方甚至海底模拟开飞机或潜水艇,来旅游整个社会风气,更多打要参见http://www.sea-seek.com/。

图片 1

祈求9-5  模拟飞行

假定图9-6所显示,我们可以在地上其他地方开着友好喜好的切削往跑,非常有趣的是,可以在计算机前开在车于祥和周围以及熟悉的地方模拟驾驶。详细信息请参见http://geoquake.jp/en/
webgame/DrivingSimulatorGM/。

图片 2

图9-6  模拟驾驶

 

类似之采用还有众多,这里我们只介绍就有限舒缓,有趣味之对象可团结去试试。现在Google
Map已经为运用及不少手机及了,这更加便民了大家的生。下面我们来看看手机及什么使Google
Map,如图9-7所出示。

图片 3

图9-7 Google Map 手机版

它们包括如下效果:

·我的职务(测试版):“我之职位”在地图上显得你的时职务(通常以 1000
米范围中)。即使没有
GPS,你也可规定好的职位。谷歌手机地图还支持内置
GPS,也足以链接到蓝牙 GPS
传感器,以便更准地确定用户的职务。“我的职务”功能是通过辨认而附近无线发射塔的音信播报而规定你的职的。

·地图跟卫星视图:谷歌手机地图为而提供所查看地区的地图跟卫星视图,其界面的用感觉跟公于台式机上相同。
可沿其中一个大方向滚动,以查看地图及之重新多内容;或下快捷键进行缩放。

·商户列表:借助于 Google
的地面搜索引擎,可以随号(如“星巴克”)或项目(如“咖啡”)搜索商家,查看企业的营业时间和评分,然后,只需要点击一下即可拨通感兴趣之店铺之电话。有了“我的职务”功能,甚至都未待输入当前位置即可方便地找到附近的铺。

·驾车路线:可以老便宜地获取驾车路线,其中会清楚地表明每次转弯。有了“我的职”功能,甚至都不需要输入出发点。

·公交换乘:查看公交及地铁线,确定转车路线,制定你于中外 80
多只市之出行计划。“公交换乘”功能时适用于黑莓、Windows Mobile、S60
和另支持 Java 的无绳电话机。

·路况信息:Google
地图中之公路会冲实时路况数据,以绿色、黄色或红显示。

·收藏夹:为你经常去的地方长书签,以便能于地形图及颇便于地回去到这些地方。

世家不用以为这些意义于大哥大及大为难实现,尤其是在咱们设读书之Android平台被,要促成这些职能是非常简单的,只需要使用Android
Maps API(地图API)和Android Location
API(定位API)即可。下面我们拿上怎么样来行使这些API开发协调之地形图应用。

9.3.2  准备工作

当Android
SDK?1.5预装的add-on中提供了一个Map扩展库com.google.android.maps,利用其便可给android应用程序加上
强大的地图功能了。这个库房的位置是“Android
SDK路径”\add-ons\google_apis-3\libs。需要证明的凡,这个库房并无是正规的Android
sdk的内容,可以好从之位置下载,并内置你的sdk中,这样就算可为您新建的以或曾经有些使用加上地图功能了。在运Android
Map API之前,还待申请一个Android Map API Key。

1.申请Android Map API Key

为能够顺利地申请Android Map API
Key,必须使预备Google的账号及体系的证件。一般Google发布Key都用Google的账号,Google的账号是通用的,Gmail的
账号就可了(没有底话语可交http://www.google.com/去申请一个)。当一个应用程序发布时必须要证明书,证明书其实就是MD5。我
们这里不是发布,而仅是为着测试,可以以Debug版的证明。下面我们来读书怎么样申请Android
Map API Key。

步骤1:找到您的debug.keystore文件。

证件的相似路径也:C:\Documents and Settings\眼前用户\Local
Settings\Application Data\
Android\debug.keystore。当然我们以Eclipse开发,便得以打开Eclipse选择
Windows→Preference→Android→Build,其中Default debug
keystore的值就是debug.keystore的不二法门,如图9-8所著。

图片 4

贪图9-8  debug.keystore文件之职务

步骤2:取得debug.keystore的MD5值。

率先以命令提示符下进入debug.keystore文件所在的门路,执行命令:keytool
-list -keystore
debug.keystore,这时可能会见提示您输入密码,这里输入默认的密码“android”,即可得到MD5价,如图9-9所著。

图片 5

图9-9  取得debug.keystore的MD5值

步骤3:申请Android Map的API Key。

开辟浏览器,输入网址:http://code.google.com/intl/zh-CN/android/maps-api-
signup.html,登录Google账号,在Google的Android Map API
Key申请页面及输入步骤2得到的MD5认证指纹,选中“I have read and agree with
the terms and conditions”选项,如图9-10所展示,按下“Generate API
Key”按钮,即可取我们提请及之API Key。

图片 6

图9-10  Android Map API Key申请页面

及此处我们便形成了Android Map API Key的申请,记下申请的Android Map API
Key值,在末端的应用程序中会为此到其。下面我们还将创一个冲Google
APIs的AVD。

2.创办基于Google APIs的AVD

于Eclipse中开拓AVD管理界面,在“Create
AVD”部分的Name处填写AVD的名字,在Target处挑“Google
APIs-1.5”,如图9-11所展示,点击“Create AVD”按钮完成创建。

图片 7

图9-11  创建AVD

3.创造基于Google APIs的工

这里需要小心的是,由于我们得采取Google APIs,所以当开创工程时,在Build
Target处用选择Google
APIs,如图9-12所著,其他选择和原先一样。当然,在运作工程时为即需要选择我们正好创建的依据Google
APIs的AVD来运作。

图片 8

图9-12  创建基于Google APIs的工

顶这里,我们着力做到了准备工作,下面我们拿开读如何用Google
API来开地图应用程序。 

9.3.3  Google Map API的使用

Android中定义了一个名也com.google.android.maps的承保,其中富含了相同名目繁多用于在Google
Map上显示、控制与层叠信息之功能类,以下是该包中尽重点的几个像样:

·MapActivity:这个类似是用于展示Google
Map的Activity类,它用连续底层网络。MapActivity是一个抽象类,任何想要展示MapView的activity都需要派生自
MapActivity,并且于其派生类的onCreate()中,都使创一个MapView实例。

·MapView:MapView是用于展示地图的View组件。它派生自android.view.ViewGroup。它必须和
MapActivity相当以,而且不得不给MapActivity创建,这是因MapView需要通过后台的线程来连续网络或文件系统,而这些线程
要由MapActivity来保管。

·MapController:MapController用于控制地图的倒、缩放等。

·Overlay:这是一个只是显示为地图之上的可绘制的目标。

·GeoPoint:这是一个涵盖经纬度位置的目标。

脚我们用以com.google.android.maps包来贯彻一个地图浏览程序(见本书所附代码:第9段\Examples_09_03)。

步骤1:创建工程,注意要挑的Build Target为“Google APIs”。

步骤2:修改AndroidManifest.xml文件。

出于我们若利用Google Map
API,所以要先行在AndroidManifest.xml中定义如下信:<uses-library
android:name=”com.google.android.maps”/>,当然如果起网获取地图数据,还需添加应用程序访问网络的且
限。代码如下:

     <uses-library android:name=”com.google.android.maps” />

步骤3:创建MapView。

若是来得地图,需要创造一个MapView,在XML文件被的布局要代码清单9-3所出示。其中android:apiKey的价值就是是我们提请的Android
Map API Key。

     代码清单9-3  main.xml

     <?xml version=”1.0″ encoding=”utf-8″?> 

     <RelativeLayout
xmlns:android=”http://schemas.android.com/apk/res/android” 

         android:layout_width=”fill_parent” 

         android:layout_height=”fill_parent” 

         > 

     <com.google.android.maps.MapView

     android:id=”@+id/MapView01″

     android:layout_width=”fill_parent”

     android:layout_height=”fill_parent”

     android:apiKey=”0dYpmIXGIdwiVm-HEpzuUW2fjNYsFQ9EvYir1sg”/>

     </RelativeLayout> 

自,可以以程序中通过如下代码创建MapView。

     MapView map = new MapView(this, “[Android Maps API Key]”);     

步骤4:实现MapActivity。

MapView需要由MapActivity来治本,所以程序部分应继续自MapActivity类,必须实现isRouteDisplayed
方法。MapView提供了3种模式的地图,分别可经过以下方式设置使用什么模式来显示地图。另外,可以透过
setBuiltInZoomControls方法设置地图是否支持缩放。

     //设置也直通模式

     //mMapView.setTraffic(true);

     //设置为卫星模式

     //mMapView.setSatellite(true); 

     //设置也街景模式

     //mMapView.setStreetView(false)   

步骤5:MapController的使用。

假定要是装地图显示的地方和放大的翻番等,就用以MapController来决定地图。可以透过如下代码获得MapController对象:

     mMapController = mMapView.getController();   

万一定位地点,需要构建一个GeoPoint来表示地点的经度和纬度,然后运animateTo方法以地图定位到指定的GeoPoint上,代码如下:

     //设置起点为成都

     mGeoPoint=new
GeoPoint((int)(30.659259*1000000),(int)(104.065762*1000000));

     //定位到成都

     mMapController.animateTo(mGeoPoint);    

步骤6:Overlay的使用。

若果要在地图上标注有图标文字等信息,就得以Overlay。这里我们先是使将地图及之经度和纬度转换成为屏幕上实际的坐标,才会用消息绘制
上去。Map API中提供了Projection.toPixels(GeoPoint in, Point
out)方法,可以以经度和纬度转换成为屏幕及之坐标。首先需要贯彻Overlay中的draw方法才能够当地形图及绘制信息,代码如下:

     class MyLocationOverlay extends Overlay

     {

     public boolean draw(Canvas canvas,MapView mapView,boolean
shadow,long when)

     {

     //…

     }

     }     

下面是现身说法运行效果,图9-13以交通模式显示地图,图9-14以卫星模式显示地图,它们还在屏幕及亮了一个图标,并标明了岗位。

图片 9               

   图9-13  交通模式地图

图片 10                     

   图9-14  卫星模式地图

欠示例显示了成都城区的地形图,标注了天府广场的大体位置,具体实现而代码清单9-4所展示。

     代码清单9-4
 第9节\Examples_09_03\src\com\yarin\android\Examples_09_03\Activity01.java

     public class Activity01 extends MapActivity

     {

     private MapView mMapView;

     private MapController mMapController; 

     private GeoPoint mGeoPoint;

     public void onCreate(Bundle savedInstanceState)

     {

     super.onCreate(savedInstanceState);

     setContentView(R.layout.main);

     mMapView = (MapView) findViewById(R.id.MapView01);

     //设置也直通模式

     //mMapView.setTraffic(true);

     //设置为卫星模式

     mMapView.setSatellite(true); 

     //设置为街景模式

     //mMapView.setStreetView(false);

     //取得MapController对象(控制MapView)

     mMapController = mMapView.getController(); 

     mMapView.setEnabled(true);

     mMapView.setClickable(true);

     //设置地图支持缩放

     mMapView.setBuiltInZoomControls(true); 

     //设置起点为成都

     mGeoPoint=new
GeoPoint((int)(30.659259*1000000),(int)(104.065762*1000000));

     //定位及成都

     mMapController.animateTo(mGeoPoint); 

     //设置倍数(1-21)

     mMapController.setZoom(12); 

     //添加Overlay,用于展示标注信息

             MyLocationOverlay myLocationOverlay = new
MyLocationOverlay();

             List<Overlay> list = mMapView.getOverlays();

             list.add(myLocationOverlay);

     }

     protected boolean isRouteDisplayed()

     {

     return false;

     }

     class MyLocationOverlay extends Overlay

     {

     public boolean draw(Canvas canvas,MapView mapView,boolean
shadow,long when)

     {

     super.draw(canvas, mapView, shadow);

     Paint paint = new Paint();

     Point myScreenCoords = new Point();

     // 将透过纬度转换成实际上屏幕坐标

     mapView.getProjection().toPixels(mGeoPoint,myScreenCoords);

     paint.setStrokeWidth(1);

     paint.setARGB(255, 255, 0, 0);

     paint.setStyle(Paint.Style.STROKE);

     Bitmap bmp = BitmapFactory.decodeResource(getResources(), 

     R.drawable.home);

     canvas.drawBitmap(bmp,myScreenCoords.x,myScreenCoords.y,paint);

     canvas.drawText(“天府广场”,myScreenCoords.x, myScreenCoords.y,

     paint);

     return true;

     }

     }

     }

9.3.4  定位系统

全球定位系统(Global Positioning
System,GPS)又称之为全球卫星定位系统,是一个受离开圆型轨道卫星导航系统,它好吗地表面的多方域(98%)提供规范之稳定、测速及高
精度的辰标准。该体系由美国国防部研制及维护,可满足位于世界其他地方要邻近地空间的武装用户连续、精确地规定三维位置、三维运动与时空的急需。该体系包
括太空中之24发GPS卫星,地面上的1单主控站、3单数据注入站和5只监测站和作为用户端的GPS接收机。最少只待中3颗卫星,就能够很快确定用户端在
地球上所处的职及海拔高度。所能连至之卫星数更是多,解码出来的职位就一发规范。GPS广泛应用于部队、物流、地理、移动电话、数码相机、航空等世界,具
有老有力的效用,主要包括:

·精确定时:广泛应用在天文台、通信系统基站、电视台中。

·工程施工:道路、桥梁、隧道的施工中大量以GPS设备进行工程测量。

·勘探测绘:野外勘探及市区规划中都生因此到。

·导航。

···武器导航:精确制导导弹、巡航导弹。

···车辆导航:车辆调度、监控系统。 

···船舶导航:远洋导航、港口/内河引水。 

···飞机导航:航线导航、进场着陆控制。 

···星际导航:卫星轨道定位。 

···个人导航:个人旅游同野外探险。 

·定位。

···车辆防盗系统。 

···手机、PDA、PPC等通信移动设备防盗及电子地图、定位系统。 

···儿童及特殊人群的防走失系统。 

·精准农业:农机具导航、自动开与土地高精度平整。

Android
支持地理定位服务之API。该地理定位服务好为此来收获当前设施的地理位置,应用程序可以定时请求更新配备时之地理定位信息。比如应用程序可以凭一个
Intent接收器来实现如下效果:以通过纬度以及半径划定一个区域,当设备出入该区域时,发出提示信息,还好同Google
Map
API一起以,完成再多之任务。关于地理定位系统的API全部厕android.location包内,其中囊括以下几只举足轻重之机能看似:

·LocationManager:本类提供访问定位服务的功用,也供取最佳定位提供者的功能。另外,临近警报功能为得以凭该类来落实。

·LocationProvider:该类是原则性提供者的抽象类。定位提供者具备周期性报告设备地理位置的效力。

·LocationListener:提供固定信息发反时之回调功能。必须事先在稳定管理器中登记监听器对象。

·Criteria:该类使得用会由此以LocationProvider中装置的性质来选当的一贯提供者。

·Geocoder:用于拍卖地理编码和反往地理编码的好像。地理编码是指以地点或其他描述转变为经度和纬度,反向地理编码则是拿经度和纬度转变吗地
址或描述语言,其中包含了点滴单构造函数,需要传入经度和纬度的坐标。getFromLocation方法可以落一致组关于地方之数组。

一旦采取地理定位,首先需取LocationManager的实例,在Android中,获得LocationManager的绝无仅有办法是透过
getSystemService()方法的调用。通过使用LocationManager,我们可以得一个位置提供者的列表。在一个真的手握紧设备
中,这个列表包含了有GPS服务。我们也得以择重复有力、更准确、不带来起另外叠加服务之GPS。代码如下:

    
LocationManager?locationManager?=?(LocationManager)?getSystemService(Context.LOC-

     ATION_SERVICE);? 

落LocationManager对象下,我们还得登记一个周期性的翻新视图,代码如下:

    
locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER,1000,
0, 

     locationListener);    

内第一单参数是安装服务提供者,第二只参数是周期,这里需要重点说明一下说到底一个参数locationListener,它用来监听定位信息的改,所以我们得实现以下几独主意:

·onLocationChanged(Location
location):当坐标改变时触发此函数,如果Provider传进相同之坐标,它就非会见让触发。
  

·onProviderDisabled(String
provider):Provider禁用时触发此函数,比如GPS被关门。

·onProviderEnabled(String
provider):Provider启用时触发是函数,比如GPS被打开。   

·onStatusChanged(String provider, int status, Bundle
extras):Provider的转态在可用、暂时无可用和任服务三个状态一直切换时触发之函数。 

下面我们经过改变上平等节的事例(本书所附代码:第9章\Examples_09_04)来落实机关通过定位系统获取用户眼前的坐标,然后加载并展示地图,将坐标信息显示在一个TextView中,运行效果使图9-15所展示。

图片 11

贪图9-15 地图定位

倘使用固定的API,首先需以AndroidManifest.xml文件中补充加其权力,具体代码如代码清单9-5所出示。

     代码清单9-5  第9回\Examples_09_04\AndroidManifest.xml

     <?xml version=”1.0″ encoding=”utf-8″?>

     <manifest
xmlns:android=”http://schemas.android.com/apk/res/android”

           package=”com.yarin.android.Examples_09_04″

           android:versionCode=”1″

           android:versionName=”1.0″>

         <application android:icon=”@drawable/icon”
android:label=”@string/app_name”>

     <uses-library android:name=”com.google.android.maps” />

             <activity android:name=”.Activity01″

                       android:label=”@string/app_name”>

                 <intent-filter>

                     <action
android:name=”android.intent.action.MAIN” />

                     <category
android:name=”android.intent.category.LAUNCHER” />

                 </intent-filter>

             </activity>

         </application>

     <uses-permission android:name=”android.permission.INTERNET”/>

          <uses-permission
android:name=”android.permission.ACCESS_COARSE_LOCATION”/>

          <uses-permission
android:name=”android.permission.ACCESS_FINE_LOCATION”/>

         <uses-sdk android:minSdkVersion=”5″ />

     </manifest>

是因为我们于模拟器上测试,所以用人工设置一个坐标。可以经过简单栽方式来装一个模拟的坐标值。第一种植办法是透过DDMS,我们好于
Eclipse的ADT插件中运用这种方式,只要开动Eclipse,选择“Window”->“Show
View”,打开“Emulator
Control”界面即可看出如下的设置窗口,我们得手动或通过KML和GPX文件来安一个坐标。如图9-16所出示。

图片 12

祈求9-16  设置显示的坐标

另一样种植办法是运geo命令,我们要telnet到本机的5554端口,然后在指令行下输入类似于geo
fix-121.45356 46.51119 4392
这样的授命,后面3只参数分别代表了经度、纬度以及(可选的)海拔。设置后于Android模拟器屏幕及即多起一个如果图9-17所显示之标志,表示拟了
GPS权限。

图片 13

贪图9-17  GPS使用标志

本咱们可采用位置管理器(LocationManager)和职位提供者进行getFromLocation的调用。这个法返回本机当前职
的一个快照,这个快照将以
Location对象形式提供。在亲手执设备中,我们可落当前职的经度和纬度;调用getFromLocationName方法可能回一个数量,表
示一个地方的名号。该例中我们还创立了一个食谱用来缩放地图,这时便应用地图控制器(MapController)的zoomIn和zoomOut方法来
放大和缩小视图,具体实现而代码清单9-6所显示。

     代码清单9-6
 第9章节\Examples_09_04\src\com\yarin\android\Examples_09_04\Activity01.java

     public class Activity01 extends MapActivity 

     {

         public MapController mapController;

         public MyLocationOverlay myPosition;

         public MapView myMapView;

         private static final int ZOOM_IN=Menu.FIRST; 

         private static final int ZOOM_OUT=Menu.FIRST+1;

     

     public void onCreate(Bundle savedInstanceState) {

             super.onCreate(savedInstanceState);

             setContentView(R.layout.main);

             //取得LocationManager实例

             LocationManager locationManager;

             String context=Context.LOCATION_SERVICE;

             locationManager=(LocationManager)getSystemService(context);

             myMapView=(MapView)findViewById(R.id.MapView01);

             //取得MapController实例,控制地图

             mapController=myMapView.getController();

             //设置显示模式

             myMapView.setSatellite(true);

             myMapView.setStreetView(true);

             //设置缩放控制,这里我们好实现缩放菜单

             myMapView.displayZoomControls(false);   

             //设置以MyLocationOverlay来绘图

             mapController.setZoom(17);

             myPosition=new MyLocationOverlay();

             List<Overlay> overlays=myMapView.getOverlays();

             overlays.add(myPosition);

             //设置Criteria(服务商)的信息

             Criteria criteria =new Criteria();

             //经度要求

             criteria.setAccuracy(Criteria.ACCURACY_FINE);

             criteria.setAltitudeRequired(false);

             criteria.setBearingRequired(false);

             criteria.setCostAllowed(false);

             criteria.setPowerRequirement(Criteria.POWER_LOW);

             //取得成效太好的criteria

             String provider=locationManager.getBestProvider(criteria,
true);

             //得到坐标相关的音信

             Location
location=locationManager.getLastKnownLocation(provider);

             //更新坐标

             updateWithNewLocation(location);

             //注册一个周期性的更新,3000ms更新一次于

             //locationListener用来监听定位信息之改

             locationManager.requestLocationUpdates(provider, 3000,
0,locationListener);

         }

         private void updateWithNewLocation(Location location) 

         {

             String latLongString;

             TextView myLocationText =
(TextView)findViewById(R.id.TextView01);

             

             String addressString=”没有找到地方\n”;

             

             if(location!=null)

             {

                 //为绘制标志的接近设置坐标

                 myPosition.setLocation(location);

                 //取得经度和纬度

                 Double geoLat=location.getLatitude()*1E6;

                 Double geoLng=location.getLongitude()*1E6;

                 //将那变为int型

                 GeoPoint point=new
GeoPoint(geoLat.intValue(),geoLng.intValue());

                 //定位及指定坐标

                 mapController.animateTo(point);

                 double lat=location.getLatitude();

                 double lng=location.getLongitude();

                 latLongString=”经度:”+lat+”\n纬度:”+lng;

                 

                 double latitude=location.getLatitude();

                 double longitude=location.getLongitude();

                 //根据地理条件来规定编码

                 Geocoder gc=new Geocoder(this,Locale.getDefault());

                 try

                 {

                 //取得地方相关的一对音、经度、纬度

                     List<Address>
addresses=gc.getFromLocation(latitude, longitude,1);

                     StringBuilder sb=new StringBuilder();

                     if(addresses.size()>0)

                     {

                         Address address=addresses.get(0);

                         for(int
i=0;i<address.getMaxAddressLineIndex();i++)

                            
sb.append(address.getAddressLine(i)).append(“\n”);

                             

                            
sb.append(address.getLocality()).append(“\n”);

                            
sb.append(address.getPostalCode()).append(“\n”);

                             sb.append(address.getCountryName());

                             addressString=sb.toString();

                     }

                 }catch(IOException e){}

             }

             else

             {

                 latLongString=”没有找到坐标.\n”;

             }

             //显示

            
myLocationText.setText(“你眼前的坐标如下:\n”+latLongString+”\n”+addressString);

         }

         private final LocationListener locationListener=new
LocationListener()

         {

         //当坐标改变时触发此函数

             public void onLocationChanged(Location location)

             {

             updateWithNewLocation(location);

             }

             //Provider禁用时触发此函数,比如GPS被关门 

             public void onProviderDisabled(String provider)

             {

             updateWithNewLocation(null);

             }

             //Provider启用时触发之函数,比如GPS被打开

             public void onProviderEnabled(String provider){}

            
//Provider的转态在可用、暂时不可用和无劳动三单状态一直切换时触发是函数

             public void onStatusChanged(String provider,int
status,Bundle extras){}

         };

         protected boolean isRouteDisplayed()

     {

     return false;

     }

         //为应用程序添加菜单

         public boolean onCreateOptionsMenu(Menu menu)

     {

     super.onCreateOptionsMenu(menu);

     menu.add(0, ZOOM_IN, Menu.NONE, “放大”);

     menu.add(0, ZOOM_OUT, Menu.NONE, “缩小”);

     return true;

     }

         public boolean onOptionsItemSelected(MenuItem item)

     {

     super.onOptionsItemSelected(item);

     switch (item.getItemId())

     {

     case (ZOOM_IN):

     //放大

     mapController.zoomIn();

     return true;

     case (ZOOM_OUT):

     //缩小

     mapController.zoomOut();

     return true;

     }

     return true;

     }

     class MyLocationOverlay extends Overlay

     {

     Location mLocation;

     //在更新坐标时,设置该坐标,以便画图

     public void setLocation(Location location)

     {

     mLocation = location;

     }

     @Override

     public boolean draw(Canvas canvas,MapView mapView,boolean
shadow,long when)

     {

     super.draw(canvas, mapView, shadow);

     Paint paint = new Paint();

     Point myScreenCoords = new Point();

     // 将经纬度转换成实际上屏幕坐标

     GeoPoint tmpGeoPoint = new GeoPoint((int)(mLocation.

     getLatitude()*1E6),(int)(mLocation.getLongitude()*1E6));

     mapView.getProjection().toPixels(tmpGeoPoint,myScreenCoords);

     paint.setStrokeWidth(1);

     paint.setARGB(255, 255, 0, 0);

     paint.setStyle(Paint.Style.STROKE);

     Bitmap bmp = BitmapFactory.decodeResource(getResources(), 

     R.drawable.home);

     canvas.drawBitmap(bmp,myScreenCoords.x,myScreenCoords.y,paint);

     canvas.drawText(“Here am I”,myScreenCoords.x,myScreenCoords.

     y, paint);

     return true;

     }

     }

     }

9.4  桌面组件

率先差启动Android模拟器时,可以看在桌面上生诸多图标,如图9-18所出示之Google搜索框、时钟、联系人、浏览器等,点击这些图标,
系统即见面实行相应的先后,与PC操作系统桌面上之快捷方式很像,但是它不完全是快捷方式,还连了实时文件夹(Live
Folder)和桌面插件(Widget),这样既美观又有益于用户操作。本节以学习这每一样种桌面组件的付出,让我们好之应用程序也克轻松地停到桌面
上。

图片 14

希冀9-18 Android桌面组件

9.4.1  快捷方式

先是我们上最好基本的桌面组件快捷方式,它和PC上之快捷方式一样,用于启动某平等用到程序的某某组件(如Activity、Service等)。其
实要以桌面上补偿加一个快捷方式很简短,只待添加准桌面或者点击“Menu”按键(如图9-19所著),就可弹来上加桌面组件的选项项,如图9-20所
示,“Shortcuts”为丰富快捷方式,“Widgets”为Widget开发的桌面插件,“Folders”为实时文件夹,进入相应的选择项后即可加
加相应的桌面组件。

图片 15

       图9-19  Menu菜才                                  图9-20
 添加桌面组件

本小节紧要介绍于应用程序中经过代码来以一个应用程序添加到图9-20的Shortcuts列表中,这里丰富一个殡葬邮件的使至快捷方式列表上去(参见本书所附代码:第9回\Examples_09_05)。

首先需以Activity注册时上加一个Action为android.intent.action.CREATE_SHORTCUT的IntentFilter,如代码清单9-7所显示,添加之后列表中就是会现出该用的图标和名字了。

     代码清单9-7  第9章节\Examples_09_05\AndroidManifest.xml

     <?xml version=”1.0″ encoding=”utf-8″?>

     <manifest
xmlns:android=”http://schemas.android.com/apk/res/android”

           package=”com.yarin.android.Examples_09_05″

           android:versionCode=”1″

           android:versionName=”1.0″>

         <application android:icon=”@drawable/icon”
android:label=”@string/app_name”>

             <activity android:name=”.Activity01″

                       android:label=”@string/app_name”>

                 <intent-filter>

                   <action android:name=”android.intent.action.MAIN”
/>

                   <category
android:name=”android.intent.category.LAUNCHER” />

     <action
android:name=”android.intent.action.CREATE_SHORTCUT”/> 

                 </intent-filter>

             </activity>

         </application>

         <uses-sdk android:minSdkVersion=”5″ />

     </manifest>   

接下还要为快捷方式设置名字、图标、事件等性。Intent.EXTRA_SHORTCUT_NAME对应快捷方式的名
字;Intent.EXTRA_SHORTCUT_ICON_RESOURCE对应快捷方式的图标;Intent.
EXTRA_SHORTCUT_INTENT对许快捷方式执行的事件。需要验证的凡,Android专门供了
Intent.ShortcutIconResource.fromContext来创造快捷方式的图标,最后通过setResult来回到,构建一个快
捷方式,如代码清单9-8所显示。

     代码清单9-8
 第9节\Examples_09_05\src\com\yarin\android\Examples_09_05\Activity01.java

     public class Activity01 extends Activity

     {

     public void onCreate(Bundle savedInstanceState)

     {

     super.onCreate(savedInstanceState);

     //要抬高的快捷方式的Intent

     Intent addShortcut; 

     //判断是否要增长快捷方式

     if
(getIntent().getAction().equals(Intent.ACTION_CREATE_SHORTCUT))

     {

     addShortcut = new Intent(); 

     //设置快捷方式的讳

     addShortcut.putExtra(Intent.EXTRA_SHORTCUT_NAME,”发送邮件”);

     //构建快捷方式中专门的图标

     Parcelable icon = Intent.ShortcutIconResource.fromContext

     (this,R.drawable.mail_edit);  

     //添加快捷方式图标

     addShortcut.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE,icon);

     //构建快捷方式执行的Intent

     Intent mailto=new  Intent(Intent.ACTION_SENDTO, Uri.parse

     (“mailto:xxx@xxx.com” )); 

     //添加快捷方式Intent

     addShortcut.putExtra(Intent.EXTRA_SHORTCUT_INTENT,mailto);  

     //正常

     setResult(RESULT_OK,addShortcut);  

     }

     else

     {

     //取消

     setResult(RESULT_CANCELED);  

     }

     //关闭

     finish();  

     }

     }   

现在我们启动模拟器,就可于Shortcuts列表中找到所增长的快捷方式,将其上加至桌面,如图9-21所出示。

图片 16

祈求9-21  桌面快捷方式

9.4.2  实时文件夹

以Android 1.5面临,Live Folders无疑是一个遭到关注之初成效。简单地游说,Live
Folders就是一个查你的无绳电话机遭负有电子书、电子邮件、rss订阅、播放列表的快捷方式,并且这些情节是实时更新的。比如你不再需要独自打开电子邮
码软件查看邮件,打开通讯录找联系人等。Live
Folders自带了排有具有联系人、所有有电话号码的关联人及Starred联系人之功用,我们尚可动用Live
Folders API开发有再多的风靡应用。

是因为Live
Folders本身不存储任何信息,都是因炫耀的章程查看其ContentProvider所指向的数信息,并得以由定义显示格式,所以当源数据有改
变后,Live
Folders可以实时更新显示内容。那么当支付时,我们只要保管所指定数量信息URI的ContentProvider支持实时文件夹的询问。

夫丰富方式及长快捷方式一样,只是以增选时只要摘“Folders”。本小节我们通过Live
Folders调用电话仍中的消息,当点击其中同样长达信息时,便执行呼叫该联系人的动作(本书所附代码:第9段\
Examples_09_06)。

以及创快捷方式一样,我们得以Activity注册时加加一个Action动作为android.intent.action.
CREATE_LIVE_FOLDER的IntentFilter。代码如下:

                 <intent-filter>

     <action android:name=
“android.intent.action.CREATE_LIVE_FOLDER” /> 

     <category android:name= “android.intent.category.DEFAULT” /> 

                 </intent-filter>

咱俩得以程序中安装该实时文件夹的数据源、图标、名字的音讯。可以经intent.setData方法来安要读取的数据源,该例中我们设置数
据源为“content://contacts/live_folders/people”,即联系人消息。其他消息的装置如表9-2所出示。

表9-2  Live Folders的常用属性

图片 17

在安装图标时,Android专门供了Intent.ShortcutIconResource.fromContext来装实时文件夹的图标。下面我们以实时文件夹上加至桌面(如图9-22所示),运行效果使图9-23所出示。

图片 18

图9-22 “电话仍”实时文件夹                

图片 19

贪图9-23  实时文件夹运行效果

下用以onCreate方法中将实时文件夹的连锁消息装入Intent对象,并透过setResult方法设置也结果Intent,最后调用finish方法了Activity,把结果回到给Home应用程序,以补充加实时文件夹,如代码清单9-9所出示。

     代码清单9-9
 第9段\Examples_09_06\src\com\yarin\android\Examples_09_06\Activity01.java

     public class Activity01 extends Activity

     {

     public void onCreate(Bundle savedInstanceState)

     {

     super.onCreate(savedInstanceState);

     // setContentView(R.layout.main);

     // 判断是否创造实时文件夹

     if
(getIntent().getAction().equals(LiveFolders.ACTION_CREATE_LIVE_FOLDER))

     {

     Intent intent = new Intent();

     // 设置数据地址

     intent.setData(Uri.parse(“content://contacts/live_folders/

     people”));

     // 设置单击之后的事件,这里单击一个关系人后,呼叫

     intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_BASE_INTENT, 

     new Intent(Intent.ACTION_CALL,Contacts.People.CONTENT_URI));

     // 设置实时文件夹的讳

     intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_NAME,”电话本”);

     // 设置实行文件夹的图标

     intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_ICON, Intent.

     ShortcutIconResource.fromContext(this,R.drawable.contacts));

     // 设置显示模式为列表

     intent.putExtra(LiveFolders.EXTRA_LIVE_FOLDER_DISPLAY_MODE, 

     LiveFolders.DISPLAY_MODE_LIST);

     // 完成

     setResult(RESULT_OK, intent);

     }

     else

     {

     setResult(RESULT_CANCELED);

     }

     finish();

     }

     }

9.4.3  Widget开发

Widget是同种植好有点的应用程序,主要作Web
2.0服务要互联网内容的前端。Web设计人员和开发者可以使用Widget来创造最为让欢迎之互联网体验。在Android
1.5中进入了AppWidget
framework框架,开发者可以采用该框架开发Widget,这些Widget可以拖到用户之桌面并且可相互。Widget可以提供一个full-
featured
apps的预览,例如可以展示将来到之日历事件,或者同一首后高播放的歌的详细信息。当Widget被耽搁到桌面上经常,指定一个保留的半空中来显示应用提供
的自定义内容。用户可经过这Widget来与以交互,例如暂停或者切换歌曲。如果您来一个后台服务,可以按你协调的Schedule更新您的
Widget,或者采取AppWidget framework提供一个自动的创新机制。

每个Widget就是一个BroadcastReceiver,它们用XML
metadata来描述Widget的底细。AppWidget framework通过Broadcast
intents和Widget通信,
Widget的翻新使用RemoteViews来发送。RemoteViews被装进成一个layout和一定内容来展示到桌面上。下面我们经过一个演示
来上Widget开发(本书所附代码:第9段\Examples_09_07)。

先是得以res\layout目录下创造桌面组件的布局文件appwidget_provider.xml,用来显示桌面布局,这里我们创建一个TextView用来显示同一截文字,如代码清单9-10所展示。

     代码清单9-10
 第9节\Examples_09_07\res\layout\appwidget_provider.xml

     <?xml version=”1.0″ encoding=”utf-8″?>

     <TextView
xmlns:android=”http://schemas.android.com/apk/res/android”

         android:id=”@+id/appwidget_text” 

         android:textColor=”#ff000000″

         android:layout_width=”wrap_content”

         android:layout_height=”wrap_content”

     />

下一场要创造一个叙述是桌面组件属性之文本,存放到res\xml文件夹下,如代码清单9-11所展示。

     代码清单9-11
 第9章节\Examples_09_07\res\xmlappwidget_provider.xml

     <?xml version=”1.0″ encoding=”utf-8″?>

     <appwidget-provider
xmlns:android=”http://schemas.android.com/apk/res/android”

         android:minWidth=”100dp”

         android:minHeight=”50dp”

         android:updatePeriodMillis=”86400000″

         android:initialLayout=”@layout/appwidget_provider”

        
android:configure=”com.yarin.android.Examples_09_07.Activity01″

         >

     </appwidget-provider>

个中android:minWidth和android:minHeight分别指定了桌面组件的最好小幅度和最好小高度,其值可以依据期望之单元格数
量并运用前介绍过的公式来计量(最小尺寸=(单元格数×74)?2),android:updatePeriodMillis是自动更新的时日中
隔,android:initialLayout是Widget的界面描述文件。Android:configure是可选的,如果您的Widget需要
在启动前先行启动一个Activity,则用设定该项为卿的Activity。这里我们要事先输入一段文字,然后显示在Widget上。

下一场使树一个Widget,创建一个像样,让该后续类AppWidgetProvider。在AppWidgetProvider中产生成千上万办法,包
括onUpdate(周期更新时调用)、onDeleted(删除组件时调用)、onEnabled(当第一只零部件创建时调用)、
onDisabled(当最后一个零部件删除时调用),如代码清单9-12所著。

代码清单9-12
 第9节\Examples_09_07\src\com\yarin\android\Examples_09_07\ExampleAppWidget-
Provider.java

     public class ExampleAppWidgetProvider extends AppWidgetProvider

     {

     //周期更新时调用

     public void onUpdate(Context context,AppWidgetManager
appWidgetManager,int[] 

     appWidgetIds)

     {

     final int N = appWidgetIds.length;

     for (int i = 0; i < N; i++)

     {

     int appWidgetId = appWidgetIds[i];

     String titlePrefix=Activity01.loadTitlePref(context,appWidgetId);

     updateAppWidget(context, appWidgetManager, appWidgetId, 

     titlePrefix);

     }

     }

     //当桌面组件删除时调用

     public void onDeleted(Context context, int[] appWidgetIds)

     {

     //删除appWidget

     final int N = appWidgetIds.length;

     for (int i = 0; i < N; i++)

     {

     Activity01.deleteTitlePref(context, appWidgetIds[i]);

     }

     }

     //当AppWidgetProvider提供的率先个零部件创建时调用

     public void onEnabled(Context context)

     {

     PackageManager pm = context.getPackageManager();

     pm.setComponentEnabledSetting(new ComponentName(“com.yarin.android.

     Examples_09_07″, “.ExampleBroadcastReceiver”),

     PackageManager.COMPONENT_ENABLED_STATE_ENABLED, 

     PackageManager.DONT_KILL_APP);

     }

     //当AppWidgetProvider提供的终极一个零部件删除时调用

     public void onDisabled(Context context)

     {

     PackageManager pm = context.getPackageManager();

     pm.setComponentEnabledSetting(new ComponentName(“com.yarin.

     android.Examples_09_07″, “.ExampleBroadcastReceiver”),

     PackageManager.COMPONENT_ENABLED_STATE_ENABLED, 

     PackageManager.DONT_KILL_APP);

     }

     //更新

     static void updateAppWidget(Context context, AppWidgetManager 

     appWidgetManager, int appWidgetId, String titlePrefix)

     {

     //构建RemoteViews对象来针对桌面组件进行更新

     RemoteViews views = new RemoteViews(context.getPackageName(), 

     R.layout.appwidget_provider);

     //更新文本内容,指定布局之组件

     views.setTextViewText(R.id.appwidget_text, titlePrefix);

     //将RemoteViews的换代传入AppWidget进行创新

     appWidgetManager.updateAppWidget(appWidgetId, views);

     }

     }

内,在updateAppWidget方法吃我们构建了一个RemoteViews对象来对桌面组件进行更新,通过
setTextViewText方法来更新一个文本的来得,然后通过updateAppWidget方法来拿创新提供给AppWidget使该创新到桌
面。在onDisabled和onEnabled方法中我们所以ComponentName来代表应用程序中有组件的一体化名字。

末段,创建一个BroadcastReceiver类来收更新的信息,在吸收更新的信之后便创新是桌面Widget组件,如代码清单9-13所显示。

代码清单9-13
 第9章\Examples_09_07\src\com\yarin\android\Examples_09_07\ExampleBroadcast-
Receiver.java

     public class ExampleBroadcastReceiver extends BroadcastReceiver

     {

     public void onReceive(Context context, Intent intent)

     {

     //通过BroadcastReceiver来更新AppWidget

     String action = intent.getAction();

     if (action.equals(Intent.ACTION_TIMEZONE_CHANGED) ||
action.equals

     (Intent.ACTION_TIME_CHANGED))

     {

     AppWidgetManager gm = AppWidgetManager.getInstance(context);

     ArrayList<Integer> appWidgetIds = new
ArrayList<Integer>();

     ArrayList<String> texts = new ArrayList<String>();

     Activity01.loadAllTitlePrefs(context, appWidgetIds, texts);

     //更新所有AppWidget

     final int N = appWidgetIds.size();

     for (int i = 0; i < N; i++)

     {

     ExampleAppWidgetProvider.updateAppWidget(context, 

     gm, appWidgetIds.get(i), texts.get(i));

     }

     }

     }

     }

接下去,处理Android:configure指定的近乎,用来输入信息,在此类中我们监听者按钮,当点击按钮之后,创建一个
AppWidgetManager实例,然后调用ExampleAppWidgetProvider.updateAppWidget方法来更新是
Widget,通过以下代码可以取一个AppWidgetManager实例:

     AppWidgetManager appWidgetManager =
AppWidgetManager.getInstance(context);

在意,还用以AndroidManifest.xml中注册AppWidget、BroadcastReceiver和用来输入信息的Activity,如代码清单9-14所展示。

     代码清单9-14  第9段\Examples_09_07\AndroidManifest.xml

     <?xml version=”1.0″ encoding=”utf-8″?>

     <manifest
xmlns:android=”http://schemas.android.com/apk/res/android”

           package=”com.yarin.android.Examples_09_07″

           android:versionCode=”1″

           android:versionName=”1.0″>

         <application android:icon=”@drawable/icon”
android:label=”@string/app_name”>

             <receiver android:name=”.ExampleAppWidgetProvider”>

                 <meta-data android:name=”android.appwidget.provider”

                         android:resource=”@xml/appwidget_provider”
/>

                 <intent-filter>

                     <action
android:name=”android.appwidget.action.APPWIDGET_UPDATE” />

                 </intent-filter>

             </receiver>

             <activity android:name=”.Activity01″>

                 <intent-filter>

                     <action
android:name=”android.appwidget.action.APPWIDGET_CONFIGURE” />

                 </intent-filter>

             </activity>

             <receiver android:name=”.ExampleBroadcastReceiver”
android:enabled=”false”>

                 <intent-filter>

                     <action
android:name=”android.intent.ACTION_TIMEZONE_CHANGED” />

                     <action
android:name=”android.intent.ACTION_TIME” />

                 </intent-filter>

             </receiver>

         </application>

         <uses-sdk android:minSdkVersion=”5″ />

     </manifest>

脚将欠Widget添加至桌面上,和长快捷方式一样,如图9-24所出示,然后输入而出示的字,如图9-25所显示,点击“确定”按钮后,桌面即展示我们输入的消息,如图9-26所出示。

图片 20             

图9-24  添加Widget到桌面    

图片 21

图9-25  输入而显的信          

图片 22

希冀9-26  桌面显示Widget

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图
Copyright @ 2010-2018 亚洲必赢手机官网 版权所有