summaryrefslogtreecommitdiffstats
path: root/src/com
diff options
context:
space:
mode:
Diffstat (limited to 'src/com')
-rw-r--r--src/com/android/camera/panorama/CaptureView.java70
-rw-r--r--src/com/android/camera/panorama/IndicationView.java120
-rw-r--r--src/com/android/camera/panorama/MosaicFrameProcessor.java41
-rw-r--r--src/com/android/camera/panorama/PanoramaActivity.java134
4 files changed, 221 insertions, 144 deletions
diff --git a/src/com/android/camera/panorama/CaptureView.java b/src/com/android/camera/panorama/CaptureView.java
deleted file mode 100644
index 3b1dd29..0000000
--- a/src/com/android/camera/panorama/CaptureView.java
+++ /dev/null
@@ -1,70 +0,0 @@
-/*
- * Copyright (C) 2011 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.camera.panorama;
-
-import android.content.Context;
-import android.graphics.Canvas;
-import android.graphics.Color;
-import android.graphics.Paint;
-import android.graphics.Paint.Align;
-import android.graphics.RectF;
-import android.graphics.Typeface;
-import android.util.AttributeSet;
-import android.util.Log;
-import android.view.View;
-
-class CaptureView extends View {
- private static final String TAG = "CaptureView";
- private int mStartAngle = 0;
- private int mSweepAngle = 0;
- private int mWidth;
- private int mHeight;
- private final Paint mPaint = new Paint();
-
- public CaptureView(Context context, AttributeSet attrs) {
- super(context, attrs);
- mPaint.setStyle(Paint.Style.FILL);
- mPaint.setColor(Color.RED);
- mPaint.setAntiAlias(true);
- mPaint.setTextSize(40);
- mPaint.setTypeface(Typeface.create((Typeface) null, Typeface.BOLD));
- mPaint.setTextAlign(Align.CENTER);
- }
-
- protected void onSizeChanged(int w, int h, int oldw, int oldh) {
- Log.v(TAG, "onSizeChanged: W = " + w + ", H = " + h);
- }
-
- public void setStartAngle(int angle) {
- mStartAngle = angle;
- }
-
- public void setSweepAngle(int angle) {
- mSweepAngle = angle;
- }
-
- @Override
- protected void onDraw(Canvas canvas) {
- mWidth = getWidth();
- mHeight = getHeight();
-
- RectF rect = new RectF(mWidth / 2 - 100, 3 * mHeight / 4,
- mWidth / 2 + 100, 3 * mHeight / 4 + 200);
- canvas.drawArc(rect, -90 + mStartAngle, mSweepAngle, true, mPaint);
- canvas.drawArc(rect, -90 - mStartAngle, mSweepAngle > 0 ? 2 : 0, true, mPaint);
- }
-}
diff --git a/src/com/android/camera/panorama/IndicationView.java b/src/com/android/camera/panorama/IndicationView.java
new file mode 100644
index 0000000..d0eaf9e
--- /dev/null
+++ b/src/com/android/camera/panorama/IndicationView.java
@@ -0,0 +1,120 @@
+/*
+ * Copyright (C) 2011 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package com.android.camera.panorama;
+
+import android.content.Context;
+import android.graphics.Canvas;
+import android.graphics.Color;
+import android.graphics.Paint;
+import android.graphics.RectF;
+import android.util.AttributeSet;
+import android.view.View;
+
+class IndicationView extends View {
+ private static final String TAG = "IndicationView";
+ private static final int PAN_DIRECTION_NONE = 0;
+ private static final int PAN_DIRECTION_LEFT = 1;
+ private static final int PAN_DIRECTION_RIGHT = 2;
+ private int mSweepAngle = 0;
+ private int mStartAngle = 0;
+ private int mMaxSweepAngle = 0;
+ private int mLeftMostAngle = 0;
+ private int mRightMostAngle = 0;
+ private int mAngleOffset = 0;
+ private int mPanningDirection = 0;
+ private final Paint mBackgroundPaint = new Paint();
+ private final Paint mSweepAreaPaint = new Paint();
+ private final Paint mIndicatorPaint = new Paint();
+ private RectF mRect;
+
+ public IndicationView(Context context, AttributeSet attrs) {
+ super(context, attrs);
+ mSweepAreaPaint.setStyle(Paint.Style.FILL);
+ mSweepAreaPaint.setColor(Color.RED);
+
+ mBackgroundPaint.setStyle(Paint.Style.FILL);
+ mBackgroundPaint.setColor(Color.BLUE);
+
+ mIndicatorPaint.setStyle(Paint.Style.FILL);
+ mIndicatorPaint.setColor(0xFFFFBBBB);
+
+ mRect = new RectF();
+ }
+
+ protected void onSizeChanged(int w, int h, int oldw, int oldh) {
+ mRect.set(0, 0, w, h * 2);
+ }
+
+ public void setMaxSweepAngle(int angle) {
+ mStartAngle = -90 - angle / 2;
+ mMaxSweepAngle = angle;
+ }
+
+ public void setSweepAngle(int angle) {
+ // The panning direction will be decided after user pan more than 10 degrees in one
+ // direction.
+ if (mPanningDirection == PAN_DIRECTION_NONE) {
+ if (angle > 10) {
+ mPanningDirection = PAN_DIRECTION_RIGHT;
+ mAngleOffset = -mMaxSweepAngle / 2;
+ }
+ if (angle < -10) {
+ mPanningDirection = PAN_DIRECTION_LEFT;
+ mAngleOffset = mMaxSweepAngle / 2;
+ }
+ } else {
+ mSweepAngle = angle;
+ if (mPanningDirection == PAN_DIRECTION_RIGHT) {
+ // Bounded by the left most angle.
+ mSweepAngle = Math.max(mLeftMostAngle, mSweepAngle);
+ // Bounded by the max angle in the right direction.
+ mSweepAngle = Math.min(mMaxSweepAngle, mSweepAngle);
+ // The right most angle is adjusted.
+ mRightMostAngle = Math.max(mRightMostAngle, mSweepAngle);
+ }
+ if (mPanningDirection == PAN_DIRECTION_LEFT) {
+ // Bounded by the right most angle.
+ mSweepAngle = Math.min(mRightMostAngle, mSweepAngle);
+ // Bounded by the max angle in the left direction.
+ mSweepAngle = Math.max(-mMaxSweepAngle, mSweepAngle);
+ // The left most angle is adjusted.
+ mLeftMostAngle = Math.min(mLeftMostAngle, mSweepAngle);
+ }
+ }
+ }
+
+ public void resetAngles() {
+ mSweepAngle = 0;
+ mLeftMostAngle = 0;
+ mRightMostAngle = 0;
+ mAngleOffset = 0;
+ mPanningDirection = PAN_DIRECTION_NONE;
+ }
+
+ @Override
+ protected void onDraw(Canvas canvas) {
+ // the background
+ canvas.drawArc(mRect, mStartAngle, mMaxSweepAngle, true, mBackgroundPaint);
+ if (mPanningDirection != PAN_DIRECTION_NONE) {
+ // the spanned area
+ canvas.drawArc(mRect, -90 + mLeftMostAngle + mAngleOffset,
+ mRightMostAngle - mLeftMostAngle, true, mSweepAreaPaint);
+ // the indication line
+ canvas.drawArc(mRect, -91 + mSweepAngle + mAngleOffset, 2, true, mIndicatorPaint);
+ }
+ }
+}
diff --git a/src/com/android/camera/panorama/MosaicFrameProcessor.java b/src/com/android/camera/panorama/MosaicFrameProcessor.java
index 2e077fa..5af074b 100644
--- a/src/com/android/camera/panorama/MosaicFrameProcessor.java
+++ b/src/com/android/camera/panorama/MosaicFrameProcessor.java
@@ -45,14 +45,6 @@ public class MosaicFrameProcessor {
private ProgressListener mProgressListener;
- private float mCompassValueX;
- private float mCompassValueY;
- private float mCompassValueXStart;
- private float mCompassValueYStart;
- private int mCompassThreshold;
- private int mTraversedAngleX;
- private int mTraversedAngleY;
-
// Panning rate is in unit of percentage of image content translation / second.
private float mPanningRateX;
private float mPanningRateY;
@@ -62,13 +54,11 @@ public class MosaicFrameProcessor {
private int mPreviewBufferSize;
public interface ProgressListener {
- public void onProgress(boolean isFinished, float panningRateX, float panningRateY,
- int traversedAngleX, int traversedAngleY);
+ public void onProgress(boolean isFinished, float panningRateX, float panningRateY);
}
- public MosaicFrameProcessor(int sweepAngle, int previewWidth, int previewHeight, int bufSize) {
+ public MosaicFrameProcessor(int previewWidth, int previewHeight, int bufSize) {
mMosaicer = new Mosaic();
- mCompassThreshold = sweepAngle;
mPreviewWidth = previewWidth;
mPreviewHeight = previewHeight;
mPreviewBufferSize = bufSize;
@@ -146,36 +136,20 @@ public class MosaicFrameProcessor {
// Access the timestamp associated with it...
long timestamp = mFrameTimestamp[mCurrProcessFrameIdx];
- // Keep track of what compass bearing we started at...
- if (mTotalFrameCount == 0) { // First frame
- mCompassValueXStart = mCompassValueX;
- mCompassValueYStart = mCompassValueY;
- }
-
- // By what angle has the camera moved since start of capture?
- mTraversedAngleX = (int) PanoUtil.calculateDifferenceBetweenAngles(
- mCompassValueX, mCompassValueXStart);
- mTraversedAngleY = (int) PanoUtil.calculateDifferenceBetweenAngles(
- mCompassValueY, mCompassValueYStart);
-
// TODO: make the termination condition regarding reaching
// MAX_NUMBER_OF_FRAMES solely determined in the library.
- if (mTotalFrameCount < MAX_NUMBER_OF_FRAMES
- && mTraversedAngleX < mCompassThreshold
- && mTraversedAngleY < mCompassThreshold) {
+ if (mTotalFrameCount < MAX_NUMBER_OF_FRAMES) {
// If we are still collecting new frames for the current mosaic,
// process the new frame.
calculateTranslationRate(timestamp);
// Publish progress of the ongoing processing
if (mProgressListener != null) {
- mProgressListener.onProgress(false, mPanningRateX, mPanningRateY,
- mTraversedAngleX, mTraversedAngleY);
+ mProgressListener.onProgress(false, mPanningRateX, mPanningRateY);
}
} else {
if (mProgressListener != null) {
- mProgressListener.onProgress(true, mPanningRateX, mPanningRateY,
- mTraversedAngleX, mTraversedAngleY);
+ mProgressListener.onProgress(true, mPanningRateX, mPanningRateY);
}
}
}
@@ -204,9 +178,4 @@ public class MosaicFrameProcessor {
mTranslationLastX = translationCurrX;
mTranslationLastY = translationCurrY;
}
-
- public void updateCompassValue(float valueX, float valueY) {
- mCompassValueX = valueX;
- mCompassValueY = valueY;
- }
}
diff --git a/src/com/android/camera/panorama/PanoramaActivity.java b/src/com/android/camera/panorama/PanoramaActivity.java
index b6246ee..beb24de 100644
--- a/src/com/android/camera/panorama/PanoramaActivity.java
+++ b/src/com/android/camera/panorama/PanoramaActivity.java
@@ -55,6 +55,7 @@ import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.Gravity;
+import android.view.OrientationEventListener;
import android.view.View;
import android.view.WindowManager;
import android.view.animation.Animation;
@@ -102,7 +103,7 @@ public class PanoramaActivity extends Activity implements
private Button mStopCaptureButton;
private View mReviewLayout;
private ImageView mReview;
- private CaptureView mCaptureView;
+ private IndicationView mIndicationView;
private MosaicRendererSurfaceView mMosaicView;
private TextView mTooFastPrompt;
private Animation mSlideIn, mSlideOut;
@@ -114,6 +115,22 @@ public class PanoramaActivity extends Activity implements
private String mDialogOk;
private AlertDialog mAlertDialog;
+ private float mCompassValueX;
+ private float mCompassValueY;
+ private float mCompassValueXStart;
+ private float mCompassValueYStart;
+ private float mCompassValueXStartBuffer;
+ private float mCompassValueYStartBuffer;
+ private int mCompassThreshold;
+ private int mTraversedAngleX;
+ private int mTraversedAngleY;
+ private long mTimestamp;
+ // Control variables for the terminate condition.
+ private int mMinAngleX;
+ private int mMaxAngleX;
+ private int mMinAngleY;
+ private int mMaxAngleY;
+
private RotateImageView mThumbnailView;
private Thumbnail mThumbnail;
private SharePopup mSharePopup;
@@ -127,14 +144,17 @@ public class PanoramaActivity extends Activity implements
private Sensor mSensor;
private ModePicker mModePicker;
private MosaicFrameProcessor mMosaicFrameProcessor;
- private String mCurrentImagePath = null;
private long mTimeTaken;
private Handler mMainHandler;
private SurfaceTexture mSurfaceTexture;
private boolean mThreadRunning;
private float[] mTransformMatrix;
private float mHorizontalViewAngle;
- private float mVerticalViewAngle;
+
+ private PanoOrientationEventListener mOrientationEventListener;
+ // The value could be 0, 1, 2, 3 for the 4 different orientations measured in clockwise
+ // respectively.
+ private int mDeviceOrientation;
private class MosaicJpeg {
public MosaicJpeg(byte[] data, int width, int height) {
@@ -148,6 +168,19 @@ public class PanoramaActivity extends Activity implements
public final int height;
}
+ private class PanoOrientationEventListener extends OrientationEventListener {
+ public PanoOrientationEventListener(Context context) {
+ super(context);
+ }
+
+ @Override
+ public void onOrientationChanged(int orientation) {
+ // Default to the last known orientation.
+ if (orientation == ORIENTATION_UNKNOWN) return;
+ mDeviceOrientation = ((orientation + 45) / 90) % 4;
+ }
+ }
+
@Override
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
@@ -163,6 +196,8 @@ public class PanoramaActivity extends Activity implements
mSensor = mSensorManager.getDefaultSensor(Sensor.TYPE_ORIENTATION);
}
+ mOrientationEventListener = new PanoOrientationEventListener(this);
+
mTransformMatrix = new float[16];
mPreparePreviewString =
@@ -289,8 +324,8 @@ public class PanoramaActivity extends Activity implements
parameters.setRecordingHint(false);
- mHorizontalViewAngle = parameters.getHorizontalViewAngle();
- mVerticalViewAngle = parameters.getVerticalViewAngle();
+ mHorizontalViewAngle = ((mDeviceOrientation % 2) == 0) ?
+ parameters.getHorizontalViewAngle() : parameters.getVerticalViewAngle();
}
public int getPreviewBufSize() {
@@ -391,20 +426,30 @@ public class PanoramaActivity extends Activity implements
mPanoControlLayout.startAnimation(mSlideOut);
mPanoControlLayout.setVisibility(View.GONE);
+ mCompassValueXStart = mCompassValueXStartBuffer;
+ mCompassValueYStart = mCompassValueYStartBuffer;
+ mMinAngleX = 0;
+ mMaxAngleX = 0;
+ mMinAngleY = 0;
+ mMaxAngleY = 0;
+ mTimestamp = 0;
+
mMosaicFrameProcessor.setProgressListener(new MosaicFrameProcessor.ProgressListener() {
@Override
- public void onProgress(boolean isFinished, float panningRateX, float panningRateY,
- int traversedAngleX, int traversedAngleY) {
- if (isFinished) {
+ public void onProgress(boolean isFinished, float panningRateX, float panningRateY) {
+ if (isFinished
+ || (mMaxAngleX - mMinAngleX >= DEFAULT_SWEEP_ANGLE)
+ || (mMaxAngleY - mMinAngleY >= DEFAULT_SWEEP_ANGLE)) {
stopCapture();
} else {
- updateProgress(panningRateX, panningRateY, traversedAngleX, traversedAngleY);
+ updateProgress(panningRateX);
}
}
});
mStopCaptureButton.setVisibility(View.VISIBLE);
- mCaptureView.setVisibility(View.VISIBLE);
+ mIndicationView.resetAngles();
+ mIndicationView.setVisibility(View.VISIBLE);
mMosaicView.setVisibility(View.VISIBLE);
}
@@ -434,25 +479,18 @@ public class PanoramaActivity extends Activity implements
}
}
- private void updateProgress(float panningRateX, float panningRateY,
- int traversedAngleX, int traversedAngleY) {
-
+ private void updateProgress(float panningRate) {
mMosaicView.setReady();
mMosaicView.requestRender();
// TODO: Now we just display warning message by the panning speed.
// Since we only support horizontal panning, we should display a warning message
// in UI when there're significant vertical movements.
- if ((panningRateX * mHorizontalViewAngle > PANNING_SPEED_THRESHOLD)
- || (panningRateY * mVerticalViewAngle > PANNING_SPEED_THRESHOLD)) {
+ if (Math.abs(panningRate * mHorizontalViewAngle) > PANNING_SPEED_THRESHOLD) {
// TODO: draw speed indication according to the UI spec.
mTooFastPrompt.setVisibility(View.VISIBLE);
- mCaptureView.setSweepAngle(Math.max(traversedAngleX, traversedAngleY) + 1);
- mCaptureView.invalidate();
} else {
mTooFastPrompt.setVisibility(View.GONE);
- mCaptureView.setSweepAngle(Math.max(traversedAngleX, traversedAngleY) + 1);
- mCaptureView.invalidate();
}
}
@@ -462,8 +500,8 @@ public class PanoramaActivity extends Activity implements
mCaptureState = CAPTURE_VIEWFINDER;
mCaptureLayout = (View) findViewById(R.id.pano_capture_layout);
- mCaptureView = (CaptureView) findViewById(R.id.pano_capture_view);
- mCaptureView.setStartAngle(-DEFAULT_SWEEP_ANGLE / 2);
+ mIndicationView = (IndicationView) findViewById(R.id.pano_capture_view);
+ mIndicationView.setMaxSweepAngle(DEFAULT_SWEEP_ANGLE);
mStopCaptureButton = (Button) findViewById(R.id.pano_capture_stop_button);
mTooFastPrompt = (TextView) findViewById(R.id.pano_capture_too_fast_textview);
@@ -604,7 +642,7 @@ public class PanoramaActivity extends Activity implements
mReviewLayout.setVisibility(View.GONE);
mStopCaptureButton.setVisibility(View.GONE);
- mCaptureView.setVisibility(View.GONE);
+ mIndicationView.setVisibility(View.GONE);
mPanoControlLayout.setVisibility(View.VISIBLE);
mPanoControlLayout.startAnimation(mSlideIn);
mCaptureLayout.setVisibility(View.VISIBLE);
@@ -623,7 +661,6 @@ public class PanoramaActivity extends Activity implements
}
mCaptureLayout.setVisibility(View.GONE);
mReviewLayout.setVisibility(View.VISIBLE);
- mCaptureView.setSweepAngle(0);
}
private Uri savePanorama(byte[] jpegData, int orientation) {
@@ -645,7 +682,7 @@ public class PanoramaActivity extends Activity implements
if (mPausing || mThreadRunning) return;
if (mMosaicFrameProcessor == null) {
// Start the activity for the first time.
- mMosaicFrameProcessor = new MosaicFrameProcessor(DEFAULT_SWEEP_ANGLE - 5,
+ mMosaicFrameProcessor = new MosaicFrameProcessor(
mPreviewWidth, mPreviewHeight, getPreviewBufSize());
}
mMosaicFrameProcessor.initialize();
@@ -660,6 +697,7 @@ public class PanoramaActivity extends Activity implements
mMosaicView.onPause();
mSensorManager.unregisterListener(mListener);
clearMosaicFrameProcessorIfNeeded();
+ mOrientationEventListener.disable();
System.gc();
}
@@ -668,9 +706,10 @@ public class PanoramaActivity extends Activity implements
super.onResume();
mPausing = false;
+ mOrientationEventListener.enable();
/*
* It is not necessary to get accelerometer events at a very high rate,
- * by using a slower rate (SENSOR_DELAY_UI), we get an automatic
+ * by using a game rate (SENSOR_DELAY_UI), we get an automatic
* low-pass filter, which "extracts" the gravity component of the
* acceleration. As an added benefit, we use less power and CPU
* resources.
@@ -688,27 +727,46 @@ public class PanoramaActivity extends Activity implements
mMosaicView.onResume();
}
- private final SensorEventListener mListener = new SensorEventListener() {
- private float mCompassCurrX; // degrees
- private float mCompassCurrY; // degrees
- private float mTimestamp;
+ private void updateCompassValue() {
+ // By what angle has the camera moved since start of capture?
+ mTraversedAngleX = (int) (mCompassValueX - mCompassValueXStart);
+ mTraversedAngleY = (int) (mCompassValueY - mCompassValueYStart);
+ mMinAngleX = Math.min(mMinAngleX, mTraversedAngleX);
+ mMaxAngleX = Math.max(mMaxAngleX, mTraversedAngleX);
+ mMinAngleY = Math.min(mMinAngleY, mTraversedAngleY);
+ mMaxAngleY = Math.max(mMaxAngleY, mTraversedAngleY);
+
+ // Use orientation to identify if the user is panning to the right or the left.
+ switch (mDeviceOrientation) {
+ case 0:
+ mIndicationView.setSweepAngle(-mTraversedAngleX);
+ break;
+ case 1:
+ mIndicationView.setSweepAngle(mTraversedAngleY);
+ break;
+ case 2:
+ mIndicationView.setSweepAngle(mTraversedAngleX);
+ break;
+ case 3:
+ mIndicationView.setSweepAngle(-mTraversedAngleY);
+ break;
+ }
+ mIndicationView.invalidate();
+ }
+ private final SensorEventListener mListener = new SensorEventListener() {
public void onSensorChanged(SensorEvent event) {
if (event.sensor.getType() == Sensor.TYPE_GYROSCOPE) {
if (mTimestamp != 0) {
final float dT = (event.timestamp - mTimestamp) * NS2S;
- mCompassCurrX += event.values[1] * dT * 180.0f / Math.PI;
- mCompassCurrY += event.values[0] * dT * 180.0f / Math.PI;
+ mCompassValueX += event.values[1] * dT * 180.0f / Math.PI;
+ mCompassValueY += event.values[0] * dT * 180.0f / Math.PI;
+ mCompassValueXStartBuffer = mCompassValueX;
+ mCompassValueYStartBuffer = mCompassValueY;
+ updateCompassValue();
}
mTimestamp = event.timestamp;
- } else if (event.sensor.getType() == Sensor.TYPE_ORIENTATION) {
- mCompassCurrX = event.values[0];
- mCompassCurrY = event.values[1];
- }
-
- if (mMosaicFrameProcessor != null) {
- mMosaicFrameProcessor.updateCompassValue(mCompassCurrX, mCompassCurrY);
}
}