自从写完那个自定义菊花加载view后,又想着来一个类似的,这种知识点可以用在很多地方,如果手表,水表、还有今天要说的指南针等,只要是带刻度的都可以。
首先我们考虑下这个指南针要包括什么:
1、方位角刻度:也就是我当前位置和目的位置的方位角,方位角是与北方向相比偏离的角度。
2、东南西北不同颜色区分
3、根据手的水平移动,指南针view也要对应移动
获取指南针角度,可用SensorManager,这个是个系统服务,即可实时获取当前指向角度了。注册代码:
if(sensorManager==null){
sensorManager = (SensorManager) context.getSystemService(SENSOR_SERVICE);
//通过 getDefaultSensor 获得指南针传感器
sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ORIENTATION);
//为传感器管理者注册监听器,第三个参数指获取速度正常
sensorManager.registerListener(this, sensor, SensorManager.SENSOR_DELAY_GAME);
这里我注册的是SENSOR_DELAY_GAME,游戏级别的传感器,之前我用的是SENSOR_DELAY_NORMAL默认的,感觉没游戏级别的好。
画刻度还是老样子,for循环360度,每一度画一个刻度,画完后将canvas旋转一度
for (int i = 0; i < 360; i++) {
canvas.drawLine(x, y - r, x, (float) (y - r + x * 0.2), mPaint);
canvas.rotate(360 / 360, x, y);
至于其他特别的刻度比如红色刻度、长一些的刻度,我们可以做判断也可以另起for循环画,达到覆盖他的效果。话不多说直接上源码:
public class DemoZhiNaZhen extends View implements SensorEventListener {
private int mWidth;
private int mHeigth;
private Sensor sensor;
private SensorManager sensorManager;
private int toDegrees;
private Paint mPaint;
public void setCureentAngla(int mCureentAngla) {
this.mCureentAngla = mCureentAngla;
if(sensorManager==null){
sensorManager = (SensorManager) context.getSystemService(SENSOR_SERVICE);
//通过 getDefaultSensor 获得指南针传感器
sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ORIENTATION);
//为传感器管理者注册监听器,第三个参数指获取速度正常
sensorManager.registerListener(this, sensor, SensorManager.SENSOR_DELAY_GAME);
private int mCureentAngla=181;
private Paint mBitmapPaint;
private Paint mTextPaint;
private Context context;
public DemoZhiNaZhen(Context context) {
this(context, null);
public DemoZhiNaZhen(Context context, @Nullable AttributeSet attrs) {
this(context, attrs, 0);
public DemoZhiNaZhen(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
this.context=context;
mPaint = new Paint();
mPaint.setDither(true);
mPaint.setAntiAlias(true);
mPaint.setColor(Color.GRAY);
mPaint.setStyle(Paint.Style.STROKE);
mPaint.setStrokeWidth(getResources().getDimension(R.dimen.dp_1));
mPaint.setStrokeCap(Paint.Cap.ROUND);
mPaint.setStrokeJoin(Paint.Join.ROUND);
mBitmapPaint = new Paint();
mBitmapPaint.setDither(true);
mBitmapPaint.setAntiAlias(true);
mTextPaint = new Paint();
mTextPaint.setDither(true);
mTextPaint.setAntiAlias(true);
mTextPaint.setColor(Color.GRAY);
mTextPaint.setTextSize(getResources().getDimension(R.dimen.sp_12));
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
mWidth = MeasureSpec.getSize(widthMeasureSpec);
mHeigth = MeasureSpec.getSize(heightMeasureSpec);
setMeasuredDimension(mWidth, mHeigth);
@Override
protected void onDraw(Canvas canvas) {
int x = mWidth / 2;
int y = mHeigth / 2;
int r = (int) (mWidth * 0.42);
canvas.save();
for (int i = 0; i < 120; i++) {
if(i==0){
mPaint.setColor(Color.RED);
drawText("北",canvas);
canvas.drawLine(x, y - r, x, (float) (y - r + x * 0.1), mPaint);
}else if(i==30){
mPaint.setColor(Color.RED);
drawText("东",canvas);
canvas.drawLine(x, y - r, x, (float) (y - r + x * 0.1), mPaint);
}else if(i==60){
mPaint.setColor(Color.RED);
drawText("南",canvas);
canvas.drawLine(x, y - r, x, (float) (y - r + x * 0.1), mPaint);
}else if(i==90){
mPaint.setColor(Color.RED);
drawText("西",canvas);
canvas.drawLine(x, y - r, x, (float) (y - r + x * 0.1), mPaint);
}else if(i==5||i==10||i==20||i==25||i==15||i==35||i==40||i==45||i==50||i==55||i==65||i==70||i==75||i==80||i==85||i==95||i==100||i==105||i==110||i==115){
mPaint.setColor(Color.BLACK);
drawText(i*3+"",canvas);
canvas.drawLine(x, y - r, x, (float) (y - r + x * 0.1), mPaint);
}else{
//绘制下层菊花
mPaint.setColor(Color.GRAY);
canvas.drawLine(x, y - r, x, (float) (y - r + x * 0.1), mPaint);
canvas.rotate(360 / 120, x, y);
for (int i = 0; i < 360; i++) {
if(mCureentAngla==i){
canvas.drawLine(x, y - r, x, (float) (y - r + x * 0.2), mPaint);
canvas.rotate(360 / 360, x, y);
canvas.restore();
public void drawText(String string,Canvas canvas){
//画中间文字
String text = string;
Rect textRect=new Rect();
mTextPaint.getTextBounds(text,0,text.length(),textRect);
int startX=getWidth()/2-textRect.width()/2;
int textHeight=textRect.height();
int baseLine=0;
Paint.FontMetricsInt fontMetricsInt = mTextPaint.getFontMetricsInt();
int dy=(fontMetricsInt.bottom-fontMetricsInt.top)/2-fontMetricsInt.bottom;
baseLine= textHeight/2+dy;
canvas.drawText(text,startX,baseLine,mTextPaint);
@Override
public void onSensorChanged(SensorEvent sensorEvent) {
switch (sensorEvent.sensor.getType()) {
case Sensor.TYPE_ORIENTATION:
//顺时针转动为正,故手机顺时针转动时,图片得逆时针转动
toDegrees = (int) -sensorEvent.values[0];
if( this.onAnglaChanged !=null){
onAnglaChanged.onChanged(toDegrees);
break;
@Override
public void onAccuracyChanged(Sensor sensor, int i) {
@Override
protected void onDetachedFromWindow() {
super.onDetachedFromWindow();
if(sensorManager!=null){
sensorManager.unregisterListener(this);
public interface OnAnglaChanged{
void onChanged(int angla);
public OnAnglaChanged onAnglaChanged;
public void setOnAnglaChanged(OnAnglaChanged onAnglaChanged){
this.onAnglaChanged=onAnglaChanged;
可以看到代码里面我留了setCureentAngla这个接口,这就是设置当前方位角的,我把求方位角的代码留在了activity中这不算是很好的封装(吃瓜群众:偷懒就是偷懒,还这么多借口)。
好吧,我承认是有那么一丢丢懒,但是我不改,哈哈哈。
下面是activity中的使用代码:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="com.easy.customeasytablayout.customviews.zhinanzhen.Main19Activity">
<TextView
android:id="@+id/tv_angla"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="@dimen/dp_80"
android:textSize="@dimen/sp_24"
android:textColor="@android:color/black"
android:layout_centerHorizontal="true"/>
<RelativeLayout
android:layout_width="@dimen/dp_240"
android:layout_height="@dimen/dp_240"
android:layout_centerInParent="true">
<com.easy.customeasytablayout.customviews.zhinanzhen.DemoZhiNaZhen
android:id="@+id/test"
android:layout_width="@dimen/dp_240"
android:layout_height="@dimen/dp_240"
android:layout_centerInParent="true"/>
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@mipmap/iv_zhishi"
android:layout_centerHorizontal="true"
android:layout_marginTop="@dimen/dp_22"/>
</RelativeLayout>
</RelativeLayout>
public class Main19Activity extends AppCompatActivity {
private DemoZhiNaZhen test;
private int fromDegrees = 0;
private TextView mTvAngla;
//31.2036,121.6012
//31.1237,121.3536
private PointF start = new PointF(31.1237f, 121.3536f);//起点
private PointF end = new PointF(31.2036f, 121.6012f);//起点
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main19);
test = (DemoZhiNaZhen) findViewById(R.id.test);
mTvAngla = (TextView) findViewById(R.id.tv_angla);
test.setCureentAngla((int) gps2d(31.1237f, 121.3536f, 31.2036f, 121.6012f));
test.setOnAnglaChanged(new DemoZhiNaZhen.OnAnglaChanged() {
@Override
public void onChanged(int toDegrees) {
mTvAngla.setText("" + Math.abs(toDegrees) + " 度");
//让图片相对自身中心点转动,开始角度默认为0;此后开始角度等于上一次结束角度
RotateAnimation ra = new RotateAnimation(fromDegrees,
toDegrees, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
ra.setInterpolator(new LinearInterpolator());
//动画时间200毫秒
ra.setDuration(Math.abs(fromDegrees-toDegrees));
ra.setFillAfter(true);
test.startAnimation(ra);
fromDegrees = toDegrees;
private float gps2d(float lat_a, float lng_a, float lat_b, float lng_b) {
float d = 0;
lat_a = (float) (lat_a * Math.PI / 180);
lng_a = (float) (lng_a * Math.PI / 180);
lat_b = (float) (lat_b * Math.PI / 180);
lng_b = (float) (lng_b * Math.PI / 180);
d = (float) (Math.sin(lat_a) * Math.sin(lat_b) +
Math.cos(lat_a) * Math.cos(lat_b) * Math.cos(lng_b - lng_a));
d = (float) Math.sqrt(1 - d * d);
d = (float) (Math.cos(lat_b) * Math.sin(lng_b - lng_a) / d);
d = (float) (Math.asin(d) * 180 / Math.PI);
System.out.println("yanjin-------方位角--" + d);
return d;