Android群英传读书笔记(第六章)

上一章

书中的示例代码:github

本章主要介绍的是Android的绘图技巧

1.屏幕尺寸的信息:

  • 屏幕参数有屏幕大小即屏幕对角线的长度,用“寸”来表示、分辨率、PPI即每英寸像素,它由对角线的像素点除以屏幕大小获得。
  • 系统屏幕密度如下
    ldpi---120---240X320分辨率
    mdpi---160---320X480分辨率
    hdpi---240---480X800分辨率
    xhdpi---320---720X1280分辨率
    xxhdpi---480---1080X1920分辨率

2.单位转换工具类(这种百度一堆 在此省略)

3.2D绘图基?。?br> Paint类的一些属性和对应的功能:

  • setAntiAlias();//设置画笔的锯齿效果
  • setColor();//设置画笔的颜色
  • setARGB();//设置画笔的A,R,G,B的值
  • setAlpha();//设置画笔的Alpha值
  • setTextSize();//设置字体的尺寸
  • setStyle();//设置画笔的风格(空心或者实心)
  • setStrokeWidth();//设置空心边框的宽度
    Canvas类主要的绘画功能:
  • canvas.drawPoint(x,y,paint);//绘制点
  • canvas.drawLine(startX,startY,endX,endY,paint);//绘制直线
  • canvas.drawRect(left,top,right,bottom,paint);//绘制矩形
  • canvas.drawRoundRect(left,top,right,bottom,radiusX,radiusY,paint);//绘制圆角矩形
  • canvas.drawCircle(circleX,circleY,radius,paint);//绘制圆
  • canvas.drawOval(left,top,right,bottom,paint);//通过椭圆的外接矩形来绘制椭圆
  • canvas.drawText(text,startX,startY,paint);//绘制文字
  • canvas.drawPosText(text,new float[]{x1,y1,...,xn,yn},paint);//指定位置绘制文本

//绘制多条直线
float[] pts={
startX1,startY1,endX1,endY1
......
startXn,startYn,endXn,endYn}
canvas.drawLines(pts,paint);
//绘制路径
Path path=new Path();
path.moveTo(50,50);
path.lineTo(100,100);
path.lineTo(300,50);
canvas.drawPath(path,paint)
//绘制弧形或扇形
canvas.drawArc(left,top,right,bottom,startAngle,sweepAngle,useCenter,paint);
这里useCenter传入true则绘制扇形 反之绘制弧形


4.Android的Xml绘图:
* Bitmap:

<bitmap xmlns:android="http://schemas.android.com/apk/res/android"
android:src="@drawable/ic_launcher"/>

这样就能直接将图片转成bitmap在程序中使用了
* Shape:

<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="line|oval|ring|rectangle">

<corners
android:bottomLeftRadius="integer"
android:bottomRightRadius="integer"
android:radius="integer"
android:topLeftRadius="integer"
android:topRightRadius="integer" />

<gradient
    android:angle="integer"
    android:centerColor="color"
    android:centerX="integer"
    android:centerY="integer"
    android:endColor="color"
    android:gradientRadius="integer"
    android:startColor="color"
    android:type="linear|radial|sweep"
    android:useLevel="boolean" />

<padding
    android:bottom="integer"
    android:left="integer"
    android:right="integer"
    android:top="integer" />

<size
    android:width="integer"
    android:height="integer" />
<!--指定大小,一般用在imageview配合scaleType使用-->

<solid android:color="color" />
<!--填充颜色-->
<stroke
    android:width="integer"
    android:color="color"
    android:dashGap="integer"
    android:dashWidth="integer" />
<!--边框,dashGap为虚线间隔宽度,dashWidth为虚线宽度-->

</shape>

* Layer:大家可以自己试试效果  这里就不贴图了

<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
<item android:drawable="@mipmap/ic_launcher" />
<item
android:drawable="@mipmap/ic_launcher"
android:left="10dp"
android:top="10dp" />
<item
android:drawable="@mipmap/ic_launcher"
android:left="20dp"
android:top="20dp" />
</layer-list>

* Selector:
通常用于view的触摸反馈

<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_pressed="true">
<shape android:shape="rectangle">
<solid android:color="#334444" />
</shape>
</item>
<item android:state_pressed="false">
<shape android:shape="rectangle">
<solid android:color="#444444" />
</shape>
</item>
</selector>


5.Android绘图技巧:
`canvas.save()`:保存之前在画布上的操作,之后的操作就像是在新的图层操作一样。
`canvas.restore()`:可以理解为Photoshop中合并图层的操作,作用是将save后的图层与之前的图层合并。
`canvas.translate()`:将画布进行平移。
`canvas.rotate()`:将画布进行旋转。这两个方法都是用来简化绘图而创建的。
`canvas.saveLayer()`:创建一个图层,并将他加入一个图层的栈,通过调用restore或者restoreToCount使其出栈,出栈后会把图像绘制到上层canvas上。
`canvas.saveLayerAlpha()`:同上,区别是可以设置图层的透明度

6.Android色彩特效处理:
图像的色调、饱和度、亮度这三个属性在图像处理中使用非常之多,Android封装了一个`ColorMatrix`类(颜色矩阵),通过改变矩阵值来处理这些属性。
* 色调:

//第一个参数分别为red、green、blue 第二个参数为要处理的值。
ColorMatrix hueMatrix=new ColorMatrix();
hueMatrix.setRotate(0, hue);
hueMatrix.setRotate(1, hue);
hueMatrix.setRotate(2, hue);

* 饱和度:

//参数即为饱和度,当饱和度为0时图像就变成灰图像了。
ColorMatrix saturationMatrix=new ColorMatrix();
saturationMatrix.setSaturation(saturation);

* 亮度:

//当三原色以相同比例进行混合的时候,就会显示出白色,从而调节亮度。
ColorMatrix lumMatrix=new ColorMatrix();
lumMatrix.setScale(lum, lum, lum, 1);

* 让三种效果叠加:

ColorMatrix imageMatrix=new ColorMatrix();
imageMatrix.postConcat(hueMatrix);
imageMatrix.postConcat(saturationMatrix);
imageMatrix.postConcat(lumMatrix);


7.Android图形特效处理:
通过Matrix类来操作图形矩阵的变换

private float[] mImageMatrix=new float[9];
Matrix matrix=new Matrix();
matrix.setValues(mImageMatrix);
//当获得一个变换矩阵以后通过以下代码将一个图像以这个变换矩阵的形式绘制出来
canvas.drawBitmap(mBitmap,matrix,null);

以下是一些变换的方法:
* `setScale();` 缩放变换
* `setSkew();` 错切变换
* `setRotate();` 旋转变换
* `setTranslate();` 平移变换
* `pre()和post()` 提供矩阵的前乘和后乘运算

图形特效这块由于开发用的比较少  暂时记到这里

8.Android画笔特效:
* `PorterDuffXfermode`:
[经典示意图](http://img.blog.csdn.net/20130828212947609?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdDEyeDM0NTY=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast)(贴图不方便偷个懒)
这里要注意的是,PorterDuffXfermode设置的是两个图层交集区域的显示方式,dst是先画的图形,而src是后画的图形。比较常用的是DST_IN和SRC_IN模式来实现一个圆角图形。

private Bitmap getRoundPic() {
//原图
Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.head1);
//创建一个和原图一样宽高的图片
Bitmap out = Bitmap.createBitmap(bitmap.getWidth(), bitmap.getHeight(), Bitmap.Config.ARGB_8888);
//构造画布
Canvas canvas = new Canvas(out);
//初始化画笔
Paint paint = new Paint();
paint.setAntiAlias(true);
//绘制形状
canvas.drawCircle(bitmap.getWidth() / 2, bitmap.getHeight() / 2, Math.min(bitmap.getWidth() / 2, bitmap.getHeight() / 2), paint);
//设置paint的xfermode
paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));
//把原图绘制上去
canvas.drawBitmap(bitmap, 0, 0, paint);
//返回新创建的bitmap,即圆形图标。
return out;
}

* `Shader`:
Shader也被称之为着色器、渲染器,它用来实现一些列的渐变、渲染的效果。
  * `BitmapShader` --- 位图Shader
  * `LinearGradient` --- 线性Shader
  * `RadialGradient` --- 光束Shader
  * `SweepGradient` --- 梯度Shader
  * `ComposeShader` --- 混合Shader
除第一个Shader以外 其他Shader都实现了名副其实的渐变。BitmapShader产生的是一个位图,它的作用就是通过Paint对画布进行指定Bitmap的填充,填充有以下几种模式可以选择:
  * `CLAMP`拉伸 --- 拉伸的是图片最后一个像素 不断重复
  * `REPEAT`重复 --- 横向纵向不断重复
  * `MIRROR`镜像 --- 横向不断旋转重复,纵向不断旋转重复
这里推荐大家看下这篇[博客](http://blog.csdn.net/aigestudio/article/details/41799811)写的非常详细。

9.`SurfaceView`:
与view的区别:
* View主要适用于主动更新的情况下,而SurfaceView 主要适用于被动更新,例如频繁的刷新。
* View在主线程中对画面进行刷新,而SurfaceView通?;嵬ü桓鲎酉叱汤唇幸趁嫠⑿?。
* View在绘图时没有使用双缓冲机制,而SurfaceView在底层实现机制中就已经实现了双缓冲。
以下是SurfaceView的模板代码:

public class MySurfaceView extends SurfaceView implements Runnable, SurfaceHolder.Callback {
//SurfaceHolder
private SurfaceHolder mSurfaceHolder;
//用于绘图的Canvas
private Canvas mCanvas;
//子线程标志位
private boolean mIsDrawing;

public MySurfaceView(Context context) {
    super(context);
    init();
}
public MySurfaceView(Context context, AttributeSet attrs) {
    super(context, attrs);
    init();
}
public MySurfaceView(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    init();
}
private void init() {
    mSurfaceHolder = getHolder();
    mSurfaceHolder.addCallback(this);
    setFocusable(true);
    setFocusableInTouchMode(true);
    this.setKeepScreenOn(true);

// mSurfaceHolder.setFormat(PixelFormat.OPAQUE);
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
mIsDrawing = true;
new Thread(this).start();
}
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
mIsDrawing = false;
}
@Override
public void run() {
while (mIsDrawing) {
draw();
}
}
private void draw() {
try {
//每次获得的canvas对象都是上次的 因此上次的绘画操作都会保存
mCanvas = mSurfaceHolder.lockCanvas();
//draw here
} catch (Exception e) {

    } finally {
        if (mCanvas != null) {
            mSurfaceHolder.unlockCanvasAndPost(mCanvas);
        }
    }

}

}

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351

推荐阅读更多精彩内容