在定义ListView的Selector时候,有个drawSelectorOnTop的属性,如果drawSelectorOnTop为true的话,Selector的效果是画在List Item的上面(Selector是盖住了ListView的文字或者图片),即Foreground前景。如果drawSelectorOnTop为false的话,Selector的效果是画在List Item的下面,即Background背景。由于项目中恰好需要自定义View,需要实现此效果。

本文借ListView的代码来剖析一下,

ListView完成此部分功能在frameworks\base\core\java\android\widget\AbsListView.java文件中。

用mSelector即ListView要画的Selector(资源文件),而mSelectorRect则是想要画的区域。

/**

* Indicates whether the list selector should be drawn on top of the children or behind

*/

boolean mDrawSelectorOnTop = false; 决定画前景还是背景

/**

* The drawable used to draw the selector

*/

Drawable mSelector; ListView用中来显示Selector的Drawable,即ListSelector对应的XML文件

/**

* The current position of the selector in the list.

*/

int mSelectorPosition = INVALID_POSITION;

/**

* Defines the selector's location and dimension at drawing time

*/

Rect mSelectorRect = new Rect(); 用来画Selector的区域,即Selector画的位置

AbsListView中构造方法中有获取selector

Drawable d = a.getDrawable(com.android.internal.R.styleable.AbsListView_listSelector);

if (d != null) {

setSelector(d);

}

//默认为false,画的是背景

mDrawSelectorOnTop = a.getBoolean(

com.android.internal.R.styleable.AbsListView_drawSelectorOnTop, false); 下面看一下setSelector是如何实现的

/**

* Controls whether the selection highlight drawable should be drawn on top of the item or

* behind it.

*

* @param onTop If true, the selector will be drawn on the item it is highlighting. The default

* is false.

*

* @attr ref android.R.styleable#AbsListView_drawSelectorOnTop

*/

public void setDrawSelectorOnTop(boolean onTop) { //提供是否画前景或者背景的接口

mDrawSelectorOnTop = onTop;

}

/**

* Set a Drawable that should be used to highlight the currently selected item.

*

* @param resID A Drawable resource to use as the selection highlight.

*

* @attr ref android.R.styleable#AbsListView_listSelector

*/

public void setSelector(int resID) {

setSelector(getResources().getDrawable(resID)); 设置listSelector的XML文件

}

public void setSelector(Drawable sel) {

if (mSelector != null) {

mSelector.setCallback(null);

unscheduleDrawable(mSelector);

}

mSelector = sel;

Rect padding = new Rect();

sel.getPadding(padding);

mSelectionLeftPadding = padding.left;

mSelectionTopPadding = padding.top;

mSelectionRightPadding = padding.right;

mSelectionBottomPadding = padding.bottom;

sel.setCallback(this); //需要给Selector设置Callback

updateSelectorState();

}

/**

* Returns the selector {@link android.graphics.drawable.Drawable} that is used to draw the

* selection in the list.

*

* @return the drawable used to display the selector

*/

public Drawable getSelector() {

return mSelector;

}

void updateSelectorState() {

if (mSelector != null) {

if (shouldShowSelector()) {

mSelector.setState(getDrawableState());//更新Selector的状态

} else {

mSelector.setState(StateSet.NOTHING);

}

}

}

这样就将Selector设置给ListView了,并且更新了drawable的状态。

接下来我们再看一下Android是如何将drawable画到ListView的Item上的。

在AbsListView中有个onTouchEvent的方法用来处理Touch事件,其中有一段代码就是确定Selector要画的区域。

if (mTouchMode == TOUCH_MODE_DOWN || mTouchMode == TOUCH_MODE_TAP) {

final Handler handler = getHandler();

if (handler != null) {

handler.removeCallbacks(mTouchMode == TOUCH_MODE_DOWN ?

mPendingCheckForTap : mPendingCheckForLongPress);

}

mLayoutMode = LAYOUT_NORMAL;

if (!mDataChanged && mAdapter.isEnabled(motionPosition)) {

mTouchMode = TOUCH_MODE_TAP;

setSelectedPositionInt(mMotionPosition);

layoutChildren();

child.setPressed(true);//设置List Item状态为 pressed

positionSelector(mMotionPosition, child);//确定画Selector的区域

setPressed(true); //设置ListView 的状态为pressed

if (mSelector != null) {

Drawable d = mSelector.getCurrent();

if (d != null && d instanceof TransitionDrawable) {

((TransitionDrawable) d).resetTransition();

}

}

if (mTouchModeReset != null) {

removeCallbacks(mTouchModeReset);

}

mTouchModeReset = new Runnable() {

@Override

public void run() {

mTouchMode = TOUCH_MODE_REST;

child.setPressed(false);

setPressed(false);

if (!mDataChanged) {

performClick.run();

}

}

};

postDelayed(mTouchModeReset,

ViewConfiguration.getPressedStateDuration());

} else {

mTouchMode = TOUCH_MODE_REST;

updateSelectorState();

}

return true;

} else if (!mDataChanged && mAdapter.isEnabled(motionPosition)) {

performClick.run();

}

}

接下来看看positionSelector的实现,

void positionSelector(int position, View sel) {

if (position != INVALID_POSITION) {

mSelectorPosition = position;

}

//设置Selector的区域为List Item View的边界

final Rect selectorRect = mSelectorRect;  selectorRect.set(sel.getLeft(), sel.getTop(), sel.getRight(), sel.getBottom());

if (sel instanceof SelectionBoundsAdjuster) {

((SelectionBoundsAdjuster)sel).adjustListItemSelectionBounds(selectorRect);

}

positionSelector(selectorRect.left, selectorRect.top, selectorRect.right,

selectorRect.bottom);

final boolean isChildViewEnabled = mIsChildViewEnabled;

if (sel.isEnabled() != isChildViewEnabled) {

mIsChildViewEnabled = !isChildViewEnabled;

if (getSelectedItemPosition() != INVALID_POSITION) {

refreshDrawableState();//根据View状态更新drawable的状态

}

}

}

private void positionSelector(int l, int t, int r, int b) {

mSelectorRect.set(l - mSelectionLeftPadding, t - mSelectionTopPadding, r

+ mSelectionRightPadding, b + mSelectionBottomPadding);

}

好了现在已经决定了将selector画在哪里,Selector的状态也已经更新OK。

还差一步没有做,那就是到底是将其怎么画上面的呢?

答案就在AbsListView.java里的dispatchDraw方法里面。

@Override

protected void dispatchDraw(Canvas canvas) {

int saveCount = 0;

final boolean clipToPadding = (mGroupFlags & CLIP_TO_PADDING_MASK) == CLIP_TO_PADDING_MASK;

if (clipToPadding) {

saveCount = canvas.save();

final int scrollX = mScrollX;

final int scrollY = mScrollY;

canvas.clipRect(scrollX + mPaddingLeft, scrollY + mPaddingTop,

scrollX + mRight - mLeft - mPaddingRight,

scrollY + mBottom - mTop - mPaddingBottom);

mGroupFlags &= ~CLIP_TO_PADDING_MASK;

}

final boolean drawSelectorOnTop = mDrawSelectorOnTop;

if (!drawSelectorOnTop) { //将Selector画为背景

drawSelector(canvas);

}

super.dispatchDraw(canvas);// 用Canvas画ListView

if (drawSelectorOnTop) { //将Selector画为前景

drawSelector(canvas);

}

if (clipToPadding) {

canvas.restoreToCount(saveCount);

mGroupFlags |= CLIP_TO_PADDING_MASK;

}

}

private void drawSelector(Canvas canvas) {

if (!mSelectorRect.isEmpty()) {

final Drawable selector = mSelector;

selector.setBounds(mSelectorRect);//设置drawable画的区域

selector.draw(canvas); //使用canvas将drawable画上去

}

}

看到这里,想必大家都已经明白如何画前景和背景了吧。在dispatchDraw之前调用就是画前景,在dispatchDraw之后调用就是画背景。

另外补充一下,本文并没有介绍动画部分,有兴趣的可以自己研究下。

总结一下,实现这个功能需要有三个步骤:

1.设置Selector,并更新状态(初始化时候)

2.确定Selector画的区域,设置View的状态,根据View状态,更新Selector的状态(一般是对Event的处理方法中)

3.使用Canvas在dispatchDraw中,将Selector画上去,画Drawable的时候需要先设置区域,再调用drawable的draw方法。

后面我再将View如何画背景和前景补上,今天就先到这里吧。

android自定义壁纸制作,Android 自定义View实现画背景和前景(ViewGroup篇)相关推荐

  1. Android 自定义View实现画背景和前景(ViewGroup篇)

    2019独角兽企业重金招聘Python工程师标准>>> 在定义ListView的Selector时候,有个drawSelectorOnTop的属性,如果drawSelectorOnT ...

  2. android 设置壁纸页面,Android实现手机壁纸改变的方法

    本文实例讲述了Android实现手机壁纸改变的方法.分享给大家供大家参考.具体如下: main.xml布局文件: android:orientation="vertical" an ...

  3. android动态壁纸引擎,Android动态壁纸开发

    新建一个Android项目: 完成下面步骤后,安装到手机,并设置动态壁纸才能看到效果. res/xml/cube1.xml AndroidManifest.xml CubeWallpaper1.jav ...

  4. android 设置壁纸,在Android中使用WallpaperManager设置壁纸

    以下是我的代码,我想使用壁纸管理器设置为壁纸.我正在使用Universal Image Loader,但我不知道如何实现壁纸管理器.我的setWall()不起作用,有点令人困惑. import and ...

  5. android 手机壁纸制作教程,教程:让你的手机桌面瞬间高逼格!

    原标题:教程:让你的手机桌面瞬间高逼格! 先上这张壁纸缅怀一下我的五儿子,陪伴了我两年.正是因为Nexus 5,才正式开启了我的搞机不归路.我个人比较喜欢简洁风格,手机桌面也力求最精简.如果你也喜欢这 ...

  6. android动态图制作,Android 教程:如何在手机上制作高质量的 GIF 图片

    相比于静态图片的一动不动,GIF 图片显得十分生动活泼,并且能表现一定的情节.虽然已经有 GIF 快手.美图 GIF 这类易用的 GIF 制作应用,但其有着诸多的限制,像时长.分辨率等都不可自行调节. ...

  7. mac 壁纸 android,可以用于任何设备的Android 12壁纸现在已可下载

    本文来自cnBeta 现在就可以为iPhone.iPad.Android.Mac.Windows PC等下载全新的Android 12壁纸.Android 12的第一个开发者预览版预计将在本月推出,完 ...

  8. android壁纸和手机屏幕之间要怎么对应,android手机壁纸

    安卓手机怎样获取当前壁纸? 手机设置锁屏壁纸方法: 方式1:待机界面-长按屏幕弹出主屏界面-壁纸-左上角选择"锁定屏幕"-从相册或内置墙纸选择图片即可. 方式2:设定-(显示/显示 ...

  9. Android开发之制作圆形头像自定义View,直接引用工具类,加快开发速度。带有源代码学习

    作者:程序员小冰,CSDN博客:http://blog.csdn.net/qq_21376985 QQ986945193 博客园主页:http://www.cnblogs.com/mcxiaobing ...

最新文章

  1. 深度学习100例-卷积神经网络(CNN)花朵识别 | 第4天
  2. kibana安装与Kibana server is not ready yet
  3. C++工作笔记- C++中的动态类型与动态绑定、虚函数、运行时多态的实现
  4. Linux环境下使用NLPIR(ICTCLAS)中文分词详解
  5. 想玩转工业界机器学习?先学Spark吧
  6. 自动化安装DHCP配置脚本
  7. mp4剪辑器_想学视频剪辑,可是专业的视频软件太难,来试试这软件吧!
  8. [转]IE首页被http://www.9798.net/篡改解决办法
  9. linux(ubuntu) 搭建java程序运行环境
  10. JDBC的下载和安装教程
  11. 开源API查询IP地址归属信息
  12. rss订阅,全文阅读,渴望大家发贴的时候在RSS中总是显示全文
  13. 区块链大咖谈之陈昌:联盟链和公有链混合架构如何实现?
  14. MF,PMF算法比较
  15. 计算机软件cd全称,cd刻录(刻录高音质cd完美教程_计算机软件及应用_IT/计算机_专业资料)...
  16. vue 生命周期(详解)
  17. Android 开发技术干货
  18. 大数据学习笔记(十)-Hive中的Storage format
  19. 街景字符识别2-图像读取及图像增广
  20. php后端mvc框架,GitHub - Tokyo-Lei/Amaya: 史上最简单的PHP MVC框架!首先你了解MVC和COMPOSER就行!...

热门文章

  1. 实战篇-OpenSSL之TripleDES加密算法-ECB模式
  2. Dreamweaver CS6实战手册
  3. 服务器如何几十台电脑一起装系统,多台电脑如何同时安装系统
  4. OCiOS开发:使用相册、照相机和录像
  5. (已更新)Discuz手机模板:NVBING5-APP手机版,界面美观大方,可封装安卓/苹果APP,模板文件+插件+分类信息导入文件
  6. Vue 之 echarts 图表数据可视化的基础使用(简单绘制各种图表、地图)
  7. php三种流程结构,3.流程控制结构
  8. R语言 CART算法和C4.5算法(决策树)
  9. mysql 统计不同成绩阶段的人数
  10. 【最小栈c++】设计一个支持 push,pop,top 操作,并能在常数时间内检索到最小元素的栈