android自定义壁纸制作,Android 自定义View实现画背景和前景(ViewGroup篇)
在定义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篇)相关推荐
- Android 自定义View实现画背景和前景(ViewGroup篇)
2019独角兽企业重金招聘Python工程师标准>>> 在定义ListView的Selector时候,有个drawSelectorOnTop的属性,如果drawSelectorOnT ...
- android 设置壁纸页面,Android实现手机壁纸改变的方法
本文实例讲述了Android实现手机壁纸改变的方法.分享给大家供大家参考.具体如下: main.xml布局文件: android:orientation="vertical" an ...
- android动态壁纸引擎,Android动态壁纸开发
新建一个Android项目: 完成下面步骤后,安装到手机,并设置动态壁纸才能看到效果. res/xml/cube1.xml AndroidManifest.xml CubeWallpaper1.jav ...
- android 设置壁纸,在Android中使用WallpaperManager设置壁纸
以下是我的代码,我想使用壁纸管理器设置为壁纸.我正在使用Universal Image Loader,但我不知道如何实现壁纸管理器.我的setWall()不起作用,有点令人困惑. import and ...
- android 手机壁纸制作教程,教程:让你的手机桌面瞬间高逼格!
原标题:教程:让你的手机桌面瞬间高逼格! 先上这张壁纸缅怀一下我的五儿子,陪伴了我两年.正是因为Nexus 5,才正式开启了我的搞机不归路.我个人比较喜欢简洁风格,手机桌面也力求最精简.如果你也喜欢这 ...
- android动态图制作,Android 教程:如何在手机上制作高质量的 GIF 图片
相比于静态图片的一动不动,GIF 图片显得十分生动活泼,并且能表现一定的情节.虽然已经有 GIF 快手.美图 GIF 这类易用的 GIF 制作应用,但其有着诸多的限制,像时长.分辨率等都不可自行调节. ...
- mac 壁纸 android,可以用于任何设备的Android 12壁纸现在已可下载
本文来自cnBeta 现在就可以为iPhone.iPad.Android.Mac.Windows PC等下载全新的Android 12壁纸.Android 12的第一个开发者预览版预计将在本月推出,完 ...
- android壁纸和手机屏幕之间要怎么对应,android手机壁纸
安卓手机怎样获取当前壁纸? 手机设置锁屏壁纸方法: 方式1:待机界面-长按屏幕弹出主屏界面-壁纸-左上角选择"锁定屏幕"-从相册或内置墙纸选择图片即可. 方式2:设定-(显示/显示 ...
- Android开发之制作圆形头像自定义View,直接引用工具类,加快开发速度。带有源代码学习
作者:程序员小冰,CSDN博客:http://blog.csdn.net/qq_21376985 QQ986945193 博客园主页:http://www.cnblogs.com/mcxiaobing ...
最新文章
- 深度学习100例-卷积神经网络(CNN)花朵识别 | 第4天
- kibana安装与Kibana server is not ready yet
- C++工作笔记- C++中的动态类型与动态绑定、虚函数、运行时多态的实现
- Linux环境下使用NLPIR(ICTCLAS)中文分词详解
- 想玩转工业界机器学习?先学Spark吧
- 自动化安装DHCP配置脚本
- mp4剪辑器_想学视频剪辑,可是专业的视频软件太难,来试试这软件吧!
- [转]IE首页被http://www.9798.net/篡改解决办法
- linux(ubuntu) 搭建java程序运行环境
- JDBC的下载和安装教程
- 开源API查询IP地址归属信息
- rss订阅,全文阅读,渴望大家发贴的时候在RSS中总是显示全文
- 区块链大咖谈之陈昌:联盟链和公有链混合架构如何实现?
- MF,PMF算法比较
- 计算机软件cd全称,cd刻录(刻录高音质cd完美教程_计算机软件及应用_IT/计算机_专业资料)...
- vue 生命周期(详解)
- Android 开发技术干货
- 大数据学习笔记(十)-Hive中的Storage format
- 街景字符识别2-图像读取及图像增广
- php后端mvc框架,GitHub - Tokyo-Lei/Amaya: 史上最简单的PHP MVC框架!首先你了解MVC和COMPOSER就行!...
热门文章
- 实战篇-OpenSSL之TripleDES加密算法-ECB模式
- Dreamweaver CS6实战手册
- 服务器如何几十台电脑一起装系统,多台电脑如何同时安装系统
- OCiOS开发:使用相册、照相机和录像
- (已更新)Discuz手机模板:NVBING5-APP手机版,界面美观大方,可封装安卓/苹果APP,模板文件+插件+分类信息导入文件
- Vue 之 echarts 图表数据可视化的基础使用(简单绘制各种图表、地图)
- php三种流程结构,3.流程控制结构
- R语言 CART算法和C4.5算法(决策树)
- mysql 统计不同成绩阶段的人数
- 【最小栈c++】设计一个支持 push,pop,top 操作,并能在常数时间内检索到最小元素的栈