android指南针



    
        

             
               
                
            
        
    
    
        
        
    

package com.example.compass;
import java.util.Locale;
import android.graphics.drawable.AnimationDrawable;
import android.hardware.Sensor;
import android.hardware.SensorEvent;
import android.hardware.SensorEventListener;
import android.hardware.SensorManager;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.text.TextUtils;
import android.view.View;
import android.view.ViewGroup.LayoutParams;
import android.view.animation.AccelerateInterpolator;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.Toast;
import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.Context;
public class CompassActivity extends Activity {
 private static final int EXIT_TIME = 2000;// 两次按返回键的间隔判断
 private final float MAX_ROATE_DEGREE = 1.0f;// 最多旋转一周,即360°
 private SensorManager mSensorManager;// 传感器管理对象
 private Sensor mOrientationSensor;// 传感器对象
 private float mDirection;// 当前浮点方向
 private float mTargetDirection;// 目标浮点方向
 private AccelerateInterpolator mInterpolator;// 动画从开始到结束,变化率是一个加速的过程,就是一个动画速率
 protected final Handler mHandler = new Handler();
 private boolean mStopDrawing;// 是否停止指南针旋转的标志位
 private boolean mChinease;// 系统当前是否使用中文
 private long firstExitTime = 0L;// 用来保存第一次按返回键的时间
 View mCompassView;
 CompassView mPointer;// 指南针view
 LinearLayout mDirectionLayout;// 显示方向(东南西北)的view
 LinearLayout mAngleLayout;// 显示方向度数的view
 View mViewGuide;
 ImageView mGuideAnimation;
 @SuppressLint("HandlerLeak")
 protected Handler invisiableHandler = new Handler() {
  public void handleMessage(Message msg) {
   mViewGuide.setVisibility(View.GONE);
  }
 };
 public void onWindowFocusChanged(boolean hasFocus) {
  AnimationDrawable anim = (AnimationDrawable) mGuideAnimation
    .getDrawable();
  anim.start();
 }
 // 这个是更新指南针旋转的线程,handler的灵活使用,每20毫秒检测方向变化值,对应更新指南针旋转
 protected Runnable mCompassViewUpdater = new Runnable() {
  @Override
  public void run() {
   if (mPointer != null && !mStopDrawing) {
    if (mDirection != mTargetDirection) {
     // calculate the short routine
     float to = mTargetDirection;
     if (to - mDirection > 180) {
      to -= 360;
     } else if (to - mDirection < -180) {
      to += 360;
     }
     // limit the max speed to MAX_ROTATE_DEGREE
     float distance = to - mDirection;
     if (Math.abs(distance) > MAX_ROATE_DEGREE) {
      distance = distance > 0 ? MAX_ROATE_DEGREE
        : (-1.0f * MAX_ROATE_DEGREE);
     }
     // need to slow down if the distance is short
     mDirection = normalizeDegree(mDirection
       + ((to - mDirection) * mInterpolator
         .getInterpolation(Math.abs(distance) > MAX_ROATE_DEGREE ? 0.4f
           : 0.3f)));// 用了一个加速动画去旋转图片,很细致
     mPointer.updateDirection(mDirection);// 更新指南针旋转
    }
    updateDirection();// 更新方向值
    mHandler.postDelayed(mCompassViewUpdater, 20);// 20毫米后重新执行自己,比定时器好
   }
  }
 };
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  initResources();// 初始化view
  initServices();// 初始化传感器和位置服务
 }
 @Override
 public void onBackPressed() {// 覆盖返回键
  long curTime = System.currentTimeMillis();
  if (curTime - firstExitTime < EXIT_TIME) {// 两次按返回键的时间小于2秒就退出应用
   finish();
  } else {
   Toast.makeText(this, "再按一次退出", Toast.LENGTH_SHORT).show();
   firstExitTime = curTime;
  }
 }
 @Override
 protected void onResume() {// 在恢复的生命周期里判断、启动位置更新服务和传感器服务
  super.onResume();
  if (mOrientationSensor != null) {
   mSensorManager.registerListener(mOrientationSensorEventListener,
     mOrientationSensor, SensorManager.SENSOR_DELAY_GAME);
  } else {
    Toast.makeText(this, "不能找到传感器!",
    Toast.LENGTH_SHORT)
    .show();
  }
  mStopDrawing = false;
  mHandler.postDelayed(mCompassViewUpdater, 20);// 20毫秒执行一次更新指南针图片旋转
 }
 @Override
 protected void onPause() {// 在暂停的生命周期里注销传感器服务和位置更新服务
  super.onPause();
  mStopDrawing = true;
  if (mOrientationSensor != null) {
   mSensorManager.unregisterListener(mOrientationSensorEventListener);
  }
 }
 // 方向传感器变化监听
 private SensorEventListener mOrientationSensorEventListener = new SensorEventListener() {
  @Override
  public void onSensorChanged(SensorEvent event) {
   float direction = event.values[mSensorManager.DATA_X] * -1.0f;
   mTargetDirection = normalizeDegree(direction);// 赋值给全局变量,让指南针旋转
  }
  @Override
  public void onAccuracyChanged(Sensor sensor, int accuracy) {
  }
 };
 // 初始化view
 private void initResources() {
  mViewGuide = findViewById(R.id.view_guide);
  mViewGuide.setVisibility(View.VISIBLE);
  invisiableHandler.sendMessageDelayed(new Message(), 3000);
  mGuideAnimation = (ImageView) findViewById(R.id.guide_animation);
  mDirection = 0.0f;// 初始化起始方向
  mTargetDirection = 0.0f;// 初始化目标方向
  mInterpolator = new AccelerateInterpolator();// 实例化加速动画对象
  mStopDrawing = true;
  mChinease = TextUtils.equals(Locale.getDefault().getLanguage(), "zh");// 判断系统当前使用的语言是否为中文
  mCompassView = findViewById(R.id.view_compass);// 实际上是一个LinearLayout,装指南针ImageView和位置TextView
  mPointer = (CompassView) findViewById(R.id.compass_pointer);// 自定义的指南针view
  mDirectionLayout = (LinearLayout) findViewById(R.id.layout_direction);// 顶部显示方向名称(东南西北)的LinearLayout
  mAngleLayout = (LinearLayout) findViewById(R.id.layout_angle);// 顶部显示方向具体度数的LinearLayout
 }
 // 初始化传感器和位置服务
 private void initServices() {
  // sensor manager
  mSensorManager = (SensorManager) getSystemService(Context.SENSOR_SERVICE);
  mOrientationSensor = mSensorManager.getSensorList(
    Sensor.TYPE_ORIENTATION).get(0);
 }
 // 调整方向传感器获取的值
 private float normalizeDegree(float degree) {
  return (degree + 720) % 360;
 }
 // 更新顶部方向显示的方法
 private void updateDirection() {
  LayoutParams lp = new LayoutParams(LayoutParams.WRAP_CONTENT,
    LayoutParams.WRAP_CONTENT);
  // 先移除layout中所有的view
  mDirectionLayout.removeAllViews();
  mAngleLayout.removeAllViews();
  // 下面是根据mTargetDirection,作方向名称图片的处理
  ImageView east = null;
  ImageView west = null;
  ImageView south = null;
  ImageView north = null;
  float direction = normalizeDegree(mTargetDirection * -1.0f);
  if (direction > 22.5f && direction < 157.5f) {
   // east
   east = new ImageView(this);
   east.setImageResource(mChinease ? R.drawable.e_cn : R.drawable.e);
   east.setLayoutParams(lp);
  } else if (direction > 202.5f && direction < 337.5f) {
   // west
   west = new ImageView(this);
   west.setImageResource(mChinease ? R.drawable.w_cn : R.drawable.w);
   west.setLayoutParams(lp);
  }
  if (direction > 112.5f && direction < 247.5f) {
   // south
   south = new ImageView(this);
   south.setImageResource(mChinease ? R.drawable.s_cn : R.drawable.s);
   south.setLayoutParams(lp);
  } else if (direction < 67.5 || direction > 292.5f) {
   // north
   north = new ImageView(this);
   north.setImageResource(mChinease ? R.drawable.n_cn : R.drawable.n);
   north.setLayoutParams(lp);
  }
  // 下面是根据系统使用语言,更换对应的语言图片资源
  if (mChinease) {
   if (east != null) {
    mDirectionLayout.addView(east);
   }
   if (west != null) {
    mDirectionLayout.addView(west);
   }
   if (south != null) {
    mDirectionLayout.addView(south);
   }
   if (north != null) {
    mDirectionLayout.addView(north);
   }
  } else {
   // north/south should be before east/west
   if (south != null) {
    mDirectionLayout.addView(south);
   }
   if (north != null) {
    mDirectionLayout.addView(north);
   }
   if (east != null) {
    mDirectionLayout.addView(east);
   }
   if (west != null) {
    mDirectionLayout.addView(west);
   }
  }
  // 下面是根据方向度数显示度数图片数字
  int direction2 = (int) direction;
  boolean show = false;
  if (direction2 >= 100) {
   mAngleLayout.addView(getNumberImage(direction2 / 100));
   direction2 %= 100;
   show = true;
  }
  if (direction2 >= 10 || show) {
   mAngleLayout.addView(getNumberImage(direction2 / 10));
   direction2 %= 10;
  }
  mAngleLayout.addView(getNumberImage(direction2));
  // 下面是增加一个°的图片
  ImageView degreeImageView = new ImageView(this);
  degreeImageView.setImageResource(R.drawable.degree);
  degreeImageView.setLayoutParams(lp);
  mAngleLayout.addView(degreeImageView);
 }
 // 获取方向度数对应的图片,返回ImageView
 private ImageView getNumberImage(int number) {
  ImageView p_w_picpath = new ImageView(this);
  LayoutParams lp = new LayoutParams(LayoutParams.WRAP_CONTENT,
    LayoutParams.WRAP_CONTENT);
  switch (number) {
  case 0:
   p_w_picpath.setImageResource(R.drawable.number_0);
   break;
  case 1:
   p_w_picpath.setImageResource(R.drawable.number_1);
   break;
  case 2:
   p_w_picpath.setImageResource(R.drawable.number_2);
   break;
  case 3:
   p_w_picpath.setImageResource(R.drawable.number_3);
   break;
  case 4:
   p_w_picpath.setImageResource(R.drawable.number_4);
   break;
  case 5:
   p_w_picpath.setImageResource(R.drawable.number_5);
   break;
  case 6:
   p_w_picpath.setImageResource(R.drawable.number_6);
   break;
  case 7:
   p_w_picpath.setImageResource(R.drawable.number_7);
   break;
  case 8:
   p_w_picpath.setImageResource(R.drawable.number_8);
   break;
  case 9:
   p_w_picpath.setImageResource(R.drawable.number_9);
   break;
  }
  p_w_picpath.setLayoutParams(lp);
  return p_w_picpath;
 }
}
package com.example.compass;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.drawable.Drawable;
import android.util.AttributeSet;
import android.widget.ImageView;
public class CompassView extends ImageView {
 private float mDirection;// 方向旋转浮点数
 private Drawable compass;// 图片资源
 // 三个构造器
 public CompassView(Context context) {
  super(context);
  mDirection = 0.0f;// 默认不旋转
  compass = null;
 }
 public CompassView(Context context, AttributeSet attrs) {
  super(context, attrs);
  mDirection = 0.0f;
  compass = null;
 }
 public CompassView(Context context, AttributeSet attrs, int defStyle) {
  super(context, attrs, defStyle);
  mDirection = 0.0f;
  compass = null;
 }
 @Override
 protected void onDraw(Canvas canvas) {
  if (compass == null) {
   compass = getDrawable();// 获取当前view的图片资源
   compass.setBounds(0, 0, getWidth(), getHeight());// 图片资源在view的位置,此处相当于充满view
  }
  canvas.save();
  canvas.rotate(mDirection, getWidth() / 2, getHeight() / 2);// 绕图片中心点旋转,
  compass.draw(canvas);// 把旋转后的图片画在view上,即保持旋转后的样子
  canvas.restore();// 保存一下
 }
 /**
  * 自定义更新方向的方法
  * 
  * @param direction
  *            传入的方向
  */
 public void updateDirection(float direction) {
  mDirection = direction;
  invalidate();// 重新刷新一下,更新方向
 }
}

文章名称:android指南针
文章源于:http://ybzwz.com/article/pdcchi.html