/*
* Copyright (C) 2009 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;
import static com.android.camera.Util.Assert;
import com.android.camera.gallery.IImage;
import com.android.camera.gallery.IImageList;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.drawable.Drawable;
import android.os.Handler;
import android.util.AttributeSet;
import android.view.GestureDetector;
import android.view.KeyEvent;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewConfiguration;
import android.view.GestureDetector.SimpleOnGestureListener;
import android.widget.Scroller;
import java.util.HashMap;
class GridViewSpecial extends View {
@SuppressWarnings("unused")
private static final String TAG = "GridViewSpecial";
private static final float MAX_FLING_VELOCITY = 2500;
public static interface Listener {
public void onImageClicked(int index);
public void onImageTapped(int index);
public void onLayoutComplete(boolean changed);
/**
* Invoked when the GridViewSpecial
scrolls.
*
* @param scrollPosition the position of the scroller in the range
* [0, 1], when 0 means on the top and 1 means on the buttom
*/
public void onScroll(float scrollPosition);
}
public static interface DrawAdapter {
public void drawImage(Canvas canvas, IImage image,
Bitmap b, int xPos, int yPos, int w, int h);
public void drawDecoration(Canvas canvas, IImage image,
int xPos, int yPos, int w, int h);
public boolean needsDecoration();
}
public static final int INDEX_NONE = -1;
// There are two cell size we will use. It can be set by setSizeChoice().
// The mLeftEdgePadding fields is filled in onLayout(). See the comments
// in onLayout() for details.
static class LayoutSpec {
LayoutSpec(int w, int h, int intercellSpacing, int leftEdgePadding) {
mCellWidth = w;
mCellHeight = h;
mCellSpacing = intercellSpacing;
mLeftEdgePadding = leftEdgePadding;
}
int mCellWidth, mCellHeight;
int mCellSpacing;
int mLeftEdgePadding;
}
private final LayoutSpec [] mCellSizeChoices = new LayoutSpec[] {
new LayoutSpec(67, 67, 8, 0),
new LayoutSpec(92, 92, 8, 0),
};
// These are set in init().
private final Handler mHandler = new Handler();
private GestureDetector mGestureDetector;
private ImageBlockManager mImageBlockManager;
// These are set in set*() functions.
private ImageLoader mLoader;
private Listener mListener = null;
private DrawAdapter mDrawAdapter = null;
private IImageList mAllImages = ImageManager.emptyImageList();
private int mSizeChoice = 1; // default is big cell size
// These are set in onLayout().
private LayoutSpec mSpec;
private int mColumns;
private int mMaxScrollY;
// We can handle events only if onLayout() is completed.
private boolean mLayoutComplete = false;
// Selection state
private int mCurrentSelection = INDEX_NONE;
private int mCurrentPressState = 0;
private static final int TAPPING_FLAG = 1;
private static final int CLICKING_FLAG = 2;
// These are cached derived information.
private int mCount; // Cache mImageList.getCount();
private int mRows; // Cache (mCount + mColumns - 1) / mColumns
private int mBlockHeight; // Cache mSpec.mCellSpacing + mSpec.mCellHeight
private boolean mRunning = false;
private Scroller mScroller = null;
public GridViewSpecial(Context context, AttributeSet attrs) {
super(context, attrs);
init(context);
}
private void init(Context context) {
setVerticalScrollBarEnabled(true);
initializeScrollbars(context.obtainStyledAttributes(
android.R.styleable.View));
mGestureDetector = new GestureDetector(context,
new MyGestureDetector());
setFocusableInTouchMode(true);
}
private final Runnable mRedrawCallback = new Runnable() {
public void run() {
invalidate();
}
};
public void setLoader(ImageLoader loader) {
Assert(mRunning == false);
mLoader = loader;
}
public void setListener(Listener listener) {
Assert(mRunning == false);
mListener = listener;
}
public void setDrawAdapter(DrawAdapter adapter) {
Assert(mRunning == false);
mDrawAdapter = adapter;
}
public void setImageList(IImageList list) {
Assert(mRunning == false);
mAllImages = list;
mCount = mAllImages.getCount();
}
public void setSizeChoice(int choice) {
Assert(mRunning == false);
if (mSizeChoice == choice) return;
mSizeChoice = choice;
}
@Override
public void onLayout(boolean changed, int left, int top,
int right, int bottom) {
super.onLayout(changed, left, top, right, bottom);
if (!mRunning) {
return;
}
mSpec = mCellSizeChoices[mSizeChoice];
int width = right - left;
// The width is divided into following parts:
//
// LeftEdgePadding CellWidth (CellSpacing CellWidth)* RightEdgePadding
//
// We determine number of cells (columns) first, then the left and right
// padding are derived. We make left and right paddings the same size.
//
// The height is divided into following parts:
//
// CellSpacing (CellHeight CellSpacing)+
mColumns = 1 + (width - mSpec.mCellWidth)
/ (mSpec.mCellWidth + mSpec.mCellSpacing);
mSpec.mLeftEdgePadding = (width
- ((mColumns - 1) * mSpec.mCellSpacing)
- (mColumns * mSpec.mCellWidth)) / 2;
mRows = (mCount + mColumns - 1) / mColumns;
mBlockHeight = mSpec.mCellSpacing + mSpec.mCellHeight;
mMaxScrollY = mSpec.mCellSpacing + (mRows * mBlockHeight)
- (bottom - top);
// Put mScrollY in the valid range. This matters if mMaxScrollY is
// changed. For example, orientation changed from portrait to landscape.
mScrollY = Math.max(0, Math.min(mMaxScrollY, mScrollY));
generateOutlineBitmap();
if (mImageBlockManager != null) {
mImageBlockManager.recycle();
}
mImageBlockManager = new ImageBlockManager(mHandler, mRedrawCallback,
mAllImages, mLoader, mDrawAdapter, mSpec, mColumns, width,
mOutline[OUTLINE_EMPTY]);
mListener.onLayoutComplete(changed);
moveDataWindow();
mLayoutComplete = true;
}
@Override
protected int computeVerticalScrollRange() {
return mMaxScrollY + getHeight();
}
// We cache the three outlines from NinePatch to Bitmap to speed up
// drawing. The cache must be updated if the cell size is changed.
public static final int OUTLINE_EMPTY = 0;
public static final int OUTLINE_PRESSED = 1;
public static final int OUTLINE_SELECTED = 2;
public Bitmap mOutline[] = new Bitmap[3];
private void generateOutlineBitmap() {
int w = mSpec.mCellWidth;
int h = mSpec.mCellHeight;
for (int i = 0; i < mOutline.length; i++) {
mOutline[i] = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888);
}
Drawable cellOutline;
cellOutline = GridViewSpecial.this.getResources()
.getDrawable(android.R.drawable.gallery_thumb);
cellOutline.setBounds(0, 0, w, h);
Canvas canvas = new Canvas();
canvas.setBitmap(mOutline[OUTLINE_EMPTY]);
cellOutline.setState(EMPTY_STATE_SET);
cellOutline.draw(canvas);
canvas.setBitmap(mOutline[OUTLINE_PRESSED]);
cellOutline.setState(
PRESSED_ENABLED_FOCUSED_SELECTED_WINDOW_FOCUSED_STATE_SET);
cellOutline.draw(canvas);
canvas.setBitmap(mOutline[OUTLINE_SELECTED]);
cellOutline.setState(ENABLED_FOCUSED_SELECTED_WINDOW_FOCUSED_STATE_SET);
cellOutline.draw(canvas);
}
private void moveDataWindow() {
// Calculate visible region according to scroll position.
int startRow = (mScrollY - mSpec.mCellSpacing) / mBlockHeight;
int endRow = (mScrollY + getHeight() - mSpec.mCellSpacing - 1)
/ mBlockHeight + 1;
// Limit startRow and endRow to the valid range.
// Make sure we handle the mRows == 0 case right.
startRow = Math.max(Math.min(startRow, mRows - 1), 0);
endRow = Math.max(Math.min(endRow, mRows), 0);
mImageBlockManager.setVisibleRows(startRow, endRow);
}
// In MyGestureDetector we have to check canHandleEvent() because
// GestureDetector could queue events and fire them later. At that time
// stop() may have already been called and we can't handle the events.
private class MyGestureDetector extends SimpleOnGestureListener {
@Override
public boolean onDown(MotionEvent e) {
if (!canHandleEvent()) return false;
if (mScroller != null && !mScroller.isFinished()) {
mScroller.forceFinished(true);
return false;
}
int index = computeSelectedIndex(e.getX(), e.getY());
if (index >= 0 && index < mCount) {
setSelectedIndex(index);
} else {
setSelectedIndex(INDEX_NONE);
}
return true;
}
@Override
public boolean onFling(MotionEvent e1, MotionEvent e2,
float velocityX, float velocityY) {
if (!canHandleEvent()) return false;
if (velocityY > MAX_FLING_VELOCITY) {
velocityY = MAX_FLING_VELOCITY;
} else if (velocityY < -MAX_FLING_VELOCITY) {
velocityY = -MAX_FLING_VELOCITY;
}
setSelectedIndex(INDEX_NONE);
mScroller = new Scroller(getContext());
mScroller.fling(0, mScrollY, 0, -(int) velocityY, 0, 0, 0,
mMaxScrollY);
computeScroll();
return true;
}
@Override
public void onLongPress(MotionEvent e) {
if (!canHandleEvent()) return;
performLongClick();
}
@Override
public boolean onScroll(MotionEvent e1, MotionEvent e2,
float distanceX, float distanceY) {
if (!canHandleEvent()) return false;
setSelectedIndex(INDEX_NONE);
scrollBy(0, (int) distanceY);
invalidate();
return true;
}
@Override
public boolean onSingleTapConfirmed(MotionEvent e) {
if (!canHandleEvent()) return false;
int index = computeSelectedIndex(e.getX(), e.getY());
if (index >= 0 && index < mCount) {
mListener.onImageTapped(index);
return true;
}
return false;
}
}
public int getCurrentSelection() {
return mCurrentSelection;
}
public void invalidateImage(int index) {
if (index != INDEX_NONE) {
mImageBlockManager.invalidateImage(index);
}
}
/**
*
* @param index INDEX_NONE
(-1) means remove selection.
*/
public void setSelectedIndex(int index) {
// A selection box will be shown for the image that being selected,
// (by finger or by the dpad center key). The selection box can be drawn
// in two colors. One color (yellow) is used when the the image is
// still being tapped or clicked (the finger is still on the touch screen
// or the dpad center key is not released). Another color (orange) is
// used after the finger leaves touch screen or the dpad center
// key is released.
if (mCurrentSelection == index) {
return;
}
// This happens when the last picture is deleted.
mCurrentSelection = Math.min(index, mCount - 1);
if (mCurrentSelection != INDEX_NONE) {
ensureVisible(mCurrentSelection);
}
invalidate();
}
public void scrollToImage(int index) {
Rect r = getRectForPosition(index);
scrollTo(0, r.top);
}
public void scrollToVisible(int index) {
Rect r = getRectForPosition(index);
int top = getScrollY();
int bottom = getScrollY() + getHeight();
if (r.bottom > bottom) {
scrollTo(0, r.bottom - getHeight());
} else if (r.top < top) {
scrollTo(0, r.top);
}
}
private void ensureVisible(int pos) {
Rect r = getRectForPosition(pos);
int top = getScrollY();
int bot = top + getHeight();
if (r.bottom > bot) {
mScroller = new Scroller(getContext());
mScroller.startScroll(mScrollX, mScrollY, 0,
r.bottom - getHeight() - mScrollY, 200);
computeScroll();
} else if (r.top < top) {
mScroller = new Scroller(getContext());
mScroller.startScroll(mScrollX, mScrollY, 0, r.top - mScrollY, 200);
computeScroll();
}
}
public void start() {
// These must be set before start().
Assert(mLoader != null);
Assert(mListener != null);
Assert(mDrawAdapter != null);
mRunning = true;
requestLayout();
}
// If the the underlying data is changed, for example,
// an image is deleted, or the size choice is changed,
// The following sequence is needed:
//
// mGvs.stop();
// mGvs.set...(...);
// mGvs.set...(...);
// mGvs.start();
public void stop() {
// Remove the long press callback from the queue if we are going to
// stop.
mHandler.removeCallbacks(mLongPressCallback);
mScroller = null;
if (mImageBlockManager != null) {
mImageBlockManager.recycle();
mImageBlockManager = null;
}
mRunning = false;
mCurrentSelection = INDEX_NONE;
}
@Override
public void onDraw(Canvas canvas) {
super.onDraw(canvas);
if (!canHandleEvent()) return;
mImageBlockManager.doDraw(canvas, getWidth(), getHeight(), mScrollY);
paintDecoration(canvas);
paintSelection(canvas);
moveDataWindow();
}
@Override
public void computeScroll() {
if (mScroller != null) {
boolean more = mScroller.computeScrollOffset();
scrollTo(0, mScroller.getCurrY());
if (more) {
invalidate(); // So we draw again
} else {
mScroller = null;
}
} else {
super.computeScroll();
}
}
// Return the rectange for the thumbnail in the given position.
Rect getRectForPosition(int pos) {
int row = pos / mColumns;
int col = pos - (row * mColumns);
int left = mSpec.mLeftEdgePadding
+ (col * (mSpec.mCellWidth + mSpec.mCellSpacing));
int top = row * mBlockHeight;
return new Rect(left, top,
left + mSpec.mCellWidth + mSpec.mCellSpacing,
top + mSpec.mCellHeight + mSpec.mCellSpacing);
}
// Inverse of getRectForPosition: from screen coordinate to image position.
int computeSelectedIndex(float xFloat, float yFloat) {
int x = (int) xFloat;
int y = (int) yFloat;
int spacing = mSpec.mCellSpacing;
int leftSpacing = mSpec.mLeftEdgePadding;
int row = (mScrollY + y - spacing) / (mSpec.mCellHeight + spacing);
int col = Math.min(mColumns - 1,
(x - leftSpacing) / (mSpec.mCellWidth + spacing));
return (row * mColumns) + col;
}
@Override
public boolean onTouchEvent(MotionEvent ev) {
if (!canHandleEvent()) {
return false;
}
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:
mCurrentPressState |= TAPPING_FLAG;
invalidate();
break;
case MotionEvent.ACTION_UP:
mCurrentPressState &= ~TAPPING_FLAG;
invalidate();
break;
}
mGestureDetector.onTouchEvent(ev);
// Consume all events
return true;
}
@Override
public void scrollBy(int x, int y) {
scrollTo(mScrollX + x, mScrollY + y);
}
public void scrollTo(float scrollPosition) {
scrollTo(0, Math.round(scrollPosition * mMaxScrollY));
}
@Override
public void scrollTo(int x, int y) {
y = Math.max(0, Math.min(mMaxScrollY, y));
if (mSpec != null) {
mListener.onScroll((float) mScrollY / mMaxScrollY);
}
super.scrollTo(x, y);
}
private boolean canHandleEvent() {
return mRunning && mLayoutComplete;
}
private final Runnable mLongPressCallback = new Runnable() {
public void run() {
mCurrentPressState &= ~CLICKING_FLAG;
showContextMenu();
}
};
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
if (!canHandleEvent()) return false;
int sel = mCurrentSelection;
if (sel != INDEX_NONE) {
switch (keyCode) {
case KeyEvent.KEYCODE_DPAD_RIGHT:
if (sel != mCount - 1 && (sel % mColumns < mColumns - 1)) {
sel += 1;
}
break;
case KeyEvent.KEYCODE_DPAD_LEFT:
if (sel > 0 && (sel % mColumns != 0)) {
sel -= 1;
}
break;
case KeyEvent.KEYCODE_DPAD_UP:
if (sel >= mColumns) {
sel -= mColumns;
}
break;
case KeyEvent.KEYCODE_DPAD_DOWN:
sel = Math.min(mCount - 1, sel + mColumns);
break;
case KeyEvent.KEYCODE_DPAD_CENTER:
mCurrentPressState |= CLICKING_FLAG;
mHandler.postDelayed(mLongPressCallback,
ViewConfiguration.getLongPressTimeout());
break;
default:
return super.onKeyDown(keyCode, event);
}
} else {
switch (keyCode) {
case KeyEvent.KEYCODE_DPAD_RIGHT:
case KeyEvent.KEYCODE_DPAD_LEFT:
case KeyEvent.KEYCODE_DPAD_UP:
case KeyEvent.KEYCODE_DPAD_DOWN:
int startRow =
(mScrollY - mSpec.mCellSpacing) / mBlockHeight;
int topPos = startRow * mColumns;
Rect r = getRectForPosition(topPos);
if (r.top < getScrollY()) {
topPos += mColumns;
}
topPos = Math.min(mCount - 1, topPos);
sel = topPos;
break;
default:
return super.onKeyDown(keyCode, event);
}
}
setSelectedIndex(sel);
return true;
}
@Override
public boolean onKeyUp(int keyCode, KeyEvent event) {
if (!canHandleEvent()) return false;
if (keyCode == KeyEvent.KEYCODE_DPAD_CENTER) {
mCurrentPressState &= ~CLICKING_FLAG;
invalidate();
// The keyUp doesn't get called when the longpress menu comes up. We
// only get here when the user lets go of the center key before the
// longpress menu comes up.
mHandler.removeCallbacks(mLongPressCallback);
// open the photo
mListener.onImageClicked(mCurrentSelection);
return true;
}
return super.onKeyUp(keyCode, event);
}
private void paintDecoration(Canvas canvas) {
if (!mDrawAdapter.needsDecoration()) return;
// Calculate visible region according to scroll position.
int startRow = (mScrollY - mSpec.mCellSpacing) / mBlockHeight;
int endRow = (mScrollY + getHeight() - mSpec.mCellSpacing - 1)
/ mBlockHeight + 1;
// Limit startRow and endRow to the valid range.
// Make sure we handle the mRows == 0 case right.
startRow = Math.max(Math.min(startRow, mRows - 1), 0);
endRow = Math.max(Math.min(endRow, mRows), 0);
int startIndex = startRow * mColumns;
int endIndex = Math.min(endRow * mColumns, mCount);
int xPos = mSpec.mLeftEdgePadding;
int yPos = mSpec.mCellSpacing + startRow * mBlockHeight;
int off = 0;
for (int i = startIndex; i < endIndex; i++) {
IImage image = mAllImages.getImageAt(i);
mDrawAdapter.drawDecoration(canvas, image, xPos, yPos,
mSpec.mCellWidth, mSpec.mCellHeight);
// Calculate next position
off += 1;
if (off == mColumns) {
xPos = mSpec.mLeftEdgePadding;
yPos += mBlockHeight;
off = 0;
} else {
xPos += mSpec.mCellWidth + mSpec.mCellSpacing;
}
}
}
private void paintSelection(Canvas canvas) {
if (mCurrentSelection == INDEX_NONE) return;
int row = mCurrentSelection / mColumns;
int col = mCurrentSelection - (row * mColumns);
int spacing = mSpec.mCellSpacing;
int leftSpacing = mSpec.mLeftEdgePadding;
int xPos = leftSpacing + (col * (mSpec.mCellWidth + spacing));
int yTop = spacing + (row * mBlockHeight);
int type = OUTLINE_SELECTED;
if (mCurrentPressState != 0) {
type = OUTLINE_PRESSED;
}
canvas.drawBitmap(mOutline[type], xPos, yTop, null);
}
}
class ImageBlockManager {
@SuppressWarnings("unused")
private static final String TAG = "ImageBlockManager";
// Number of rows we want to cache.
// Assume there are 6 rows per page, this caches 5 pages.
private static final int CACHE_ROWS = 30;
// mCache maps from row number to the ImageBlock.
private final HashMap mCache;
// These are parameters set in the constructor.
private final Handler mHandler;
private final Runnable mRedrawCallback; // Called after a row is loaded,
// so GridViewSpecial can draw
// again using the new images.
private final IImageList mImageList;
private final ImageLoader mLoader;
private final GridViewSpecial.DrawAdapter mDrawAdapter;
private final GridViewSpecial.LayoutSpec mSpec;
private final int mColumns; // Columns per row.
private final int mBlockWidth; // The width of an ImageBlock.
private final Bitmap mOutline; // The outline bitmap put on top of each
// image.
private final int mCount; // Cache mImageList.getCount().
private final int mRows; // Cache (mCount + mColumns - 1) / mColumns
private final int mBlockHeight; // The height of an ImageBlock.
// Visible row range: [mStartRow, mEndRow). Set by setVisibleRows().
private int mStartRow = 0;
private int mEndRow = 0;
ImageBlockManager(Handler handler, Runnable redrawCallback,
IImageList imageList, ImageLoader loader,
GridViewSpecial.DrawAdapter adapter,
GridViewSpecial.LayoutSpec spec,
int columns, int blockWidth, Bitmap outline) {
mHandler = handler;
mRedrawCallback = redrawCallback;
mImageList = imageList;
mLoader = loader;
mDrawAdapter = adapter;
mSpec = spec;
mColumns = columns;
mBlockWidth = blockWidth;
mOutline = outline;
mBlockHeight = mSpec.mCellSpacing + mSpec.mCellHeight;
mCount = imageList.getCount();
mRows = (mCount + mColumns - 1) / mColumns;
mCache = new HashMap();
mPendingRequest = 0;
initGraphics();
}
// Set the window of visible rows. Once set we will start to load them as
// soon as possible (if they are not already in cache).
public void setVisibleRows(int startRow, int endRow) {
if (startRow != mStartRow || endRow != mEndRow) {
mStartRow = startRow;
mEndRow = endRow;
startLoading();
}
}
int mPendingRequest; // Number of pending requests (sent to ImageLoader).
// We want to keep enough requests in ImageLoader's queue, but not too
// many.
static final int REQUESTS_LOW = 3;
static final int REQUESTS_HIGH = 6;
// After clear requests currently in queue, start loading the thumbnails.
// We need to clear the queue first because the proper order of loading
// may have changed (because the visible region changed, or some images
// have been invalidated).
private void startLoading() {
clearLoaderQueue();
continueLoading();
}
private void clearLoaderQueue() {
int[] tags = mLoader.clearQueue();
for (int pos : tags) {
int row = pos / mColumns;
int col = pos - row * mColumns;
ImageBlock blk = mCache.get(row);
Assert(blk != null); // We won't reuse the block if it has pending
// requests. See getEmptyBlock().
blk.cancelRequest(col);
}
}
// Scan the cache and send requests to ImageLoader if needed.
private void continueLoading() {
// Check if we still have enough requests in the queue.
if (mPendingRequest >= REQUESTS_LOW) return;
// Scan the visible rows.
for (int i = mStartRow; i < mEndRow; i++) {
if (scanOne(i)) return;
}
int range = (CACHE_ROWS - (mEndRow - mStartRow)) / 2;
// Scan other rows.
// d is the distance between the row and visible region.
for (int d = 1; d <= range; d++) {
int after = mEndRow - 1 + d;
int before = mStartRow - d;
if (after >= mRows && before < 0) {
break; // Nothing more the scan.
}
if (after < mRows && scanOne(after)) return;
if (before >= 0 && scanOne(before)) return;
}
}
// Returns true if we can stop scanning.
private boolean scanOne(int i) {
mPendingRequest += tryToLoad(i);
return mPendingRequest >= REQUESTS_HIGH;
}
// Returns number of requests we issued for this row.
private int tryToLoad(int row) {
Assert(row >= 0 && row < mRows);
ImageBlock blk = mCache.get(row);
if (blk == null) {
// Find an empty block
blk = getEmptyBlock();
blk.setRow(row);
blk.invalidate();
mCache.put(row, blk);
}
return blk.loadImages();
}
// Get an empty block for the cache.
private ImageBlock getEmptyBlock() {
// See if we can allocate a new block.
if (mCache.size() < CACHE_ROWS) {
return new ImageBlock();
}
// Reclaim the old block with largest distance from the visible region.
int bestDistance = -1;
int bestIndex = -1;
for (int index : mCache.keySet()) {
// Make sure we don't reclaim a block which still has pending
// request.
if (mCache.get(index).hasPendingRequests()) {
continue;
}
int dist = 0;
if (index >= mEndRow) {
dist = index - mEndRow + 1;
} else if (index < mStartRow) {
dist = mStartRow - index;
} else {
// Inside the visible region.
continue;
}
if (dist > bestDistance) {
bestDistance = dist;
bestIndex = index;
}
}
return mCache.remove(bestIndex);
}
public void invalidateImage(int index) {
int row = index / mColumns;
int col = index - (row * mColumns);
ImageBlock blk = mCache.get(row);
if (blk == null) return;
if ((blk.mCompletedMask & (1 << col)) != 0) {
blk.mCompletedMask &= ~(1 << col);
}
startLoading();
}
// After calling recycle(), the instance should not be used anymore.
public void recycle() {
for (ImageBlock blk : mCache.values()) {
blk.recycle();
}
mCache.clear();
mEmptyBitmap.recycle();
}
// Draw the images to the given canvas.
public void doDraw(Canvas canvas, int thisWidth, int thisHeight,
int scrollPos) {
final int height = mBlockHeight;
// Note that currentBlock could be negative.
int currentBlock = (scrollPos < 0)
? ((scrollPos - height + 1) / height)
: (scrollPos / height);
while (true) {
final int yPos = currentBlock * height;
if (yPos >= scrollPos + thisHeight) {
break;
}
ImageBlock blk = mCache.get(currentBlock);
if (blk != null) {
blk.doDraw(canvas, 0, yPos);
} else {
drawEmptyBlock(canvas, 0, yPos, currentBlock);
}
currentBlock += 1;
}
}
// Return number of columns in the given row. (This could be less than
// mColumns for the last row).
private int numColumns(int row) {
return Math.min(mColumns, mCount - row * mColumns);
}
// Draw a block which has not been loaded.
private void drawEmptyBlock(Canvas canvas, int xPos, int yPos, int row) {
// Draw the background.
canvas.drawRect(xPos, yPos, xPos + mBlockWidth, yPos + mBlockHeight,
mBackgroundPaint);
// Draw the empty images.
int x = xPos + mSpec.mLeftEdgePadding;
int y = yPos + mSpec.mCellSpacing;
int cols = numColumns(row);
for (int i = 0; i < cols; i++) {
canvas.drawBitmap(mEmptyBitmap, x, y, null);
x += (mSpec.mCellWidth + mSpec.mCellSpacing);
}
}
// mEmptyBitmap is what we draw if we the wanted block hasn't been loaded.
// (If the user scrolls too fast). It is a gray image with normal outline.
// mBackgroundPaint is used to draw the (black) background outside
// mEmptyBitmap.
Paint mBackgroundPaint;
private Bitmap mEmptyBitmap;
private void initGraphics() {
mBackgroundPaint = new Paint();
mBackgroundPaint.setStyle(Paint.Style.FILL);
mBackgroundPaint.setColor(0xFF000000); // black
mEmptyBitmap = Bitmap.createBitmap(mSpec.mCellWidth, mSpec.mCellHeight,
Bitmap.Config.RGB_565);
Canvas canvas = new Canvas(mEmptyBitmap);
canvas.drawRGB(0xDD, 0xDD, 0xDD);
canvas.drawBitmap(mOutline, 0, 0, null);
}
// ImageBlock stores bitmap for one row. The loaded thumbnail images are
// drawn to mBitmap. mBitmap is later used in onDraw() of GridViewSpecial.
private class ImageBlock {
private Bitmap mBitmap;
private final Canvas mCanvas;
// Columns which have been requested to the loader
private int mRequestedMask;
// Columns which have been completed from the loader
private int mCompletedMask;
// The row number this block represents.
private int mRow;
public ImageBlock() {
mBitmap = Bitmap.createBitmap(mBlockWidth, mBlockHeight,
Bitmap.Config.RGB_565);
mCanvas = new Canvas(mBitmap);
mRow = -1;
}
public void setRow(int row) {
mRow = row;
}
public void invalidate() {
// We do not change mRequestedMask or do cancelAllRequests()
// because the data coming from pending requests are valid. (We only
// invalidate data which has been drawn to the bitmap).
mCompletedMask = 0;
}
// After recycle, the ImageBlock instance should not be accessed.
public void recycle() {
cancelAllRequests();
mBitmap.recycle();
mBitmap = null;
}
private boolean isVisible() {
return mRow >= mStartRow && mRow < mEndRow;
}
// Returns number of requests submitted to ImageLoader.
public int loadImages() {
Assert(mRow != -1);
int columns = numColumns(mRow);
// Calculate what we need.
int needMask = ((1 << columns) - 1)
& ~(mCompletedMask | mRequestedMask);
if (needMask == 0) {
return 0;
}
int retVal = 0;
int base = mRow * mColumns;
for (int col = 0; col < columns; col++) {
if ((needMask & (1 << col)) == 0) {
continue;
}
int pos = base + col;
final IImage image = mImageList.getImageAt(pos);
if (image != null) {
// This callback is passed to ImageLoader. It will invoke
// loadImageDone() in the main thread. We limit the callback
// thread to be in this very short function. All other
// processing is done in the main thread.
final int colFinal = col;
ImageLoader.LoadedCallback cb =
new ImageLoader.LoadedCallback() {
public void run(final Bitmap b) {
mHandler.post(new Runnable() {
public void run() {
loadImageDone(image, b,
colFinal);
}
});
}
};
// Load Image
mLoader.getBitmap(image, cb, pos);
mRequestedMask |= (1 << col);
retVal += 1;
}
}
return retVal;
}
// Whether this block has pending requests.
public boolean hasPendingRequests() {
return mRequestedMask != 0;
}
// Called when an image is loaded.
private void loadImageDone(IImage image, Bitmap b,
int col) {
if (mBitmap == null) return; // This block has been recycled.
int spacing = mSpec.mCellSpacing;
int leftSpacing = mSpec.mLeftEdgePadding;
final int yPos = spacing;
final int xPos = leftSpacing
+ (col * (mSpec.mCellWidth + spacing));
drawBitmap(image, b, xPos, yPos);
if (b != null) {
b.recycle();
}
int mask = (1 << col);
Assert((mCompletedMask & mask) == 0);
Assert((mRequestedMask & mask) != 0);
mRequestedMask &= ~mask;
mCompletedMask |= mask;
mPendingRequest--;
if (isVisible()) {
mRedrawCallback.run();
}
// Kick start next block loading.
continueLoading();
}
// Draw the loaded bitmap to the block bitmap.
private void drawBitmap(
IImage image, Bitmap b, int xPos, int yPos) {
mDrawAdapter.drawImage(mCanvas, image, b, xPos, yPos,
mSpec.mCellWidth, mSpec.mCellHeight);
mCanvas.drawBitmap(mOutline, xPos, yPos, null);
}
// Draw the block bitmap to the specified canvas.
public void doDraw(Canvas canvas, int xPos, int yPos) {
int cols = numColumns(mRow);
if (cols == mColumns) {
canvas.drawBitmap(mBitmap, xPos, yPos, null);
} else {
// This must be the last row -- we draw only part of the block.
// Draw the background.
canvas.drawRect(xPos, yPos, xPos + mBlockWidth,
yPos + mBlockHeight, mBackgroundPaint);
// Draw part of the block.
int w = mSpec.mLeftEdgePadding
+ cols * (mSpec.mCellWidth + mSpec.mCellSpacing);
Rect srcRect = new Rect(0, 0, w, mBlockHeight);
Rect dstRect = new Rect(srcRect);
dstRect.offset(xPos, yPos);
canvas.drawBitmap(mBitmap, srcRect, dstRect, null);
}
// Draw the part which has not been loaded.
int isEmpty = ((1 << cols) - 1) & ~mCompletedMask;
if (isEmpty != 0) {
int x = xPos + mSpec.mLeftEdgePadding;
int y = yPos + mSpec.mCellSpacing;
for (int i = 0; i < cols; i++) {
if ((isEmpty & (1 << i)) != 0) {
canvas.drawBitmap(mEmptyBitmap, x, y, null);
}
x += (mSpec.mCellWidth + mSpec.mCellSpacing);
}
}
}
// Mark a request as cancelled. The request has already been removed
// from the queue of ImageLoader, so we only need to mark the fact.
public void cancelRequest(int col) {
int mask = (1 << col);
Assert((mRequestedMask & mask) != 0);
mRequestedMask &= ~mask;
mPendingRequest--;
}
// Try to cancel all pending requests for this block. After this
// completes there could still be requests not cancelled (because it is
// already in progress). We deal with that situation by setting mBitmap
// to null in recycle() and check this in loadImageDone().
private void cancelAllRequests() {
for (int i = 0; i < mColumns; i++) {
int mask = (1 << i);
if ((mRequestedMask & mask) != 0) {
int pos = (mRow * mColumns) + i;
if (mLoader.cancel(mImageList.getImageAt(pos))) {
mRequestedMask &= ~mask;
mPendingRequest--;
}
}
}
}
}
}