Windows Phone笔记(5)加速计和位置服务(转)

Windows Phone笔记(5)加速计和位置服务

 

  这篇笔记主要是讲述如Windows Phone中两个提供外界信息的设备以及如何在程序中适用它们,在用户授权允许的情况下,加速计能够获取当前手机设备的具体方向;而位置服务则可以对手机当前所在地的位置进行定位。

1.传感器之加速计

  加速计时Windows Phone手机中的一个硬件设备,加速计测量在某一时刻施加于设备的力。可以使用这些力来确定用户正在向哪个方向移动设备。加速度值采用 3 维矢量表示,该矢量表示在 X、Y 和 Z 轴中的加速度分量(采用重力单位)。当设备面朝平台时,加速度的方向相对于设备以便对 Z 轴应用 -1g,当垂直于平台顶部放置设备时,对 Y 轴应用 -1g。从加速计的这些特性我们可以知道,Windows Phone中的程序中处理方向改变的基础。加速度也会对突然的移动做出响应,比如摇晃(微信中的摇一摇功能)等等。对于加速计的使用,创意是对开发人员最大的挑战

 

  程序使用加速计需要引用Microsoft.Devices.Sensors库,并且在Properties文件夹中的WMAppManifest.xml文件中设置(这是默认设置):

<Capability Name="ID_CAP_SENSORS"/>

 

下面让我们通过观察一个示例程序,该程序获取加速计测量的数据,并使用Line元素和TextBlock元素展示出来:

 这是MainPage.xaml页面的代码:

复制代码
 1   <!--ContentPanel - 在此处放置其他内容-->
2 <Grid x:Name="ContentPanel" Grid.Row="1" Margin="12,0,12,0">
3 <StackPanel Orientation="Vertical">
4 <StackPanel Orientation="Horizontal">
5 <TextBlock Text="状态:" Margin="10 0"/>
6 <TextBlock Name="txtblkAccelerometerState" Text="停止采集数据"/>
7 </StackPanel>
8 <StackPanel Orientation="Horizontal">
9 <TextBlock Text="更新加度计数据时间:" Margin="10 0"/>
10 <TextBlock Name="txtblkupdateDataAccelerometer" Text="20ms"/>
11 </StackPanel>
12 <Grid>
13 <TextBlock Height="45" HorizontalAlignment="Left" Name="txtblkX" Text="X:1.0" VerticalAlignment="Top" Foreground="Red" FontWeight="Bold" FontSize="28" Margin="10 0" />
14 <TextBlock Height="45" HorizontalAlignment="Center" Name="txtblkY" Text="Y:1.0" VerticalAlignment="Top" Foreground="Green" FontWeight="Bold" FontSize="28" />
15 <TextBlock Height="45" HorizontalAlignment="Right" Name="txtblkZ" Text="Z:1.0" VerticalAlignment="Top" Foreground="Blue" FontWeight="Bold" FontSize="28" Margin="10 0" />
16 </Grid>
17 <Grid Height="300">
18 <Line x:Name="xLine" X1="240" Y1="150" X2="340" Y2="150" Stroke="Red" StrokeThickness="4"></Line>
19 <Line x:Name="yLine" X1="240" Y1="150" X2="240" Y2="60" Stroke="Green" StrokeThickness="4"></Line>
20 <Line x:Name="zLine" X1="240" Y1="150" X2="190" Y2="200" Stroke="Blue" StrokeThickness="4"></Line>
21 </Grid>
22 <Button Content="开始采集" Margin="0 100" Name="btnSwitch" Click="btnSwitch_Click"></Button>
23 </StackPanel>
24 </Grid>
复制代码

这是MainPage.xaml.cs后台处理程序:

复制代码
 public partial class MainPage : PhoneApplicationPage
{
Accelerometer accelerometer;
DispatcherTimer timer;
Vector3 acceleration;
bool isDataValid;
// 构造函数
public MainPage()
{
InitializeComponent();

if (!Accelerometer.IsSupported)
{
txtblkAccelerometerState.Text = "设备不支持加速计";
}
else
{
//初始化timer,并绑定Tick事件
timer = new DispatcherTimer();
timer.Interval = TimeSpan.FromMilliseconds(30);
timer.Tick += new EventHandler(timer_Tick);
}
}

/// <summary>
/// 开始和停止采集加速计数据
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btnSwitch_Click(object sender, RoutedEventArgs e)
{
if (accelerometer != null && accelerometer.IsDataValid)
{
//停止采集加速计数据
accelerometer.Stop();
timer.Stop();
txtblkAccelerometerState.Text = "停止加速度度计";
btnSwitch.Content = "开始采集数据";
}
else
{
if (accelerometer == null)
{
//实例化一个accelerometer对象
accelerometer = new Accelerometer();

//20毫秒更新一次数据
accelerometer.TimeBetweenUpdates = TimeSpan.FromMilliseconds(20);

//更新传感器数据的时间
txtblkupdateDataAccelerometer.Text = accelerometer.TimeBetweenUpdates.TotalMilliseconds + " ms";
//从传感器获得新数据时发生
accelerometer.CurrentValueChanged += new EventHandler<SensorReadingEventArgs<AccelerometerReading>>(accelerometer_CurrentValueChanged);
}

try
{
txtblkAccelerometerState.Text = "开始采集数据";
btnSwitch.Content = "停止采集数据";
accelerometer.Start();
timer.Start();
}
catch (InvalidOperationException)
{
txtblkAccelerometerState.Text = "无法启动加速计.";
}
}
}

void accelerometer_CurrentValueChanged(object sender, SensorReadingEventArgs<AccelerometerReading> e)
{
//这个事件处理程序是被主执行线程(UI)之外的线程调用的
//不能直接访问页面上的元素,因为它们位于UI线程中
isDataValid = accelerometer.IsDataValid;
acceleration = e.SensorReading.Acceleration;
}

void timer_Tick(object sender, EventArgs e)
{
if (isDataValid)
{
txtblkAccelerometerState.Text = "正在从加速计获取数据";

// 显示加速计的数值
txtblkX.Text = "X: " + acceleration.X.ToString("0.00");
txtblkY.Text = "Y: " + acceleration.Y.ToString("0.00");
txtblkZ.Text = "Z: " + acceleration.Z.ToString("0.00");

// 使用Line元素显示加速计返回的值
xLine.X2 = xLine.X1 + acceleration.X * 100;
yLine.Y2 = yLine.Y1 - acceleration.Y * 100;
zLine.X2 = zLine.X1 - acceleration.Z * 50;
zLine.Y2 = zLine.Y1 + acceleration.Z * 50;
}
}

}
复制代码

编译运行上面的程序:

  这是运行程序后的效果(没有开始采集数据)    点击开始采集数据后,Y等于-1g,表示手机(模拟器)是纵向竖立的

    

但这样看好像并不能直观的了解加速计的作用,我们可以通过使用该模拟器的加速计模拟器工具来测试应用程序。

点击模拟器的加速计工具的“播放按钮”或者是拖动加速计工作中手机上的桔红色圆点,模拟手机的各个移动状态。我们可看到页面的3个分别代表着X/Y/Z的三个Line元素随着加速计的数据变化而不断变化。

 

2.位置服务(Location Service)  

Windows Phone应用程序可以通过一种被称为A-GPS(Assisted-GPS,辅助GPS)的技术获取手机当前所在的地理位置。对手机进行定位所实用的核心类是:GeocoordinateWatcher。使用时需要引用System.Device.DLL程序集,并引用System.Device.Location命名空间。WMAppManifest.xml做如下标记(默认包含):

<Capability Name="ID_CAP_LOCTION"/>

GeocoordinateWatcher的构造方法可以接收一个GetPositionAccuracy枚举类型的参数,该枚举的成员有:

  • Default(默认精度)
  • High(高精度)

GeocoordinateWatcher对象需要注册一个名为:PositionChanged事件,检测到位置更改时发生。PostitionChanged事件会传递一个GeoCoordinate对象,该对象有八个属性,分别是:

  • Latitude(纬度),double类型,-90至90之间。
  • Longitude(经度),double类型,-180到180之间
  • Altitude(高度),double类型
  • HorizontalAccuracy(水平精度)VerticalAccuracy(竖直精度),double类型
  • Course(航向),double类型,0至360之间
  • Speed(速度),double类型
  • IsUnknown,Boolean类型,当Latitude(纬度)和Longitude(经度)为非数字时,则为ture

GeoCoordinate类有一个方法GetDistanceTo,用于计算两个GeoCoordinate对象的距离

北纬为正值,南纬为负值;东经为正值,西经为负值如果应用程序没有获取用户位置的授权,Latitude(纬度)和Longitude(经度)的值为Double.NaN。

   

现在我们通过一个示例来学习如何使用Windows Phone中的位置服务。

MainPage.xaml

复制代码
 1     <!--ContentPanel - 在此处放置其他内容-->
2 <Grid x:Name="ContentPanel" Grid.Row="1" Margin="12,0,12,0">
3 <StackPanel >
4 <StackPanel Orientation="Horizontal">
5 <TextBlock Height="30" Name="textBlock1" Text="状态:" Margin="10 0" />
6 <TextBlock Height="30" Name="txtblkLocationState" Text="位置服务当前不可用" />
7 </StackPanel>
8 <StackPanel>
9 <Grid Height="500">
10 <TextBlock Text="经度:" FontSize="28" Foreground="Red" HorizontalAlignment="Left" VerticalAlignment="Top"/>
11 <TextBlock Text="0.000" FontSize="28" Foreground="Red" HorizontalAlignment="Center" VerticalAlignment="Top" Name="txtblkLatitude" Margin="0 0 200 0" />
12 <TextBlock Text="纬度:" FontSize="28" Foreground="Red" HorizontalAlignment="Left" VerticalAlignment="Center" />
13 <TextBlock Text="0.000" FontSize="28" Foreground="Red" HorizontalAlignment="Center" VerticalAlignment="Center" Name="txtblkLongitude" Margin="0 0 200 0"/>
14 </Grid>
15 </StackPanel>
16 <Grid>
17 <Button Content="开始定位" Height="72" Name="btnStartLocation" Width="160" HorizontalAlignment="Left" VerticalAlignment="Bottom" Margin="50 0" Click="btnStartLocation_Click" />
18 <Button Content="停止定位" Height="72" Name="btnStopLocation" Width="160" HorizontalAlignment="Right" VerticalAlignment="Bottom" Margin="50 0" Click="btnStopLocation_Click" />
19 </Grid>
20 </StackPanel>
21 </Grid>
复制代码

这里是MainPage.xaml.cs后台处理程序:

复制代码
 1 public partial class MainPage : PhoneApplicationPage
2 {
3 GeoCoordinateWatcher watcher;
4 // 构造函数
5 public MainPage()
6 {
7 InitializeComponent();
8 }
9
10 /// <summary>
11     /// 开始定位
12      /// </summary>
13      /// <param name="sender"></param>
14      /// <param name="e"></param>
15 private void btnStartLocation_Click(object sender, RoutedEventArgs e)
16 {
17 if (watcher == null)
18 {
19 watcher = new GeoCoordinateWatcher(GeoPositionAccuracy.High); // 采用高精度
20 watcher.MovementThreshold = 20; // PositionChanged事件之间传送的最小距离
21
22 watcher.StatusChanged += new EventHandler<GeoPositionStatusChangedEventArgs>(watcher_StatusChanged);
23 watcher.PositionChanged += new EventHandler<GeoPositionChangedEventArgs<GeoCoordinate>>(watcher_PositionChanged);
24
25 watcher.Start();//开始使用位置服务
26 }
27 else//方便测试,实际使用需要注意
28 {
29 watcher.Start();//开始使用位置服务
30 }
31 }
32
33 //检测到位置更改时
34 //当定位服务已准备就绪并接收数据时,它将开始引发 PositionChanged 事件
35 void watcher_PositionChanged(object sender, GeoPositionChangedEventArgs<GeoCoordinate> e)
36 {
37 this.txtblkLatitude.Text = e.Position.Location.Latitude.ToString("0.000");
38 this.txtblkLongitude.Text = e.Position.Location.Longitude.ToString("0.000");
39 }
40
41 //当位置服务状态发生变化时
42      //在 GeoPositionStatusChangedEventArgs 对象中传递的 GeoPositionStatus 枚举获取该服务的当前状态。
43      //可以使用它在应用程序中启用基于位置的功能,以及将服务的当前状态通知给用户。
44 void watcher_StatusChanged(object sender, GeoPositionStatusChangedEventArgs e)
45 {
46 switch (e.Status)
47 {
48 //如果服务的状态为 Disabled,则可以检查 Permission 属性,看用户是否禁用了应用程序的定位服务功能。
49 case GeoPositionStatus.Disabled:
50 if (watcher.Permission == GeoPositionPermission.Denied)
51 {
52 //用户禁用了定位服务
53 this.txtblkLocationState.Text = "对位置服务的访问被拒绝。";
54 }
55 else
56 {
57 this.txtblkLocationState.Text = "设备的定位服务不能够正常使用。";
58 }
59 break;
60 case GeoPositionStatus.Initializing:
61 // 位置服务正在尝试获取数据
62 this.btnStartLocation.IsEnabled = false;
63 break;
64 case GeoPositionStatus.NoData:
65 this.txtblkLocationState.Text = "当前位置无法进行定位";
66 this.btnStopLocation.IsEnabled = true;
67 break;
68 case GeoPositionStatus.Ready:
69 this.txtblkLocationState.Text = "位置服务已启用,并准备就绪";
70 this.btnStopLocation.IsEnabled = true;
71 break;
72 }
73 }
74
75 /// <summary>
76      /// 停止定位
77      /// </summary>
78      /// <param name="sender"></param>
79      /// <param name="e"></param>
80 private void btnStopLocation_Click(object sender, RoutedEventArgs e)
81 {
82 watcher.Stop();
83 this.btnStartLocation.IsEnabled = true;
84 this.txtblkLocationState.Text = "位置服务已经停止";
85 }
86 }
复制代码

编译运行程序后,通过模拟器可以模拟当前位置,通过使用模拟器中模拟位置工具变化当前设备所在坐标,我们可以看到程序上的数字也随之不断变化:

 

参考资料:

  http://msdn.microsoft.com/zh-cn/library/ff431810(v=vs.92).aspx(重要)

  http://msdn.microsoft.com/zh-cn/library/hh202933(v=vs.92).aspx

  http://msdn.microsoft.com/zh-cn/library/ff431782(v=vs.92).aspx(重要)

  http://msdn.microsoft.com/zh-cn/library/ee425989(v=VS.100).aspx

  《Programming Windows Phone 7 Microsoft Silverlight Edition》

作者:晴天猪

出处:http://www.cnblogs.com/IPrograming

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

posted on 2012-05-03 17:14  voker  阅读(274)  评论(0编辑  收藏  举报

导航