android之Display.getRotation()_传感器控制屏幕旋转

在看android自带的samples源码里面的AccelerometerPlayActivity时,看到下面这段代码,很不理解

[java][/java] view plaincopyprint?

  1. public void onSensorChanged(SensorEvent event) {
  2.     if(event.sensor.getType() != Sensor.TYPE_ACCELEROMETER){
  3.         return;
  4.     }
  5.     switch (mDisplay.getRotation()) {
  6.         case Surface.ROTATION_0://手机处于正常状态
  7.                   mSensorX = event.values[0];
  8.                 mSensorY = event.values[1];
  9.                 break;
  10.             case Surface.ROTATION_90://手机旋转90度
  11.                   mSensorX = -event.values[1];
  12.                 mSensorY = event.values[0];
  13.                 break;
  14.             case Surface.ROTATION_180:
  15.                 mSensorX = -event.values[0];
  16.                 mSensorY = -event.values[1];
  17.                 break;
  18.             case Surface.ROTATION_270:
  19.                 mSensorX = event.values[1];
  20.                 mSensorY = -event.values[0];
  21.                 break;
  22.     }
  23. }

经过查阅资料大体了解了
通过AndroidManifest.xml设置屏幕方向的话,安装后就不能改变,而程序内部设置屏幕方向就不会有这个限制。主要靠这两个API:getRequestedOrientation()和setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT)这两个API通过ActivityManagerService.java的转换后,实际上都是调用的WindowManagerService的同名方法。

每个Activity在WindowManagerService端都有一个AppWindowToken做代表,而屏幕的方向信息就存储在这里。
PhoneWindowManager会自动根据屏幕物理特性决定屏幕方向,看这段代码:

[java][/java] view plaincopyprint?

  1. if (mPortraitRotation < 0) {
  2.     // Initialize the rotation angles for each orientation once.
  3.     Display d = ((WindowManager)mContext.getSystemService(Context.WINDOW_SERVICE))
  4.             .getDefaultDisplay();
  5.     if (d.getWidth() > d.getHeight()) {
  6.         mPortraitRotation = Surface.ROTATION_90;
  7.         mLandscapeRotation = Surface.ROTATION_0;
  8.         mUpsideDownRotation = Surface.ROTATION_270;
  9.         mSeascapeRotation = Surface.ROTATION_180;
  10.     } else {
  11.         mPortraitRotation = Surface.ROTATION_0;
  12.         mLandscapeRotation = Surface.ROTATION_90;
  13.         mUpsideDownRotation = Surface.ROTATION_180;
  14.         mSeascapeRotation = Surface.ROTATION_270;
  15.     }
  16. }

这里的d.getWidth() 和 d.getHeight()得到的是物理屏幕的宽高。一般来说,平板和手机的是不一样的。

平板是宽比高大(0度时位于landscape模式,右转90度进入porit模式),手机是高比宽大(0度是位于porit模式,右转90度进入landscape模式)。如果应用程序只关心当前是横屏还是竖屏,而不直接使用传感器的话,没什么问题。如果像依靠重力感应的游戏那样直接使用传感器,就需要自己根据物理屏幕的坐标系对传感器数据做转化,否则就会出现坐标系混乱的问题。

如果没有没有通过上面的d.getWidth()和d.getHeight()来检测设备的物理屏幕从确定哪个是landscape和porit模式,而是直接假设设备是和手机一样的模式。由于游戏运行在landscape模式下,它们都把传感器数据右转90度。这样做法在手机上是没有问题,但在平板电脑上是不应该转化的,这是因为物理屏幕宽比高大的情况下,默认就是landscape模式。

 

现在回到源代码,在这里没有区分手机和平板,仅仅是用来转换加速度的方向而已,也没有必要区分.

[java][/java] view plaincopyprint?

  1. case Surface.ROTATION_0://手机处于正常状态
  2.                 mSensorX = event.values[0];
  3.                 mSensorY = event.values[1];
  4.                 break;

这段就是如果手机的方向没有旋转,不管手机处于landscape还是porit模式,加速度的方向都不用变,而下面,如果手机旋转了180度,说明x轴和y轴的方向完全反过来了,这时候对于加速度的方向就要调整到反向.还有90度和270度的情况都类似.

[java][/java] view plaincopyprint?

  1. case Surface.ROTATION_180:
  2.                 mSensorX = -event.values[0];
  3.                 mSensorY = -event.values[1];
  4.                 break;

标签