1 /* 2 * Copyright (C) 2013 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.cts.verifier.sensors; 18 19 import android.content.Context; 20 import android.graphics.Canvas; 21 import android.graphics.Color; 22 import android.graphics.Paint; 23 import android.graphics.PorterDuff; 24 import android.graphics.PorterDuffXfermode; 25 import android.graphics.RectF; 26 import android.hardware.SensorManager; 27 import android.util.AttributeSet; 28 import android.util.Log; 29 import android.view.Surface; 30 import android.view.View; 31 32 /** 33 * A view class that draws the user prompt 34 * 35 * The following piece of code should show how to use this view. 36 * 37 * public void testUI() { 38 * final int MAX_TILT_ANGLE = 70; // +/- 70 39 * 40 * final int TILT_ANGLE_STEP = 5; // 5 degree(s) per step 41 * final int YAW_ANGLE_STEP = 10; // 10 degree(s) per step 42 * 43 * RangeCoveredRegister xCovered, yCovered, zCovered; 44 * xCovered = new RangeCoveredRegister(-MAX_TILT_ANGLE, +MAX_TILT_ANGLE, TILT_ANGLE_STEP); 45 * 46 * yCovered = new RangeCoveredRegister(-MAX_TILT_ANGLE, +MAX_TILT_ANGLE, TILT_ANGLE_STEP); 47 * zCovered = new RangeCoveredRegister(YAW_ANGLE_STEP); 48 * 49 * xCovered.update(40); 50 * xCovered.update(-40); 51 * xCovered.update(12); 52 * 53 * yCovered.update(50); 54 * yCovered.update(-51); 55 * 56 * zCovered.update(150); 57 * zCovered.update(42); 58 * 59 * setDataProvider(xCovered, yCovered, zCovered); 60 * enableAxis(RVCVRecordActivity.AXIS_ALL); //debug mode, show all three axis 61 * } 62 */ 63 public class MotionIndicatorView extends View { 64 private final String TAG = "MotionIndicatorView"; 65 private final boolean LOCAL_LOGV = false; 66 67 private Paint mCursorPaint; 68 private Paint mLimitPaint; 69 private Paint mCoveredPaint; 70 private Paint mRangePaint; 71 private Paint mEraserPaint; 72 73 // UI settings 74 private final int XBAR_WIDTH = 50; 75 private final int XBAR_MARGIN = 50; 76 private final int XBAR_CURSOR_ADD = 20; 77 78 private final int YBAR_WIDTH = 50; 79 private final int YBAR_MARGIN = 50; 80 private final int YBAR_CURSOR_ADD = 20; 81 82 private final int ZRING_WIDTH = 50; 83 private final int ZRING_CURSOR_ADD = 30; 84 85 86 private int mXSize, mYSize; 87 private RectF mZBoundOut, mZBoundOut2, mZBoundIn, mZBoundIn2; 88 89 private RangeCoveredRegister mXCovered, mYCovered, mZCovered; 90 91 private boolean mXEnabled, mYEnabled, mZEnabled; 92 93 private boolean mIsDeviceRotated = false; 94 95 /** 96 * Constructor 97 * @param context 98 */ MotionIndicatorView(Context context)99 public MotionIndicatorView(Context context) { 100 super(context); 101 init(); 102 } 103 104 /** 105 * Constructor 106 * @param context Application context 107 * @param attrs 108 */ MotionIndicatorView(Context context, AttributeSet attrs)109 public MotionIndicatorView(Context context, AttributeSet attrs) { 110 super(context, attrs); 111 init(); 112 } 113 114 /** 115 * Initialize the Paint objects 116 */ init()117 private void init() { 118 119 mCursorPaint = new Paint(); 120 mCursorPaint.setColor(Color.BLUE); 121 122 mLimitPaint = new Paint(); 123 mLimitPaint.setColor(Color.YELLOW); 124 125 mCoveredPaint = new Paint(); 126 mCoveredPaint.setColor(Color.CYAN); 127 128 mRangePaint = new Paint(); 129 mRangePaint.setColor(Color.DKGRAY); 130 131 mEraserPaint = new Paint(); 132 mEraserPaint.setColor(Color.TRANSPARENT); 133 // ensure the erasing effect 134 mEraserPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC)); 135 } 136 137 /** 138 * Connect the view to certain data provider objects 139 * @param x Data provider for x direction tilt angle 140 * @param y Data provider for y direction tilt angle 141 * @param z Data provider for z rotation 142 */ setDataProvider(RangeCoveredRegister x, RangeCoveredRegister y, RangeCoveredRegister z)143 public void setDataProvider(RangeCoveredRegister x, 144 RangeCoveredRegister y, 145 RangeCoveredRegister z) { 146 mXCovered = x; 147 mYCovered = y; 148 mZCovered = z; 149 } 150 151 /** 152 * Set the device's current rotation 153 * @param rotation Surface.ROTATION_0, Surface.ROTATION_90, Surface.ROTATION_180, or 154 * Surface.ROTATION_270 155 */ setDeviceRotation(int rotation)156 public void setDeviceRotation(int rotation) { 157 mIsDeviceRotated = (rotation == Surface.ROTATION_90 || rotation == Surface.ROTATION_270); 158 } 159 160 /** 161 * Set the active axis for display 162 * 163 * @param axis AXIS_X, AXIS_Y, AXIS_Z for x, y, z axis indicators, or AXIS_ALL for all three. 164 */ enableAxis(int axis)165 public void enableAxis(int axis) { 166 mXEnabled = mYEnabled = mZEnabled = false; 167 168 switch(axis) 169 { 170 case SensorManager.AXIS_X: 171 mXEnabled = true; 172 break; 173 case SensorManager.AXIS_Y: 174 mYEnabled = true; 175 break; 176 case SensorManager.AXIS_Z: 177 mZEnabled = true; 178 break; 179 case RVCVRecordActivity.AXIS_ALL: 180 mXEnabled = mYEnabled = mZEnabled = true; 181 } 182 } 183 184 /** 185 * Doing some pre-calculation that only changes when view dimensions are changed. 186 * @param w 187 * @param h 188 * @param oldw 189 * @param oldh 190 */ 191 @Override onSizeChanged(int w, int h, int oldw, int oldh)192 protected void onSizeChanged (int w, int h, int oldw, int oldh) { 193 mXSize = w; 194 mYSize = h; 195 196 float halfSideLength = 0.4f * Math.min(w, h); 197 float leftSide = w/2 - halfSideLength; 198 float topSide = h/2 - halfSideLength; 199 float rightSide = w/2 + halfSideLength; 200 float bottomSide = h/2 + halfSideLength; 201 202 mZBoundOut = new RectF(leftSide, topSide, rightSide, bottomSide); 203 mZBoundOut2 = new RectF( 204 leftSide-ZRING_CURSOR_ADD, topSide-ZRING_CURSOR_ADD, 205 rightSide+ZRING_CURSOR_ADD, bottomSide+ZRING_CURSOR_ADD); 206 mZBoundIn = new RectF( 207 leftSide+ZRING_WIDTH, topSide+ZRING_WIDTH, 208 rightSide-ZRING_WIDTH, bottomSide-ZRING_WIDTH); 209 mZBoundIn2 = new RectF( 210 leftSide+ZRING_WIDTH+ZRING_CURSOR_ADD, topSide+ZRING_WIDTH+ZRING_CURSOR_ADD, 211 rightSide-ZRING_WIDTH-ZRING_CURSOR_ADD, bottomSide-ZRING_WIDTH-ZRING_CURSOR_ADD); 212 213 if (LOCAL_LOGV) Log.v(TAG, "New view size = ("+w+", "+h+")"); 214 } 215 216 /** 217 * Draw UI depends on the selected axis and registered value 218 * 219 * @param canvas the canvas to draw on 220 */ 221 @Override onDraw(Canvas canvas)222 protected void onDraw(Canvas canvas) { 223 super.onDraw(canvas); 224 int i,t; 225 226 Paint p = new Paint(); 227 p.setColor(Color.YELLOW); 228 canvas.drawRect(10,10, 50, 50, p); 229 230 // In order to determine which progress bar to draw, the device's rotation must be accounted 231 // for since the accelerometer rotates with the display. 232 boolean drawX = (mXEnabled && !mIsDeviceRotated) || (mYEnabled && mIsDeviceRotated); 233 boolean drawY = (mYEnabled && !mIsDeviceRotated) || (mXEnabled && mIsDeviceRotated); 234 235 if (drawX && mXCovered != null) { 236 RangeCoveredRegister covered = mIsDeviceRotated ? mYCovered : mXCovered; 237 int xNStep = covered.getNSteps() + 4; // two on each side as a buffer 238 int xStepSize = mXSize * 3/4 / xNStep; 239 int xLeft = mXSize * 1/8 + (mXSize * 3/4 % xNStep)/2; 240 241 // base bar 242 canvas.drawRect(xLeft, XBAR_MARGIN, 243 xLeft+xStepSize*xNStep-1, XBAR_WIDTH+XBAR_MARGIN, mRangePaint); 244 245 // covered range 246 for (i=0; i<covered.getNSteps(); ++i) { 247 if (covered.isCovered(i)) { 248 canvas.drawRect( 249 xLeft+xStepSize*(i+2), XBAR_MARGIN, 250 xLeft+xStepSize*(i+3)-1, XBAR_WIDTH + XBAR_MARGIN, 251 mCoveredPaint); 252 } 253 } 254 255 // limit 256 canvas.drawRect(xLeft+xStepSize*2-4, XBAR_MARGIN, 257 xLeft+xStepSize*2+3, XBAR_WIDTH+XBAR_MARGIN, mLimitPaint); 258 canvas.drawRect(xLeft+xStepSize*(xNStep-2)-4, XBAR_MARGIN, 259 xLeft+xStepSize*(xNStep-2)+3, XBAR_WIDTH+XBAR_MARGIN, mLimitPaint); 260 261 // cursor 262 t = (int)(xLeft+xStepSize*(covered.getLastValue()+2)); 263 canvas.drawRect(t-4, XBAR_MARGIN-XBAR_CURSOR_ADD, t+3, 264 XBAR_WIDTH+XBAR_MARGIN+XBAR_CURSOR_ADD, mCursorPaint); 265 } 266 267 if (drawY && mYCovered != null) { 268 RangeCoveredRegister covered = mIsDeviceRotated ? mXCovered : mYCovered; 269 int yNStep = covered.getNSteps() + 4; // two on each side as a buffer 270 int yStepSize = mYSize * 3/4 / yNStep; 271 int yLeft = mYSize * 1/8 + (mYSize * 3/4 % yNStep)/2; 272 273 // base bar 274 canvas.drawRect(YBAR_MARGIN, yLeft, 275 YBAR_WIDTH+YBAR_MARGIN, yLeft+yStepSize*yNStep-1, mRangePaint); 276 277 // covered range 278 for (i=0; i<covered.getNSteps(); ++i) { 279 if (covered.isCovered(i)) { 280 canvas.drawRect( 281 YBAR_MARGIN, yLeft+yStepSize*(i+2), 282 YBAR_WIDTH + YBAR_MARGIN, yLeft+yStepSize*(i+3)-1, 283 mCoveredPaint); 284 } 285 } 286 287 // limit 288 canvas.drawRect(YBAR_MARGIN, yLeft + yStepSize * 2 - 4, 289 YBAR_WIDTH + YBAR_MARGIN, yLeft + yStepSize * 2 + 3, mLimitPaint); 290 canvas.drawRect(YBAR_MARGIN, yLeft + yStepSize * (yNStep - 2) - 4, 291 YBAR_WIDTH + YBAR_MARGIN, yLeft + yStepSize * (yNStep - 2) + 3, mLimitPaint); 292 293 // cursor 294 t = (int)(yLeft+yStepSize*(covered.getLastValue()+2)); 295 canvas.drawRect( YBAR_MARGIN-YBAR_CURSOR_ADD, t-4, 296 YBAR_WIDTH+YBAR_MARGIN+YBAR_CURSOR_ADD, t+3, mCursorPaint); 297 } 298 299 if (mZEnabled && mZCovered != null) { 300 float stepSize = 360.0f/mZCovered.getNSteps(); 301 302 // base bar 303 canvas.drawArc(mZBoundOut,0, 360, true, mRangePaint); 304 305 // covered range 306 for (i=0; i<mZCovered.getNSteps(); ++i) { 307 if (mZCovered.isCovered(i)) { 308 canvas.drawArc(mZBoundOut,i*stepSize-0.2f, stepSize+0.4f, 309 true, mCoveredPaint); 310 } 311 } 312 // clear center 313 canvas.drawArc(mZBoundIn, 0, 360, true, mEraserPaint); 314 // cursor 315 canvas.drawArc(mZBoundOut2, mZCovered.getLastValue()*stepSize- 1, 2, 316 true, mCursorPaint); 317 canvas.drawArc(mZBoundIn2, mZCovered.getLastValue()*stepSize-1.5f, 3, 318 true, mEraserPaint); 319 } 320 } 321 } 322 323 /** 324 * A range register class for the RVCVRecord Activity 325 */ 326 class RangeCoveredRegister { 327 enum MODE { 328 LINEAR, 329 ROTATE2D 330 } 331 332 private boolean[] mCovered; 333 private MODE mMode; 334 private int mStep; 335 private int mLow, mHigh; 336 private int mLastData; 337 338 // high is not inclusive RangeCoveredRegister(int low, int high, int step)339 RangeCoveredRegister(int low, int high, int step) { 340 mMode = MODE.LINEAR; 341 mStep = step; 342 mLow = low; 343 mHigh = high; 344 init(); 345 } 346 RangeCoveredRegister(int step)347 RangeCoveredRegister(int step) { 348 mMode = MODE.ROTATE2D; 349 mStep = step; 350 mLow = 0; 351 mHigh = 360; 352 init(); 353 } 354 init()355 private void init() { 356 if (mMode == MODE.LINEAR) { 357 mCovered = new boolean[(mHigh-mLow)/mStep]; 358 }else { 359 mCovered = new boolean[360/mStep]; 360 } 361 } 362 363 /** 364 * Test if the range specified by (low, high) is covered. 365 * 366 * If it is LINEAR mode, the range will be quantized to nearest step boundary. If it is the 367 * ROTATE2D mode, it is the same as isFullyCovered(). 368 * 369 * @param low The low end of the range. 370 * @param high The high end of the range. 371 * @return if the specified range is covered, return true; otherwise false. 372 */ isRangeCovered(int low, int high)373 public boolean isRangeCovered(int low, int high) { 374 if (mMode == MODE.LINEAR) { 375 int iLow = Math.max((low - mLow) / mStep, 0); 376 int iHigh = Math.min((high - mLow) / mStep, mCovered.length - 1); 377 378 for (int i = iLow; i <= iHigh; ++i) { 379 if (!mCovered[i]) { 380 return false; 381 } 382 } 383 return true; 384 385 } else { 386 return isFullyCovered(); 387 } 388 } 389 390 /** 391 * Test if the range defined is fully covered. 392 * 393 * @return if the range is fully covered, return true; otherwise false. 394 */ isFullyCovered()395 public boolean isFullyCovered() { 396 for (boolean i : mCovered) { 397 if (!i) return false; 398 } 399 return true; 400 } 401 402 /** 403 * Test if a specific step is covered. 404 * 405 * @param i the step number 406 * @return if the step specified is covered, return true; otherwise false. 407 */ isCovered(int i)408 public boolean isCovered(int i) { 409 return mCovered[i]; 410 } 411 412 /** 413 * 414 * 415 * @param data 416 * @return if this update changes the status of 417 */ update(int data)418 public boolean update(int data) { 419 mLastData = data; 420 421 if (mMode == MODE.ROTATE2D) { 422 data %= 360; 423 } 424 425 int iStep = (data - mLow)/mStep; 426 427 if (iStep>=0 && iStep<getNSteps()) { 428 // only record valid data 429 mLastData = data; 430 431 if (mCovered[iStep]) { 432 return false; 433 } else { 434 mCovered[iStep] = true; 435 return true; 436 } 437 } 438 return false; 439 } 440 441 /** 442 * Get the number of steps in this register 443 * 444 * @return The number of steps in this register 445 */ getNSteps()446 public int getNSteps() { 447 //if (mCovered == null) { 448 //return 0; 449 //} 450 return mCovered.length; 451 } 452 453 /** 454 * Get the last value updated 455 * 456 * @return The last value updated 457 */ getLastValue()458 public float getLastValue() { 459 // ensure float division 460 return ((float)(mLastData - mLow))/mStep; 461 } 462 } 463