在前面的一些关于Android的博文中,涉及到一些自定义控件的文章。那么,在这篇博文中,我将向大家介绍以自定义控件的方式来实现罗盘视图的效果。我们将会通过扩展View类来创建一个新的罗盘视图。它通过显示传统的罗盘来指示当前朝向的方向。
这个类扩展自View类,然后添加允许在代码中对视图进行实例化或者从资源布局填充它的构造函数。之后,添加一个新的initCompassView方法,用来初始化控件,并在每个构造函数中调用它。
具体结构代码如下:
package com.lyz.compass.view; import android.content.Context; import android.util.AttributeSet; import android.view.View; /** * 自定义View类 * @author liuyazhuang * */ public class CompassView extends View { public CompassView(Context context){ super(context); initCompassView(); } public CompassView(Context context, AttributeSet attrs){ super(context, attrs); initCompassView(); } public CompassView(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); initCompassView(); } /** * 初始化视图的各个属性 */ protected void initCompassView(){ setFocusable(true); } }
罗盘视图应该是一个正圆,而且应该占据画布允许的尽可能大的空间。因此,可以通过重写onMeasure方法来计算最短边的长度,然后通过这个值并通过setMeasuredDimension方法来设置高度和宽度。
具体代码如下:
@Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { //罗盘是一个尽可能填充更多的空间的圆,通过设置最短的边界,高度或者宽度来设置测量的尺寸 int measureWidth = measure(widthMeasureSpec); int measureHeight = measure(heightMeasureSpec); int d = Math.min(measureWidth, measureHeight); setMeasuredDimension(d, d); }
/** * 解码数据值 * @param measureSpec * @return */ private int measure(int measureSpec) { int result = 0; //对测量说明进行解码 int specMode = MeasureSpec.getMode(measureSpec); int specSize = MeasureSpec.getSize(measureSpec); //如果没有指定界限,则返回默认大小200 if(specMode == MeasureSpec.UNSPECIFIED){ result = 200; }else{ //由于是希望填充可用的空间,所以总是返回整个可用的边界 result = specSize; } return result; }
用新的CompassView替换TextView
具体代码如下:
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"> <!-- 引用自定义的View --> <com.lyz.compass.view.CompassView android:id="@+id/compassView" android:layout_width="match_parent" android:layout_height="match_parent"/> </FrameLayout>
<?xml version="1.0" encoding="utf-8"?> <resources> <string name="app_name">Compass</string> <string name="action_settings">Settings</string> <string name="hello_world">Hello world!</string> <string name="cardinal_north">N</string> <string name="cardinal_east">E</string> <string name="cardinal_south">S</string> <string name="cardinal_west">W</string> </resources>
<?xml version="1.0" encoding="utf-8"?> <resources> <color name="background_color">#555</color> <color name="marker_color">#AFFF</color> <color name="text_color">#AFFF</color> </resources>
在CompassView类中,为显示的方向添加一个新的属性,并创建它的set和get方法。
//显示的方向 private float bearing; public float getBearing() { return bearing; } public void setBearing(float bearing) { this.bearing = bearing; }
返回到CompassView类,引用4中创建的每一个资源,把字符串值存储为实例变量,并使用颜色值来创建新的类作用域的Paint对象。在下一步中将使用这些对象来绘制罗盘字盘。
具体代码如下:
//显示的方向 private float bearing; private Paint markerPaint; private Paint textPaint; private Paint circlePaint; private String northString; private String eastString; private String southString; private String westString; private int textHeight; /** * 初始化视图的各个属性 */ protected void initCompassView(){ setFocusable(true); Resources r = this.getResources(); circlePaint = new Paint(Paint.ANTI_ALIAS_FLAG); circlePaint.setColor(r.getColor(R.color.background_color)); circlePaint.setStrokeWidth(1); circlePaint.setStyle(Paint.Style.FILL_AND_STROKE); northString = r.getString(R.string.cardinal_north); eastString = r.getString(R.string.cardinal_east); southString = r.getString(R.string.cardinal_south); westString = r.getString(R.string.cardinal_west); textPaint = new Paint(Paint.ANTI_ALIAS_FLAG); textPaint.setColor(r.getColor(R.color.text_color)); textHeight = (int) textPaint.measureText("yY"); markerPaint = new Paint(Paint.ANTI_ALIAS_FLAG); markerPaint.setColor(r.getColor(R.color.marker_color)); }
使用6中创建的String和Paint来绘制罗盘的字盘。
具体代码如下:
@Override protected void onDraw(Canvas canvas) { // TODO Auto-generated method stub // super.onDraw(canvas); int measuredWidth = getMeasuredWidth(); int measuredHeight = getMeasuredHeight(); int px = measuredWidth / 2; int py = measuredHeight / 2; //取较小的值为半径 int radius = Math.min(px, py); //绘制背景 canvas.drawCircle(px, py, radius, circlePaint); canvas.save(); canvas.rotate(-bearing, px, py); int textWidth = (int) textPaint.measureText("W"); int cadinalX = px - textWidth / 2; int cadinalY = py - radius + textHeight; //每15度绘制一个标记,每45度绘制一个文本 for(int i = 0; i < 24; i++){ //绘制一个标记 canvas.drawLine(px, px - radius, py, py - radius + 10, markerPaint); canvas.save(); canvas.translate(0, textHeight); //绘制基本方位 if(i % 6 == 0){ String dirString = ""; switch (i) { case 0: dirString = northString; int arrowY = 2 * textHeight; canvas.drawLine(px , arrowY, px - 5, 3 * textHeight, markerPaint); canvas.drawLine(px, arrowY, px + 5, 3 * textHeight, markerPaint); break; case 6: dirString = eastString; break; case 12: dirString = southString; break; case 18: dirString = westString; break; default: break; } canvas.drawText(dirString, cadinalX, cadinalY, textPaint); }else if(i % 3 == 0){ //每45度绘制文本 String angle = String.valueOf(i * 15); float angleTextWidth = textPaint.measureText(angle); int angleTextX = (int)(px - angleTextWidth/2); int angleTextY = py - radius + textHeight; canvas.drawText(angle, angleTextX, angleTextY, textPaint); } canvas.restore(); canvas.rotate(15, px, py); } canvas.restore(); }
罗盘视图以可视方式显示方向,所以为了提高可访问性,当方向变化时,需要广播一个可访问性事件,说明“文本(在本例中是内容)”发生了变化。为了,需要修改setBearing方法。
具体代码如下:
public void setBearing(float bearing) { this.bearing = bearing; sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_TEXT_CHANGED); }
将当前方向用作可访问性事件使用的内容值。
具体代码如下:
@Override public boolean dispatchPopulateAccessibilityEvent(AccessibilityEvent event) { // TODO Auto-generated method stub super.dispatchPopulateAccessibilityEvent(event); if(isShown()){ String bearingStr = String.valueOf(bearing); if(bearingStr.length() > AccessibilityEvent.MAX_TEXT_LENGTH){ bearingStr = bearingStr.substring(0, AccessibilityEvent.MAX_TEXT_LENGTH); event.getText().add(bearingStr); return true; } return false; } return false; }
package com.lyz.compass.view; import com.lyz.compass.activity.R; import android.content.Context; import android.content.res.Resources; import android.graphics.Canvas; import android.graphics.Paint; import android.util.AttributeSet; import android.view.View; import android.view.accessibility.AccessibilityEvent; /** * 自定义View类 * @author liuyazhuang * */ public class CompassView extends View { //显示的方向 private float bearing; private Paint markerPaint; private Paint textPaint; private Paint circlePaint; private String northString; private String eastString; private String southString; private String westString; private int textHeight; public CompassView(Context context){ super(context); initCompassView(); } public CompassView(Context context, AttributeSet attrs){ super(context, attrs); initCompassView(); } public CompassView(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); initCompassView(); } /** * 初始化视图的各个属性 */ protected void initCompassView(){ setFocusable(true); Resources r = this.getResources(); circlePaint = new Paint(Paint.ANTI_ALIAS_FLAG); circlePaint.setColor(r.getColor(R.color.background_color)); circlePaint.setStrokeWidth(1); circlePaint.setStyle(Paint.Style.FILL_AND_STROKE); northString = r.getString(R.string.cardinal_north); eastString = r.getString(R.string.cardinal_east); southString = r.getString(R.string.cardinal_south); westString = r.getString(R.string.cardinal_west); textPaint = new Paint(Paint.ANTI_ALIAS_FLAG); textPaint.setColor(r.getColor(R.color.text_color)); textHeight = (int) textPaint.measureText("yY"); markerPaint = new Paint(Paint.ANTI_ALIAS_FLAG); markerPaint.setColor(r.getColor(R.color.marker_color)); } public float getBearing() { return bearing; } public void setBearing(float bearing) { this.bearing = bearing; sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_TEXT_CHANGED); } @Override public boolean dispatchPopulateAccessibilityEvent(AccessibilityEvent event) { // TODO Auto-generated method stub super.dispatchPopulateAccessibilityEvent(event); if(isShown()){ String bearingStr = String.valueOf(bearing); if(bearingStr.length() > AccessibilityEvent.MAX_TEXT_LENGTH){ bearingStr = bearingStr.substring(0, AccessibilityEvent.MAX_TEXT_LENGTH); event.getText().add(bearingStr); return true; } return false; } return false; } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { //罗盘是一个尽可能填充更多的空间的圆,通过设置最短的边界,高度或者宽度来设置测量的尺寸 int measureWidth = measure(widthMeasureSpec); int measureHeight = measure(heightMeasureSpec); int d = Math.min(measureWidth, measureHeight); setMeasuredDimension(d, d); } /** * 解码数据值 * @param measureSpec * @return */ private int measure(int measureSpec) { int result = 0; //对测量说明进行解码 int specMode = MeasureSpec.getMode(measureSpec); int specSize = MeasureSpec.getSize(measureSpec); //如果没有指定界限,则返回默认大小200 if(specMode == MeasureSpec.UNSPECIFIED){ result = 200; }else{ //由于是希望填充可用的空间,所以总是返回整个可用的边界 result = specSize; } return result; } @Override protected void onDraw(Canvas canvas) { // TODO Auto-generated method stub // super.onDraw(canvas); int measuredWidth = getMeasuredWidth(); int measuredHeight = getMeasuredHeight(); int px = measuredWidth / 2; int py = measuredHeight / 2; //取较小的值为半径 int radius = Math.min(px, py); //绘制背景 canvas.drawCircle(px, py, radius, circlePaint); canvas.save(); canvas.rotate(-bearing, px, py); int textWidth = (int) textPaint.measureText("W"); int cadinalX = px - textWidth / 2; int cadinalY = py - radius + textHeight; //每15度绘制一个标记,每45度绘制一个文本 for(int i = 0; i < 24; i++){ //绘制一个标记 canvas.drawLine(px, px - radius, py, py - radius + 10, markerPaint); canvas.save(); canvas.translate(0, textHeight); //绘制基本方位 if(i % 6 == 0){ String dirString = ""; switch (i) { case 0: dirString = northString; int arrowY = 2 * textHeight; canvas.drawLine(px , arrowY, px - 5, 3 * textHeight, markerPaint); canvas.drawLine(px, arrowY, px + 5, 3 * textHeight, markerPaint); break; case 6: dirString = eastString; break; case 12: dirString = southString; break; case 18: dirString = westString; break; default: break; } canvas.drawText(dirString, cadinalX, cadinalY, textPaint); }else if(i % 3 == 0){ //每45度绘制文本 String angle = String.valueOf(i * 15); float angleTextWidth = textPaint.measureText(angle); int angleTextX = (int)(px - angleTextWidth/2); int angleTextY = py - radius + textHeight; canvas.drawText(angle, angleTextX, angleTextY, textPaint); } canvas.restore(); canvas.rotate(15, px, py); } canvas.restore(); } }
在MainActivity中有两种方式引用自定义控件,一种是引用布局文件,一种是创建自定义控件对象,将自定义对象设置给当前视图。
package com.lyz.compass.activity; import android.app.Activity; import android.os.Bundle; import com.lyz.compass.view.CompassView; /** * 程序的主入口 * @author liuyazhuang * */ public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); CompassView compassView = (CompassView) this.findViewById(R.id.compassView); compassView.setBearing(0); } }
package com.lyz.compass.activity; import android.app.Activity; import android.os.Bundle; import com.lyz.compass.view.CompassView; /** * 程序的主入口 * @author liuyazhuang * */ public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); CompassView compassView = new CompassView(this); setContentView(compassView); compassView.setBearing(0); } }
版权声明:本文为博主原创文章,未经博主允许不得转载。
原文地址:http://blog.csdn.net/l1028386804/article/details/47337949