summaryrefslogtreecommitdiffstats
path: root/src/com/android/camera/ui
diff options
context:
space:
mode:
Diffstat (limited to 'src/com/android/camera/ui')
-rw-r--r--src/com/android/camera/ui/AbstractIndicator.java8
-rw-r--r--src/com/android/camera/ui/BasicIndicator.java14
-rw-r--r--src/com/android/camera/ui/GLListView.java6
-rw-r--r--src/com/android/camera/ui/GLOptionHeader.java7
-rw-r--r--src/com/android/camera/ui/GLOptionItem.java53
-rw-r--r--src/com/android/camera/ui/GLRootView.java160
-rw-r--r--src/com/android/camera/ui/GLView.java10
-rw-r--r--src/com/android/camera/ui/GpsIndicator.java4
-rw-r--r--src/com/android/camera/ui/HeadUpDisplay.java43
-rw-r--r--src/com/android/camera/ui/OtherSettingsIndicator.java14
-rw-r--r--src/com/android/camera/ui/PopupWindow.java33
-rw-r--r--src/com/android/camera/ui/PopupWindowStencilImpl.java5
-rw-r--r--src/com/android/camera/ui/PreferenceAdapter.java29
-rw-r--r--src/com/android/camera/ui/RawTexture.java27
-rw-r--r--src/com/android/camera/ui/StateListTexture.java79
-rw-r--r--src/com/android/camera/ui/Texture.java31
-rw-r--r--src/com/android/camera/ui/ZoomIndicator.java (renamed from src/com/android/camera/ui/GLZoomIndicator.java)10
17 files changed, 303 insertions, 230 deletions
diff --git a/src/com/android/camera/ui/AbstractIndicator.java b/src/com/android/camera/ui/AbstractIndicator.java
index 531640d..77fbc87 100644
--- a/src/com/android/camera/ui/AbstractIndicator.java
+++ b/src/com/android/camera/ui/AbstractIndicator.java
@@ -1,17 +1,19 @@
package com.android.camera.ui;
+import android.content.Context;
import android.graphics.Rect;
import javax.microedition.khronos.opengles.GL11;
public abstract class AbstractIndicator extends GLView {
- private static final int DEFAULT_PADDING = 5;
+ private static final int DEFAULT_PADDING = 3;
abstract protected ResourceTexture getIcon();
- public AbstractIndicator() {
- setPaddings(DEFAULT_PADDING, 0, DEFAULT_PADDING, 0);
+ public AbstractIndicator(Context context) {
+ int padding = GLRootView.dpToPixel(context, DEFAULT_PADDING);
+ setPaddings(padding, 0, padding, 0);
}
@Override
diff --git a/src/com/android/camera/ui/BasicIndicator.java b/src/com/android/camera/ui/BasicIndicator.java
index 0d97ee7..3a5b330 100644
--- a/src/com/android/camera/ui/BasicIndicator.java
+++ b/src/com/android/camera/ui/BasicIndicator.java
@@ -4,6 +4,7 @@ import android.content.Context;
import com.android.camera.IconListPreference;
import com.android.camera.R;
+import com.android.camera.Util;
import com.android.camera.ui.GLListView.OnItemSelectedListener;
public class BasicIndicator extends AbstractIndicator {
@@ -15,7 +16,8 @@ public class BasicIndicator extends AbstractIndicator {
private PreferenceAdapter mModel;
private String mOverride;
- public BasicIndicator(IconListPreference preference) {
+ public BasicIndicator(Context context, IconListPreference preference) {
+ super(context);
mPreference = preference;
mIcon = new ResourceTexture[preference.getIconIds().length];
mIndex = preference.findIndexOfValue(preference.getValue());
@@ -24,12 +26,16 @@ public class BasicIndicator extends AbstractIndicator {
@Override
public void overrideSettings(String key, String settings) {
IconListPreference pref = mPreference;
-
if (!pref.getKey().equals(key)) return;
+ if (Util.equals(mOverride, settings)) return;
+
mOverride = settings;
- mIndex = pref.findIndexOfValue(
+ int index = pref.findIndexOfValue(
settings == null ? pref.getValue() : settings);
- invalidate();
+ if (mIndex != index) {
+ mIndex = index;
+ invalidate();
+ }
}
@Override
diff --git a/src/com/android/camera/ui/GLListView.java b/src/com/android/camera/ui/GLListView.java
index c5161ea..bbee042 100644
--- a/src/com/android/camera/ui/GLListView.java
+++ b/src/com/android/camera/ui/GLListView.java
@@ -66,8 +66,8 @@ public class GLListView extends GLView {
int height = bounds.height();
mHighLight.setSize(width, height);
if (mHighLight.bind(root, gl)) {
- root.draw2D(bounds.left - mScrollX,
- bounds.top - mScrollY, width, height);
+ mHighLight.draw(root,
+ bounds.left - mScrollX, bounds.top - mScrollY);
}
}
}
@@ -80,7 +80,7 @@ public class GLListView extends GLView {
mScrollbar.setSize(width, height);
if (mScrollbar.bind(root, gl)) {
int yoffset = mScrollY * getHeight() / mScrollHeight;
- root.draw2D(getWidth() - width, yoffset, width, height);
+ mScrollbar.draw(root, getWidth() - width, yoffset);
}
}
}
diff --git a/src/com/android/camera/ui/GLOptionHeader.java b/src/com/android/camera/ui/GLOptionHeader.java
index d49fbc4..bec9b76 100644
--- a/src/com/android/camera/ui/GLOptionHeader.java
+++ b/src/com/android/camera/ui/GLOptionHeader.java
@@ -1,5 +1,6 @@
package com.android.camera.ui;
+import android.content.Context;
import android.graphics.Rect;
import com.android.camera.ListPreference;
@@ -8,15 +9,17 @@ import javax.microedition.khronos.opengles.GL11;
public class GLOptionHeader extends GLView {
private static final int FONT_COLOR = 0xFF979797;
+ private static final float FONT_SIZE = 12;
private final ListPreference mPreference;
private final StringTexture mTitle;
private NinePatchTexture mBackground;
- public GLOptionHeader(ListPreference preference) {
+ public GLOptionHeader(Context context, ListPreference preference) {
+ float fontSize = GLRootView.dpToPixel(context, FONT_SIZE);
mPreference = preference;
mTitle = StringTexture.newInstance(
- preference.getTitle(), 16, FONT_COLOR);
+ preference.getTitle(), fontSize, FONT_COLOR);
}
public ListPreference getPreference() {
diff --git a/src/com/android/camera/ui/GLOptionItem.java b/src/com/android/camera/ui/GLOptionItem.java
index 00d7a55..9229813 100644
--- a/src/com/android/camera/ui/GLOptionItem.java
+++ b/src/com/android/camera/ui/GLOptionItem.java
@@ -1,5 +1,6 @@
package com.android.camera.ui;
+import static com.android.camera.ui.GLRootView.dpToPixel;
import android.content.Context;
import android.graphics.Color;
import android.graphics.Rect;
@@ -11,46 +12,68 @@ import javax.microedition.khronos.opengles.GL11;
public class GLOptionItem extends GLView {
private static final int FONT_COLOR = Color.WHITE;
- private static final int MINIMAL_WIDTH = 180;
- private static final int MINIMAL_HEIGHT = 48;
- private static final int NO_ICON_LEADING_SPACE = 15;
- private static final int TEXT_LEFT_PADDING = 9;
- private static final int TEXT_RIGHT_PADDING = 15;
+ private static final float FONT_SIZE = 18;
+
+ private static final int MINIMAL_WIDTH = 120;
+ private static final int MINIMAL_HEIGHT = 32;
+
+ private static final int NO_ICON_LEADING_SPACE = 10;
+ private static final int TEXT_LEFT_PADDING = 6;
+ private static final int TEXT_RIGHT_PADDING = 10;
+
private static final float ENABLED_ALPHA = 1f;
private static final float DISABLED_ALPHA = 0.3f;
private static ResourceTexture sCheckOn;
private static ResourceTexture sCheckOff;
+ private static int sNoIconLeadingSpace;
+ private static int sTextLeftPadding;
+ private static int sTextRightPadding;
+ private static int sMinimalWidth;
+ private static int sMinimalHeight;
+ private static float sFontSize;
+
private final ResourceTexture mIcon;
private final StringTexture mText;
private boolean mEnabled = true;
private ResourceTexture mCheckBox;
- private static void initCheckIcons(Context context) {
+
+ private static void initializeStaticVariables(Context context) {
if (sCheckOn != null) return;
+
sCheckOn = new ResourceTexture(context, R.drawable.ic_menuselect_on);
sCheckOff = new ResourceTexture(context, R.drawable.ic_menuselect_off);
+
+ sNoIconLeadingSpace = dpToPixel(context, NO_ICON_LEADING_SPACE);
+ sTextLeftPadding = dpToPixel(context, TEXT_LEFT_PADDING);
+ sTextRightPadding = dpToPixel(context, TEXT_RIGHT_PADDING);
+ sMinimalWidth = dpToPixel(context, MINIMAL_WIDTH);
+ sMinimalHeight = dpToPixel(context, MINIMAL_HEIGHT);
+
+ sFontSize = dpToPixel(context, FONT_SIZE);
}
public GLOptionItem(Context context, int iconId, String title) {
- initCheckIcons(context);
+ initializeStaticVariables(context);
mIcon = iconId == 0 ? null : new ResourceTexture(context, iconId);
- mText = StringTexture.newInstance(title, 26, FONT_COLOR);
+ mText = StringTexture.newInstance(title, sFontSize, FONT_COLOR);
mCheckBox = sCheckOff;
}
@Override
protected void onMeasure(int widthSpec, int heightSpec) {
- int width = (mIcon == null ? NO_ICON_LEADING_SPACE : mIcon.getWidth())
- + mText.getWidth() + mCheckBox.getWidth()
- + TEXT_RIGHT_PADDING + TEXT_LEFT_PADDING;
+ int width = mIcon == null ? sNoIconLeadingSpace : mIcon.getWidth();
+ width += mText.getWidth() + mCheckBox.getWidth();
+ width += sTextRightPadding + sTextLeftPadding;
+
int height = Math.max(Math.max(mIcon == null ? 0 : mIcon.getHeight(),
mText.getHeight()), mCheckBox.getHeight());
- width = Math.max(MINIMAL_WIDTH, width);
- height = Math.max(MINIMAL_HEIGHT, height);
+ width = Math.max(sMinimalWidth, width);
+ height = Math.max(sMinimalHeight, height);
new MeasureHelper(this)
.setPreferedContentSize(width, height)
@@ -78,11 +101,11 @@ public class GLOptionItem extends GLView {
}
xoffset += icon.getWidth();
} else {
- xoffset += NO_ICON_LEADING_SPACE;
+ xoffset += sNoIconLeadingSpace;
}
StringTexture title = mText;
- xoffset += TEXT_LEFT_PADDING;
+ xoffset += sTextLeftPadding;
if (title.bind(root, gl)) {
int yoffset = p.top + (height - title.getHeight()) / 2;
//TODO: cut the text if it is too long
diff --git a/src/com/android/camera/ui/GLRootView.java b/src/com/android/camera/ui/GLRootView.java
index 23db5f8..b16d443 100644
--- a/src/com/android/camera/ui/GLRootView.java
+++ b/src/com/android/camera/ui/GLRootView.java
@@ -7,21 +7,22 @@ import android.graphics.PixelFormat;
import android.opengl.GLSurfaceView;
import android.opengl.GLU;
import android.os.Process;
+import android.os.SystemClock;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.MotionEvent;
+import android.view.animation.Animation;
import android.view.animation.Transformation;
import com.android.camera.Util;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
+import java.util.ArrayList;
import java.util.Stack;
import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;
-import java.util.concurrent.TimeUnit;
-import java.util.concurrent.TimeoutException;
import javax.microedition.khronos.egl.EGLConfig;
import javax.microedition.khronos.opengles.GL10;
@@ -36,18 +37,19 @@ public class GLRootView extends GLSurfaceView
private int mFrameCount = 0;
private long mFrameCountingStart = 0;
- // The event processing timeout for GL events, 1.5 seconds
- private static final long EVENT_TIMEOUT = 1500;
private static final int VERTEX_BUFFER_SIZE = 8;
private static final int FLAG_INITIALIZED = 1;
private static final int FLAG_NEED_LAYOUT = 2;
- private static final int FLAG_TEXTURE_MODE = 4;
+
+ private static float sPixelDensity = -1f;
private GL11 mGL;
private GLView mContentView;
private DisplayMetrics mDisplayMetrics;
+ private final ArrayList<Animation> mAnimations = new ArrayList<Animation>();
+
private final Stack<Transformation> mFreeTransform =
new Stack<Transformation>();
@@ -59,16 +61,46 @@ public class GLRootView extends GLSurfaceView
private final float mMatrixValues[] = new float[16];
+ private final float mCoordBuffer[] = new float[8];
+ private final float mPointBuffer[] = new float[4];
+
private ByteBuffer mVertexBuffer;
private ByteBuffer mTexCoordBuffer;
private int mFlags = FLAG_NEED_LAYOUT;
+ private long mAnimationTime;
public GLRootView(Context context) {
super(context);
initialize();
}
+ void registerLaunchedAnimation(Animation animation) {
+ // Register the newly launched animation so that we can set the start
+ // time more precisely. (Usually, it takes much longer for the first
+ // rendering, so we set the animation start time as the time we
+ // complete rendering)
+ mAnimations.add(animation);
+ }
+
+ public long currentAnimationTimeMillis() {
+ return mAnimationTime;
+ }
+
+ public synchronized static float dpToPixel(Context context, float dp) {
+ if (sPixelDensity < 0) {
+ DisplayMetrics metrics = new DisplayMetrics();
+ ((Activity) context).getWindowManager()
+ .getDefaultDisplay().getMetrics(metrics);
+ sPixelDensity = metrics.density;
+ }
+ return sPixelDensity * dp;
+ }
+
+ public static int dpToPixel(Context context, int dp) {
+ return (int)(dpToPixel(context, (float) dp) + .5f);
+ }
+
public Transformation obtainTransformation() {
return mFreeTransform.isEmpty() ? new Transformation() : mFreeTransform.pop();
}
@@ -103,7 +135,7 @@ public class GLRootView extends GLSurfaceView
setEGLConfigChooser(8, 8, 8, 8, 0, 0);
getHolder().setFormat(PixelFormat.TRANSLUCENT);
setZOrderOnTop(true);
- super.setDebugFlags(DEBUG_CHECK_GL_ERROR);
+ setDebugFlags(DEBUG_CHECK_GL_ERROR);
setRenderer(this);
@@ -196,7 +228,6 @@ public class GLRootView extends GLSurfaceView
gl.glEnableClientState(GL10.GL_VERTEX_ARRAY);
gl.glEnableClientState(GL10.GL_TEXTURE_COORD_ARRAY);
gl.glEnable(GL11.GL_TEXTURE_2D);
- gl.glEnable(GL11.GL_STENCIL_TEST);
gl.glTexEnvf(GL11.GL_TEXTURE_ENV,
GL11.GL_TEXTURE_ENV_MODE, GL11.GL_REPLACE);
@@ -210,10 +241,6 @@ public class GLRootView extends GLSurfaceView
gl.glTexCoordPointer(2, GL11.GL_FLOAT, 0, mTexCoordBuffer);
}
- protected void setTexCoords(float ... point) {
- mTexCoordBuffer.asFloatBuffer().put(point).position(0);
- }
-
/**
* Called when the OpenGL surface is recreated without destroying the
* context.
@@ -223,6 +250,7 @@ public class GLRootView extends GLSurfaceView
Log.v(TAG, "onSurfaceChanged: " + width + "x" + height
+ ", gl10: " + gl1.toString());
GL11 gl = (GL11) gl1;
+ mGL = gl;
gl.glMatrixMode(GL11.GL_PROJECTION);
gl.glLoadIdentity();
@@ -250,26 +278,39 @@ public class GLRootView extends GLSurfaceView
}
public void drawRect(int x, int y, int width, int height) {
- drawRect(x, y, width, height, mTransformation.getAlpha());
+ float matrix[] = mMatrixValues;
+ mTransformation.getMatrix().getValues(matrix);
+ drawRect(x, y, width, height, matrix, mTransformation.getAlpha());
}
- public void drawRect(int x, int y, int width, int height, float alpha) {
+ private void drawRect(
+ int x, int y, int width, int height, float matrix[], float alpha) {
GL11 gl = mGL;
gl.glPushMatrix();
- mTransformation.getMatrix().getValues(mMatrixValues);
-
setAlphaValue(alpha);
- gl.glMultMatrixf(toGLMatrix(mMatrixValues), 0);
+ gl.glMultMatrixf(toGLMatrix(matrix), 0);
gl.glTranslatef(x, y, 0);
gl.glScalef(width, height, 1);
gl.glDrawArrays(GL11.GL_TRIANGLE_STRIP, 0, 4);
gl.glPopMatrix();
}
- public void clipRect(int x, int y, int width, int height) {
- float point[] = new float[]{x, y + height, x + width, y};
+ public void drawRect(int x, int y, int width, int height, float alpha) {
+ float matrix[] = mMatrixValues;
+ mTransformation.getMatrix().getValues(matrix);
+ drawRect(x, y, width, height, matrix, alpha);
+ }
- mTransformation.getMatrix().mapPoints(point);
+ private float[] mapPoints(Matrix matrix, int x1, int y1, int x2, int y2) {
+ float[] point = mPointBuffer;
+ point[0] = x1; point[1] = y1; point[2] = x2; point[3] = y2;
+ matrix.mapPoints(point);
+ return point;
+ }
+
+ public void clipRect(int x, int y, int width, int height) {
+ float point[] = mapPoints(
+ mTransformation.getMatrix(), x, y + height, x + width, y);
// mMatrix could be a rotation matrix. In this case, we need to find
// the boundaries after rotation. (only handle 90 * n degrees)
@@ -302,20 +343,23 @@ public class GLRootView extends GLSurfaceView
return v;
}
- public void draw2D(int x, int y, int width, int height, float alpha) {
+ public void drawTexture(
+ Texture texture, int x, int y, int width, int height, float alpha) {
+
if (width <= 0 || height <= 0) return ;
Matrix matrix = mTransformation.getMatrix();
matrix.getValues(mMatrixValues);
- // Test if it has been rotated, if it is, glDrawTexiOES won't work
- if (mMatrixValues[1] != 0
- || mMatrixValues[3] != 0 || mMatrixValues[0] < 0) {
- drawRect(x, y, width, height, alpha);
+ // Test whether it has been rotated or flipped, if so, glDrawTexiOES
+ // won't work
+ if (isMatrixRotatedOrFlipped(mMatrixValues)) {
+ texture.getTextureCoords(mCoordBuffer, 0);
+ mTexCoordBuffer.asFloatBuffer().put(mCoordBuffer).position(0);
+ drawRect(x, y, width, height, mMatrixValues, alpha);
} else {
// draw the rect from bottom-left to top-right
- float points[] = new float[]{x, y + height, x + width, y};
- matrix.mapPoints(points);
+ float points[] = mapPoints(matrix, x, y + height, x + width, y);
x = (int) points[0];
y = (int) points[1];
width = (int) points[2] - x;
@@ -327,8 +371,14 @@ public class GLRootView extends GLSurfaceView
}
}
- public void draw2D(int x, int y, int width, int height) {
- draw2D(x, y, width, height, mTransformation.getAlpha());
+ private static boolean isMatrixRotatedOrFlipped(float matrix[]) {
+ return matrix[Matrix.MSKEW_X] != 0 || matrix[Matrix.MSKEW_Y] != 0
+ || matrix[Matrix.MSCALE_X] < 0 || matrix[Matrix.MSCALE_Y] > 0;
+ }
+
+ public void drawTexture(
+ Texture texture, int x, int y, int width, int height) {
+ drawTexture(texture, x, y, width, height, mTransformation.getAlpha());
}
// This is a GLSurfaceView.Renderer callback
@@ -345,34 +395,36 @@ public class GLRootView extends GLSurfaceView
}
++mFrameCount;
}
+
if ((mFlags & FLAG_NEED_LAYOUT) != 0) layoutContentPane();
- // gl.glScissor(0, 0, getWidth(), getHeight());
- gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_STENCIL_BUFFER_BIT);
+ gl.glClear(GL10.GL_COLOR_BUFFER_BIT);
gl.glEnable(GL11.GL_BLEND);
gl.glBlendFunc(GL11.GL_ONE, GL11.GL_ONE_MINUS_SRC_ALPHA);
- /*gl.glBindTexture(GL11.GL_TEXTURE_2D, 0);
+ /*gl.glDisable(GL11.GL_TEXTURE_2D);
gl.glColor4f(0, 0, 0.5f, 0.4f);
- drawRect(30, 30, 30, 30);*/
+ drawRect(30, 30, 30, 30);
+ gl.glEnable(GL11.GL_TEXTURE_2D);*/
+ mAnimationTime = SystemClock.uptimeMillis();
if (mContentView != null) {
mContentView.render(GLRootView.this, (GL11) gl);
}
+ long now = SystemClock.uptimeMillis();
+ for (Animation animation : mAnimations) {
+ animation.setStartTime(now);
+ }
+ mAnimations.clear();
}
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
-
- // Allocate a new event to prevent concurrency access
FutureTask<Boolean> task = new FutureTask<Boolean>(
- new TouchEventHandler(MotionEvent.obtain(event)));
+ new TouchEventHandler(event));
queueEvent(task);
try {
- return task.get(EVENT_TIMEOUT, TimeUnit.MILLISECONDS);
- } catch (TimeoutException e) {
- Log.w(TAG, "event timeout, assume the event is handled");
- return true;
+ return task.get();
} catch (Exception e) {
throw new RuntimeException(e);
}
@@ -387,6 +439,7 @@ public class GLRootView extends GLSurfaceView
}
public Boolean call() throws Exception {
+ if (mContentView == null) return false;
return mContentView.dispatchTouchEvent(mEvent);
}
}
@@ -400,19 +453,16 @@ public class GLRootView extends GLSurfaceView
return mDisplayMetrics;
}
- public Texture copyTexture2D(int x, int y, int width, int height)
+ public void copyTexture2D(
+ RawTexture texture, int x, int y, int width, int height)
throws GLOutOfMemoryException {
-
Matrix matrix = mTransformation.getMatrix();
matrix.getValues(mMatrixValues);
- // Test if it has been rotated, if it is, glDrawTexiOES won't work
- if (mMatrixValues[1] != 0 || mMatrixValues[3] != 0) {
+ if (isMatrixRotatedOrFlipped(mMatrixValues)) {
throw new IllegalArgumentException("cannot support rotated matrix");
}
-
- float points[] = new float[]{x, y + height, x + width, y};
- matrix.mapPoints(points);
+ float points[] = mapPoints(matrix, x, y + height, x + width, y);
x = (int) points[0];
y = (int) points[1];
width = (int) points[2] - x;
@@ -423,9 +473,8 @@ public class GLRootView extends GLSurfaceView
int newHeight = Util.nextPowerOf2(height);
int glError = GL11.GL_NO_ERROR;
- int[] textureId = new int[1];
- gl.glGenTextures(1, textureId, 0);
- gl.glBindTexture(GL11.GL_TEXTURE_2D, textureId[0]);
+ gl.glBindTexture(GL11.GL_TEXTURE_2D, texture.getId());
+
int[] cropRect = {0, 0, width, height};
gl.glTexParameteriv(GL11.GL_TEXTURE_2D,
GL11Ext.GL_TEXTURE_CROP_RECT_OES, cropRect, 0);
@@ -445,12 +494,13 @@ public class GLRootView extends GLSurfaceView
throw new GLOutOfMemoryException();
}
- if (glError == GL11.GL_NO_ERROR) {
- return new RawTexture(gl, textureId[0], width, height,
- (float) width / newWidth, (float) height / newHeight);
+ if (glError != GL11.GL_NO_ERROR) {
+ throw new RuntimeException(
+ "Texture copy fail, glError " + glError);
}
- throw new RuntimeException(
- "Texture copy fail, glError " + glError);
- }
+ texture.setSize(width, height);
+ texture.setTexCoordSize(
+ (float) width / newWidth, (float) height / newHeight);
+ }
}
diff --git a/src/com/android/camera/ui/GLView.java b/src/com/android/camera/ui/GLView.java
index 1df1f9e..3154ff7 100644
--- a/src/com/android/camera/ui/GLView.java
+++ b/src/com/android/camera/ui/GLView.java
@@ -4,7 +4,6 @@ import android.graphics.Matrix;
import android.graphics.Rect;
import android.view.MotionEvent;
import android.view.animation.Animation;
-import android.view.animation.AnimationUtils;
import android.view.animation.Transformation;
import java.util.ArrayList;
@@ -69,10 +68,14 @@ public class GLView {
}
public void startAnimation(Animation animation) {
+ GLRootView root = getGLRootView();
+ if (root == null) throw new IllegalStateException();
+
mAnimation = animation;
animation.initialize(getWidth(),
getHeight(), mParent.getWidth(), mParent.getHeight());
mAnimation.start();
+ root.registerLaunchedAnimation(animation);
invalidate();
}
@@ -183,15 +186,16 @@ public class GLView {
}
protected void renderChild(GLRootView root, GL11 gl, GLView component) {
- Animation anim = component.mAnimation;
int xoffset = component.mBounds.left - mScrollX;
int yoffset = component.mBounds.top - mScrollY;
Transformation transform = root.getTransformation();
Matrix matrix = transform.getMatrix();
matrix.preTranslate(xoffset, yoffset);
+
+ Animation anim = component.mAnimation;
if (anim != null) {
- long now = AnimationUtils.currentAnimationTimeMillis();
+ long now = root.currentAnimationTimeMillis();
Transformation temp = root.obtainTransformation();
temp.clear();
if (!anim.getTransformation(now, temp)) {
diff --git a/src/com/android/camera/ui/GpsIndicator.java b/src/com/android/camera/ui/GpsIndicator.java
index 27fe0a1..955b3ae 100644
--- a/src/com/android/camera/ui/GpsIndicator.java
+++ b/src/com/android/camera/ui/GpsIndicator.java
@@ -12,8 +12,8 @@ public class GpsIndicator extends BasicIndicator {
private ResourceTexture mNoSignalIcon;
private boolean mHasSignal = false;
- public GpsIndicator(IconListPreference preference) {
- super(preference);
+ public GpsIndicator(Context context, IconListPreference preference) {
+ super(context, preference);
}
@Override
diff --git a/src/com/android/camera/ui/HeadUpDisplay.java b/src/com/android/camera/ui/HeadUpDisplay.java
index c2e444c..585884a 100644
--- a/src/com/android/camera/ui/HeadUpDisplay.java
+++ b/src/com/android/camera/ui/HeadUpDisplay.java
@@ -1,5 +1,6 @@
package com.android.camera.ui;
+import static com.android.camera.ui.GLRootView.dpToPixel;
import android.content.Context;
import android.graphics.Rect;
import android.os.Handler;
@@ -23,13 +24,20 @@ import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;
public class HeadUpDisplay extends GLView {
- private static final int INDICATOR_BAR_RIGHT_MARGIN = 15;
- private static final int POPUP_WINDOW_OVERLAP = 30;
+ private static final int INDICATOR_BAR_RIGHT_MARGIN = 10;
+ private static final int POPUP_WINDOW_OVERLAP = 20;
+ private static final int POPUP_TRIANGLE_OFFSET = 16;
+
private static final float MAX_HEIGHT_RATIO = 0.8f;
private static final float MAX_WIDTH_RATIO = 0.8f;
private static final int HIDE_POPUP_WINDOW = 0;
private static final int DEACTIVATE_INDICATOR_BAR = 1;
+
+ private static int sIndicatorBarRightMargin = -1;
+ private static int sPopupWindowOverlap;
+ private static int sPopupTriangleOffset;
+
protected static final String TAG = "HeadUpDisplay";
private IndicatorBar mIndicatorBar;
@@ -48,7 +56,9 @@ public class HeadUpDisplay extends GLView {
private final Handler mHandler;
- public HeadUpDisplay() {
+ public HeadUpDisplay(Context context) {
+ initializeStaticVariables(context);
+ mTimerThread.setDaemon(true);
mTimerThread.start();
mHandler = new Handler(mTimerThread.getLooper()) {
@@ -76,6 +86,14 @@ public class HeadUpDisplay extends GLView {
};
}
+ private static void initializeStaticVariables(Context context) {
+ if (sIndicatorBarRightMargin >= 0) return;
+
+ sIndicatorBarRightMargin = dpToPixel(context, INDICATOR_BAR_RIGHT_MARGIN);
+ sPopupWindowOverlap = dpToPixel(context, POPUP_WINDOW_OVERLAP);
+ sPopupTriangleOffset = dpToPixel(context, POPUP_TRIANGLE_OFFSET);
+ }
+
private final Callable<Void> mHidePopupWindow = new Callable<Void> () {
public Void call() throws Exception {
hidePopupWindow();
@@ -206,7 +224,7 @@ public class HeadUpDisplay extends GLView {
mPopupWindow.setBackground(
new NinePatchTexture(context, R.drawable.menu_popup));
mPopupWindow.setAnchor(new ResourceTexture(
- context, R.drawable.menu_popup_triangle), 23);
+ context, R.drawable.menu_popup_triangle), sPopupTriangleOffset);
mPopupWindow.setVisibility(GLView.INVISIBLE);
mPopupWindow.setOrientation(mOrientation);
addComponent(mPopupWindow);
@@ -253,12 +271,12 @@ public class HeadUpDisplay extends GLView {
return list.toArray(new ListPreference[list.size()]);
}
- private static BasicIndicator addIndicator(
+ private static BasicIndicator addIndicator(Context context,
IndicatorBar indicatorBar, PreferenceGroup group, String key) {
IconListPreference iconPref =
(IconListPreference) group.findPreference(key);
if (iconPref == null) return null;
- BasicIndicator indicator = new BasicIndicator(iconPref);
+ BasicIndicator indicator = new BasicIndicator(context, iconPref);
indicatorBar.addComponent(indicator);
return indicator;
}
@@ -274,6 +292,7 @@ public class HeadUpDisplay extends GLView {
context, R.drawable.ic_viewfinder_iconbar_highlight));
OtherSettingsIndicator otherSettings = new OtherSettingsIndicator(
+ context,
getListPreferences(group,
CameraSettings.KEY_FOCUS_MODE,
CameraSettings.KEY_SCENE_MODE,
@@ -282,13 +301,17 @@ public class HeadUpDisplay extends GLView {
CameraSettings.KEY_COLOR_EFFECT));
mIndicatorBar.addComponent(otherSettings);
- GpsIndicator gpsIndicator = new GpsIndicator((IconListPreference)
+ GpsIndicator gpsIndicator = new GpsIndicator(
+ context, (IconListPreference)
group.findPreference(CameraSettings.KEY_RECORD_LOCATION));
+
mGpsIndicator = gpsIndicator;
mIndicatorBar.addComponent(gpsIndicator);
- addIndicator(mIndicatorBar, group, CameraSettings.KEY_WHITE_BALANCE);
- addIndicator(mIndicatorBar, group, CameraSettings.KEY_FLASH_MODE);
+ addIndicator(context, mIndicatorBar, group,
+ CameraSettings.KEY_WHITE_BALANCE);
+ addIndicator(context, mIndicatorBar, group,
+ CameraSettings.KEY_FLASH_MODE);
addComponent(mIndicatorBar);
mIndicatorBar.setOnItemSelectedListener(new IndicatorBarListener());
@@ -313,7 +336,7 @@ public class HeadUpDisplay extends GLView {
public void onItemSelected(GLView view, int position) {
Rect rect = new Rect();
getBoundsOf(view, rect);
- int anchorX = rect.left + POPUP_WINDOW_OVERLAP;
+ int anchorX = rect.left + sPopupWindowOverlap;
int anchorY = (rect.top + rect.bottom) / 2;
AbstractIndicator indicator = (AbstractIndicator) view;
diff --git a/src/com/android/camera/ui/OtherSettingsIndicator.java b/src/com/android/camera/ui/OtherSettingsIndicator.java
index 27d0d80..2833cfe 100644
--- a/src/com/android/camera/ui/OtherSettingsIndicator.java
+++ b/src/com/android/camera/ui/OtherSettingsIndicator.java
@@ -15,8 +15,10 @@ public class OtherSettingsIndicator extends AbstractIndicator {
private GLListView mPopupContent;
private final HashMap<String, String> mOverrides = new HashMap<String, String>();
- public OtherSettingsIndicator(ListPreference preference[]) {
- mPreference = preference.clone();
+ public OtherSettingsIndicator(
+ Context context, ListPreference preference[]) {
+ super(context);
+ mPreference = preference;
mAdapters = new PreferenceAdapter[preference.length];
}
@@ -56,7 +58,7 @@ public class OtherSettingsIndicator extends AbstractIndicator {
String override = mOverrides.get(prefs[i].getKey());
if (override != null) adapters[i].overrideSettings(override);
}
- return new UberAdapter(adapters);
+ return new UberAdapter();
}
@Override
@@ -78,12 +80,6 @@ public class OtherSettingsIndicator extends AbstractIndicator {
private class UberAdapter implements
GLListView.Model, GLListView.OnItemSelectedListener {
- private final PreferenceAdapter mAdapters[];
-
- public UberAdapter(PreferenceAdapter[] adapters) {
- mAdapters = adapters;
- }
-
public GLView getView(int index) {
for (PreferenceAdapter adapter : mAdapters) {
if (index < adapter.size()) {
diff --git a/src/com/android/camera/ui/PopupWindow.java b/src/com/android/camera/ui/PopupWindow.java
index 616b3de..6ebf08c 100644
--- a/src/com/android/camera/ui/PopupWindow.java
+++ b/src/com/android/camera/ui/PopupWindow.java
@@ -17,14 +17,15 @@ public class PopupWindow extends GLView {
protected int mAnchorPosition;
private final RotatePane mRotatePane = new RotatePane();
+ private RawTexture mBackupTexture;
- protected NinePatchTexture mBackground;
+ protected FrameTexture mBackground;
public PopupWindow() {
super.addComponent(mRotatePane);
}
- public void setBackground(NinePatchTexture background) {
+ public void setBackground(FrameTexture background) {
if (background == mBackground) return;
mBackground = background;
if (background != null) {
@@ -89,7 +90,7 @@ public class PopupWindow extends GLView {
}
@Override
- protected void renderBackground(GLRootView rootView, GL11 gl) {
+ protected void renderBackground(GLRootView root, GL11 gl) {
int width = getWidth();
int height = getHeight();
int aWidth = mAnchor.getWidth();
@@ -101,30 +102,32 @@ public class PopupWindow extends GLView {
aYoffset = Math.min(aYoffset, height - p.bottom - aHeight);
if (mAnchor != null) {
- if (mAnchor.bind(rootView, gl)) {
- rootView.draw2D(aXoffset, aYoffset, aWidth, aHeight);
+ if (mAnchor.bind(root, gl)) {
+ mAnchor.draw(root, aXoffset, aYoffset);
}
}
- Texture backup = null;
+ if (mBackupTexture == null || mBackupTexture.getBoundGL() != gl) {
+ mBackupTexture = RawTexture.newInstance(gl);
+ }
+
+ RawTexture backup = mBackupTexture;
try {
- backup = rootView.copyTexture2D(
- aXoffset, aYoffset, aWidth, aHeight);
+ root.copyTexture2D(backup, aXoffset, aYoffset, aWidth, aHeight);
} catch (GLOutOfMemoryException e) {
e.printStackTrace();
}
if (mBackground != null) {
mBackground.setSize(width - aWidth + mAnchorOffset, height);
- if (mBackground.bind(rootView, gl)) {
- rootView.draw2D(
- 0, 0, mBackground.getWidth(), mBackground.getHeight());
+ if (mBackground.bind(root, gl)) {
+ mBackground.draw(root, 0, 0);
}
}
- if (backup.bind(rootView, gl)) {
+ if (backup.bind(root, gl)) {
gl.glBlendFunc(GL11.GL_ONE, GL11.GL_ZERO);
- rootView.draw2D(aXoffset, aYoffset, aWidth, aHeight, 1);
+ backup.draw(root, aXoffset, aYoffset, aWidth, aHeight, 1);
gl.glBlendFunc(GL11.GL_ONE, GL11.GL_ONE_MINUS_SRC_ALPHA);
}
}
@@ -148,8 +151,8 @@ public class PopupWindow extends GLView {
set.addAnimation(scale);
set.addAnimation(alpha);
- scale.setDuration(200);
- alpha.setDuration(200);
+ scale.setDuration(150);
+ alpha.setDuration(100);
scale.setInterpolator(new OvershootInterpolator());
startAnimation(set);
}
diff --git a/src/com/android/camera/ui/PopupWindowStencilImpl.java b/src/com/android/camera/ui/PopupWindowStencilImpl.java
index eaf9134..b05b4e5 100644
--- a/src/com/android/camera/ui/PopupWindowStencilImpl.java
+++ b/src/com/android/camera/ui/PopupWindowStencilImpl.java
@@ -23,7 +23,7 @@ public class PopupWindowStencilImpl extends PopupWindow {
gl.glStencilOp(GL11.GL_KEEP, GL11.GL_KEEP, GL11.GL_REPLACE);
gl.glStencilFunc(GL11.GL_ALWAYS, 1, 1);
if (mAnchor.bind(rootView, gl)) {
- rootView.draw2D(aXoffset, aYoffset, aWidth, aHeight);
+ mAnchor.draw(rootView, aXoffset, aYoffset);
}
gl.glStencilFunc(GL11.GL_NOTEQUAL, 1, 1);
gl.glStencilOp(GL11.GL_KEEP, GL11.GL_KEEP, GL11.GL_KEEP);
@@ -32,8 +32,7 @@ public class PopupWindowStencilImpl extends PopupWindow {
if (mBackground != null) {
mBackground.setSize(width - aWidth + mAnchorOffset, height);
if (mBackground.bind(rootView, gl)) {
- rootView.draw2D(
- 0, 0, mBackground.getWidth(), mBackground.getHeight());
+ mBackground.draw(rootView, 0, 0);
}
}
}
diff --git a/src/com/android/camera/ui/PreferenceAdapter.java b/src/com/android/camera/ui/PreferenceAdapter.java
index 233f154..de3cda9 100644
--- a/src/com/android/camera/ui/PreferenceAdapter.java
+++ b/src/com/android/camera/ui/PreferenceAdapter.java
@@ -1,5 +1,6 @@
package com.android.camera.ui;
+import static com.android.camera.ui.GLRootView.dpToPixel;
import android.content.Context;
import com.android.camera.IconListPreference;
@@ -13,16 +14,26 @@ public class PreferenceAdapter
implements GLListView.Model, GLListView.OnItemSelectedListener {
private static final int ICON_NONE = 0;
+ private static final int HORIZONTAL_PADDINGS = 4;
+ private static final int VERTICAL_PADDINGS = 2;
+
+ private static int sHorizontalPaddings = -1;
+ private static int sVerticalPaddings;
- private final Context mContext;
private final ArrayList<GLView> mContent = new ArrayList<GLView>();
private final ListPreference mPreference;
private String mOverride;
+ private static void initializeStaticVariable(Context context) {
+ if (sHorizontalPaddings >= 0) return;
+ sHorizontalPaddings = dpToPixel(context, HORIZONTAL_PADDINGS);
+ sVerticalPaddings = dpToPixel(context, VERTICAL_PADDINGS);
+ }
+
public PreferenceAdapter(Context context, ListPreference preference) {
- mContext = context;
+ initializeStaticVariable(context);
mPreference = preference;
- generateContent(preference);
+ generateContent(context, preference);
}
public void overrideSettings(String settings) {
@@ -47,13 +58,12 @@ public class PreferenceAdapter
}
}
- private void generateContent(ListPreference preference) {
- Context context = mContext;
-
- GLOptionHeader header = new GLOptionHeader(preference);
+ private void generateContent(Context context, ListPreference preference) {
+ GLOptionHeader header = new GLOptionHeader(context, preference);
header.setBackground(new NinePatchTexture(
context, R.drawable.optionheader_background));
- header.setPaddings(5, 2, 5, 2);
+ header.setPaddings(sHorizontalPaddings,
+ sVerticalPaddings, sHorizontalPaddings, sVerticalPaddings);
mContent.add(header);
CharSequence[] entries = preference.getEntries();
CharSequence[] values = preference.getEntryValues();
@@ -67,7 +77,8 @@ public class PreferenceAdapter
GLOptionItem item = new GLOptionItem(
context, icons == null ? ICON_NONE : icons[i],
entries[i].toString());
- item.setPaddings(5, 2, 5, 2);
+ item.setPaddings(sHorizontalPaddings,
+ sVerticalPaddings, sHorizontalPaddings, sVerticalPaddings);
item.setChecked(values[i].equals(value));
mContent.add(item);
}
diff --git a/src/com/android/camera/ui/RawTexture.java b/src/com/android/camera/ui/RawTexture.java
index d8c9cda..aecea9d 100644
--- a/src/com/android/camera/ui/RawTexture.java
+++ b/src/com/android/camera/ui/RawTexture.java
@@ -6,11 +6,22 @@ import javax.microedition.khronos.opengles.GL11;
public class RawTexture extends Texture {
- protected RawTexture(GL11 gl, int id,
- int width, int height, float widthf, float heightf) {
+ private RawTexture(GL11 gl, int id) {
super(gl, id, STATE_LOADED);
- super.setSize(width, height);
- super.setTexCoordSize(widthf, heightf);
+ }
+
+ public GL11 getBoundGL() {
+ return mGL;
+ }
+
+ public static RawTexture newInstance(GL11 gl) {
+ int[] textureId = new int[1];
+ gl.glGenTextures(1, textureId, 0);
+ int glError = gl.glGetError();
+ if (glError != GL11.GL_NO_ERROR) {
+ throw new RuntimeException("GL_ERROR: " + glError);
+ }
+ return new RawTexture(gl, textureId[0]);
}
@Override
@@ -23,4 +34,12 @@ public class RawTexture extends Texture {
throw new UnsupportedOperationException();
}
+ @Override
+ protected boolean bind(GLRootView glRootView, GL11 gl) {
+ if (mGL == gl) {
+ gl.glBindTexture(GL11.GL_TEXTURE_2D, getId());
+ return true;
+ }
+ return false;
+ }
}
diff --git a/src/com/android/camera/ui/StateListTexture.java b/src/com/android/camera/ui/StateListTexture.java
deleted file mode 100644
index 432450c..0000000
--- a/src/com/android/camera/ui/StateListTexture.java
+++ /dev/null
@@ -1,79 +0,0 @@
-package com.android.camera.ui;
-
-import android.graphics.Bitmap;
-import android.graphics.Rect;
-
-import java.util.ArrayList;
-
-import javax.microedition.khronos.opengles.GL11;
-
-public class StateListTexture extends FrameTexture {
-
- private int mIndex;
- private final ArrayList<Entry> mEntries = new ArrayList<Entry>();
-
- private static class Entry {
- public int mMustHave;
- public int mMustNotHave;
- public FrameTexture mTexture;
-
- public Entry(int mustHave, int mustNotHave, FrameTexture texture) {
- mMustHave = mustHave;
- mMustNotHave = mustNotHave;
- mTexture = texture;
- }
- }
-
- @Override
- protected void freeBitmap(Bitmap bitmap) {
- mEntries.get(mIndex).mTexture.freeBitmap(bitmap);
- }
-
- @Override
- protected Bitmap getBitmap() {
- return mEntries.get(mIndex).mTexture.getBitmap();
- }
-
- private int getStateIndex(int state) {
- for (int i = 0, n = mEntries.size(); i < n; ++i) {
- Entry entry = mEntries.get(i);
- if ((entry.mMustHave & state) == entry.mMustHave
- && (state & entry.mMustNotHave) == 0) return i;
- }
- return -1;
- }
-
- @Override
- public void setSize(int width, int height) {
- super.setSize(width, height);
- for (Entry entry : mEntries) {
- entry.mTexture.setSize(width, height);
- }
- }
-
- public boolean isDifferent(int stateA, int stateB) {
- return getStateIndex(stateA) != getStateIndex(stateB);
- }
-
- public boolean setState(int state) {
- int oldIndex = mIndex;
- mIndex = getStateIndex(state);
- return mIndex != oldIndex;
- }
-
- @Override
- public boolean bind(GLRootView root, GL11 gl) {
- if (mIndex < 0) return false;
- return mEntries.get(mIndex).mTexture.bind(root, gl);
- }
-
- public void addState(
- int mustHave, int mustNotHave, FrameTexture texture) {
- mEntries.add(new Entry(mustHave, mustNotHave, texture));
- }
-
- @Override
- public Rect getPaddings() {
- return mEntries.get(mIndex).mTexture.getPaddings();
- }
-}
diff --git a/src/com/android/camera/ui/Texture.java b/src/com/android/camera/ui/Texture.java
index 6d5f8d8..aa116e7 100644
--- a/src/com/android/camera/ui/Texture.java
+++ b/src/com/android/camera/ui/Texture.java
@@ -19,10 +19,10 @@ public abstract class Texture {
public static final int STATE_LOADED = 1;
public static final int STATE_ERROR = -1;
- private GL11 mGL;
+ protected GL11 mGL;
- private int mId;
- private int mState;
+ protected int mId;
+ protected int mState;
protected int mWidth = UNSPECIFIED;
protected int mHeight = UNSPECIFIED;
@@ -119,31 +119,40 @@ public abstract class Texture {
}
public void draw(GLRootView root, int x, int y) {
- draw(root, x, y, getWidth(), getHeight());
+ root.drawTexture(this, x, y, mWidth, mHeight);
}
- public void draw(GLRootView root,
- int x, int y, int width, int height) {
- root.draw2D(x, y, width, height);
+ public void draw(GLRootView root, int x, int y, int w, int h, float alpha) {
+ root.drawTexture(this, x, y, w, h, alpha);
}
- protected boolean bind(GLRootView glRootView, GL11 gl) {
+ protected boolean bind(GLRootView root, GL11 gl) {
if (mState == Texture.STATE_UNLOADED || mGL != gl) {
mState = Texture.STATE_UNLOADED;
try {
uploadToGL(gl);
} catch (GLOutOfMemoryException e) {
- glRootView.handleLowMemory();
+ root.handleLowMemory();
return false;
}
} else {
gl.glBindTexture(GL11.GL_TEXTURE_2D, getId());
}
+ return true;
+ }
+ public void getTextureCoords(float coord[], int offset) {
float w = mTexCoordWidth;
float h = mTexCoordHeight;
- glRootView.setTexCoords(0, 0, w, 0, 0, h, w, h);
- return true;
+
+ coord[offset++] = 0;
+ coord[offset++] = 0;
+ coord[offset++] = w;
+ coord[offset++] = 0;
+ coord[offset++] = 0;
+ coord[offset++] = h;
+ coord[offset++] = w;
+ coord[offset] = h;
}
protected Bitmap generateGLCompatibleBitmap(int width, int height) {
diff --git a/src/com/android/camera/ui/GLZoomIndicator.java b/src/com/android/camera/ui/ZoomIndicator.java
index c6612bf..36c06a3 100644
--- a/src/com/android/camera/ui/GLZoomIndicator.java
+++ b/src/com/android/camera/ui/ZoomIndicator.java
@@ -1,16 +1,20 @@
package com.android.camera.ui;
+import android.content.Context;
import android.graphics.Color;
import android.graphics.Rect;
import javax.microedition.khronos.opengles.GL11;
-public class GLZoomIndicator extends GLView {
+public class ZoomIndicator extends GLView {
+
+ private static final int FONT_SIZE = 16;
private final StringTexture mTitle;
- public GLZoomIndicator(String title) {
- mTitle = StringTexture.newInstance(title, 24, Color.WHITE);
+ public ZoomIndicator(Context context, String title) {
+ float fontSize = GLRootView.dpToPixel(context, FONT_SIZE);
+ mTitle = StringTexture.newInstance(title, fontSize, Color.WHITE);
}
@Override