我们知道,进入百度图片后,输入一个关键字后,首先看到的是很多缩略图,当我们点击某张缩略图时,我们就可以进入到大图显示页面,在
大图显示页面,中包含了一个图片画廊,同时当前大图为刚刚我们点击的那张图片。现在我们看看在Android中如何实现类似的效果:
首先,我们需要有一个控件来显示缩略图,这里没有什么比GridView更加合适了。
配置文件如下:
对于GridView中每一项是一张缩略图,我们需要继承BaseAdapter,实现自己的一个GridImageAdapter,代码:
然后,我们就可以在Activity中通过查询MediaStore的多媒体图片库来查询所有的图片的缩略图,缩略图所在的位置是:
MediaStore.Images.Thumbnails。Activity代码如下:
注意到,我们记录了,所有缩略图对应的id号,和当前的用户选择的位置,然后通过Intent传递到第二个展示界面。第二个界面的布局文件如下:我们用了一个Gallery和一个ImageButton来实现
然后,对应的Activity如下:
可以看到,当用户点击Gallery中某一项时,触发onItemClick事件,在其中,我们通过根据该缩略图对应的Image_ID来从MediaStore.Images.Media中查询该缩略图对应的大图。并在ImageButton中显示。
这里当图片很多时,可能会出现内存溢出,为了避免这种情况,可以更加Gallery的特点,使用缓存。保存当前可见的缩略图的前三个到后三个。其余的全部recycle。当用户点击Gallery的时候,在判断当前的位置,如果大于或小于某个值时,则重新更新缓存。这样保证内存中的缩略图的个数总是6+Gallery.getLastVisiblePosition-Gallery.getFirstVisiblePosition个。其实这就是浮动缓存窗口,一个固定大小窗口在整个坐标(全部缩略图)上游动。这里没有实现,以后待续。
同时,你可能已经注意到,程序中使用到了一个BitmapUtils类,这个类是封装了一系列对查询图片,并将其解析为Bitmap的类。
代码如下:
这样就实现了,类似百度图片浏览的效果。
前几天使用一款android手机测试的时候,
发现了应用的 shortcut 九宫格页面有一个点击效果,
就是当点击一个应用的icon图标的时候,会在icon的周围有荧光效果,
无论icon的形状是什么样子的都会有这样的效果,然后又想到Apidemo里面有个alphaDrawable例子
大家可以去在回顾一下,之后我就想到了会不会是使用这个extractAlpha实现的,自己就动手写了个例子
发现效果确实不错,分享给大家
主要关键点
1、设置imageview的src drawable
2、从src drawable中抽取 bitmap的alpha(只有透明度没有颜色)
3、使用bitmap的alpha填充一种颜色后生产新的bitmap
4、使用新生成的bitmap做一个statelistdrawable作为imageview的backgroud
5、这需要注意的是要跟imageview设置几个像素的padding这样才不会让src和backgroud重合
主要的代码:
/* * 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.study; import android.content.Context; import android.content.res.TypedArray; import android.graphics.Bitmap; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.graphics.Bitmap.Config; import android.graphics.drawable.BitmapDrawable; import android.graphics.drawable.StateListDrawable; import android.util.AttributeSet; import android.view.View; import android.view.ViewGroup; import android.widget.ImageView; /** * A layout that arranges its children in a grid. The size of the * cells is set by the {@link #setCellSize} method and the * android:cell_width and android:cell_height attributes in XML. * The number of rows and columns is determined at runtime. Each * cell contains exactly one view, and they flow in the natural * child order (the order in which they were added, or the index * in {@link #addViewAt}. Views can not span multiple cells. */ public class FixedGridLayout extends ViewGroup { int mCellWidth; int mCellHeight; public FixedGridLayout(Context context) { super(context); } public FixedGridLayout(Context context, AttributeSet attrs) { super(context, attrs); // Read the resource attributes. TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.FixedGridLayout); mCellWidth = a.getDimensionPixelSize(R.styleable.FixedGridLayout_cellWidth, -1); mCellHeight = a.getDimensionPixelSize(R.styleable.FixedGridLayout_cellHeight, -1); a.recycle(); } public void setCellWidth(int px) { mCellWidth = px; requestLayout(); } public void setCellHeight(int px) { mCellHeight = px; requestLayout(); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { int cellWidthSpec = MeasureSpec.makeMeasureSpec(mCellWidth, MeasureSpec.AT_MOST); int cellHeightSpec = MeasureSpec.makeMeasureSpec(mCellHeight, MeasureSpec.AT_MOST); int count = getChildCount(); for (int index=0; index<count; index++) { final View child = getChildAt(index); child.measure(cellWidthSpec, cellHeightSpec); } // Use the size our parents gave us setMeasuredDimension(resolveSize(mCellWidth*count, widthMeasureSpec), resolveSize(mCellHeight*count, heightMeasureSpec)); } @Override protected void onLayout(boolean changed, int l, int t, int r, int b) { int cellWidth = mCellWidth; int cellHeight = mCellHeight; int columns = (r - l) / cellWidth; if (columns < 0) { columns = 1; } int x = 0; int y = 0; int i = 0; int count = getChildCount(); for (int index=0; index<count; index++) { final View child = getChildAt(index); int w = child.getMeasuredWidth(); int h = child.getMeasuredHeight(); int left = x + ((cellWidth-w)/2); int top = y + ((cellHeight-h)/2); child.layout(left, top, left+w, top+h); if (i >= (columns-1)) { // advance to next row i = 0; x = 0; y += cellHeight; } else { i++; x += cellWidth; } } } @Override protected void onFinishInflate() { super.onFinishInflate(); int cnt = getChildCount(); Paint p = new Paint(); p.setColor(Color.CYAN); for (int i=0; i<cnt; i++) { View v = getChildAt(i); setStateDrawable((ImageView)v, p); } } private void setStateDrawable(ImageView v, Paint p) { BitmapDrawable bd = (BitmapDrawable) v.getDrawable(); Bitmap b = bd.getBitmap(); Bitmap bitmap = Bitmap.createBitmap(bd.getIntrinsicWidth(), bd.getIntrinsicHeight(), Config.ARGB_8888); Canvas canvas = new Canvas(bitmap); canvas.drawBitmap(b.extractAlpha(), 0, 0, p); StateListDrawable sld = new StateListDrawable(); sld.addState(new int[]{android.R.attr.state_pressed}, new BitmapDrawable(bitmap)); v.setBackgroundDrawable(sld); } }
具体见附件。
为ImageView或者LinearLayout画图片阴影
http://407827531.iteye.com/blog/1194984
图片阴影效果和影子效果
http://www.eoeandroid.com/thread-90575-1-2.html
1. 中文乱码问题
网页和js文件都需要按相同的编码格式保存,否则因为编码不一致会导致中文乱码。建议文件全部使用utf-8编码。
在网页<head>标签内加入:
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
引入js文件时:
<script type="text/javascript" charset="utf-8"></script>
2. 页面载入速度过慢问题
某些情况下,导致页面载入速度变慢的原因是因为在<head>中载入了大量的js脚本文件。这些脚本文件的载入会拖慢页面的载入速度。
可以把这些脚本的载入代码放置到</body>前,让html代码优先载入完。这样至少“表面上”看起来页面的载入速度似乎变快了。
3. jQuery载入太过耗时问题
jQuery的设计目标是能够兼容多种浏览器,因此代码量会比较大。对于移动开发来说,jQuery中大量代码是无用,且浪费内存和性能。果断放弃之。
替代品: XUI, jQuery-Mobile