自定义控件(11)---Canvas的平移、旋转、缩放、错切、Matrix直接变换Canvas

本文主要是介绍自定义控件(11)---Canvas的平移、旋转、缩放、错切、Matrix直接变换Canvas,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Canvas中的变换操作,说起变换,无非就几种:平移、旋转、缩放和错切,而我们的Canvas也继承了变换的精髓,同样提供了这几种相应的方法,前面的很多章节我们也都用到了,像translate(float dx, float dy)方法平移画布用了无数次,这里再次强调,translate方法会改变画布的原点坐标,原点坐标对变换的影响弥足轻重,前面也多次强调了!scale(float sx, float sy)缩放也很好理解,但是它有一个重载方法scale(float sx, float sy, float px, float py),后两个参数用于指定缩放的中心点,前两个参数用于指定横纵向的缩放比率值在0-1之间为缩小:

scale画布

package com.example.customview;import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.util.AttributeSet;
import android.view.View;public class CustomeView extends View {private Bitmap mBitmap;// 位图对象private int mViewWidth, mViewHeight;// 控件宽高public CustomeView(Context context, AttributeSet attrs) {super(context, attrs);// 从资源中获取位图对象mBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.z);}@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {/** 获取控件宽高*/mViewWidth = w;mViewHeight = h;// 缩放位图与控件一致mBitmap = Bitmap.createScaledBitmap(mBitmap, mViewWidth, mViewHeight, true);}@Overrideprotected void onDraw(Canvas canvas) {//或者可以直接canvas.save() canvas.save(Canvas.MATRIX_SAVE_FLAG);canvas.scale(0.8F, 0.35F);//缩放中心在左上角canvas.drawBitmap(mBitmap, 0, 0, null);canvas.restore();}
}

public void scale (float sx, float sy) 
public final void scale (float sx, float sy, float px, float py)

float sx:水平方向伸缩的比例,sx为小数为缩小,sx为整数为放大
float sy:垂直方向伸缩的比例,同样,小数为缩小,整数为放大

注意:这里有X、Y轴的密度的改变,显示到图形上就会正好相同,比如X轴缩小,那么显示的图形也会缩小。一样的。

protected void onDraw(Canvas canvas) {  // TODO Auto-generated method stub  super.onDraw(canvas);  //  //scale 缩放坐标系密度  Paint paint_green = generatePaint(Color.GREEN, Style.STROKE, 5);  Paint paint_red   = generatePaint(Color.RED, Style.STROKE, 5);  Rect rect1 = new Rect(10,10,200,100);  canvas.drawRect(rect1, paint_green);  canvas.scale(0.5f, 1);  canvas.drawRect(rect1, paint_red);  
} 


-----------------基线------------------------------------------------------------------------------------------

translate画布

protected void onDraw(Canvas canvas) {  // TODO Auto-generated method stub  super.onDraw(canvas);  //构造两个画笔,一个红色,一个绿色  Paint paint_green = generatePaint(Color.GREEN, Style.STROKE, 3);  Paint paint_red   = generatePaint(Color.RED, Style.STROKE, 3);  //构造一个矩形  Rect rect1 = new Rect(0,0,400,220);  //在平移画布前用绿色画下边框  canvas.drawRect(rect1, paint_green);  //平移画布后,再用红色边框重新画下这个矩形  canvas.translate(100, 100);  canvas.drawRect(rect1, paint_red);  }  
private Paint generatePaint(int color,Paint.Style style,int width)  
{  Paint paint = new Paint();  paint.setColor(color);  paint.setStyle(style);  paint.setStrokeWidth(width);  return paint;  
}  

1、调用canvas.drawRect(rect1, paint_green);时,产生一个Canvas透明图层,由于当时还没有对坐标系平移,所以坐标原点是(0,0);再在系统在Canvas上画好之后,覆盖到屏幕上显示出来

2、然后再第二次调用canvas.drawRect(rect1, paint_red);时,又会重新产生一个全新的Canvas画布,但此时画布坐标已经改变了,即向右和向下分别移动了100像素,所以此时的绘图方式为:(合成视图,从上往下看的合成方式)


Rotate画布

画布的旋转是默认是围绕坐标原点来旋转的,这里容易产生错觉,看起来觉得是图片旋转了,其实我们旋转的是画布,以后在此画布上画的东西显示出来的时候全部看起来都是旋转的。其实Roate函数有两个构造函数:void rotate(float degrees)
void rotate (float degrees, float px, float py)第一个构造函数直接输入旋转的度数,正数是顺时针旋转,负数指逆时针旋转,它的旋转中心点是原点(0,0)
第二个构造函数除了度数以外,还可以指定旋转的中心点坐标(px,py)

protected void onDraw(Canvas canvas) {  // TODO Auto-generated method stub  super.onDraw(canvas);  Paint paint_green = generatePaint(Color.GREEN, Style.FILL, 5);  Paint paint_red   = generatePaint(Color.RED, Style.STROKE, 5);  Rect rect1 = new Rect(300,10,500,100);  canvas.drawRect(rect1, paint_red); //画出原轮廓  canvas.rotate(30);//顺时针旋转画布  canvas.drawRect(rect1, paint_green);//画出旋转后的矩形  
}   


Matrix


一个类似的用来专门操作变换的玩意Matrix,之前我也说过我们会在很多地方用到这,Canvas也提供了对应的方法来便于我们设置Matrix直接变换Canvas:

@Overrideprotected void onDraw(Canvas canvas) {canvas.save(Canvas.MATRIX_SAVE_FLAG);  Matrix matrix = new Matrix();  matrix.setScale(0.8F, 0.35F);  matrix.postTranslate(100, 100);  canvas.setMatrix(matrix);  canvas.drawBitmap(mBitmap, 0, 0, null);  canvas.restore();  }


效果如下图所示,它是以屏幕为基准的

package com.example.customview;import android.app.Activity;
import android.os.Bundle;
import android.view.Window;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);this.requestWindowFeature(Window.FEATURE_NO_TITLE);//去掉标题栏 setContentView(R.layout.activity_main);}}





这篇关于自定义控件(11)---Canvas的平移、旋转、缩放、错切、Matrix直接变换Canvas的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1125308

相关文章

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

poj 2187 凸包or旋转qia壳法

题意: 给n(50000)个点,求这些点与点之间距离最大的距离。 解析: 先求凸包然后暴力。 或者旋转卡壳大法。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <s

自定义类型:结构体(续)

目录 一. 结构体的内存对齐 1.1 为什么存在内存对齐? 1.2 修改默认对齐数 二. 结构体传参 三. 结构体实现位段 一. 结构体的内存对齐 在前面的文章里我们已经讲过一部分的内存对齐的知识,并举出了两个例子,我们再举出两个例子继续说明: struct S3{double a;int b;char c;};int mian(){printf("%zd\n",s

Spring 源码解读:自定义实现Bean定义的注册与解析

引言 在Spring框架中,Bean的注册与解析是整个依赖注入流程的核心步骤。通过Bean定义,Spring容器知道如何创建、配置和管理每个Bean实例。本篇文章将通过实现一个简化版的Bean定义注册与解析机制,帮助你理解Spring框架背后的设计逻辑。我们还将对比Spring中的BeanDefinition和BeanDefinitionRegistry,以全面掌握Bean注册和解析的核心原理。

Android 10.0 mtk平板camera2横屏预览旋转90度横屏拍照图片旋转90度功能实现

1.前言 在10.0的系统rom定制化开发中,在进行一些平板等默认横屏的设备开发的过程中,需要在进入camera2的 时候,默认预览图像也是需要横屏显示的,在上一篇已经实现了横屏预览功能,然后发现横屏预览后,拍照保存的图片 依然是竖屏的,所以说同样需要将图片也保存为横屏图标了,所以就需要看下mtk的camera2的相关横屏保存图片功能, 如何实现实现横屏保存图片功能 如图所示: 2.mtk

Oracle type (自定义类型的使用)

oracle - type   type定义: oracle中自定义数据类型 oracle中有基本的数据类型,如number,varchar2,date,numeric,float....但有时候我们需要特殊的格式, 如将name定义为(firstname,lastname)的形式,我们想把这个作为一个表的一列看待,这时候就要我们自己定义一个数据类型 格式 :create or repla

lvgl8.3.6 控件垂直布局 label控件在image控件的下方显示

在使用 LVGL 8.3.6 创建一个垂直布局,其中 label 控件位于 image 控件下方,你可以使用 lv_obj_set_flex_flow 来设置布局为垂直,并确保 label 控件在 image 控件后添加。这里是如何步骤性地实现它的一个基本示例: 创建父容器:首先创建一个容器对象,该对象将作为布局的基础。设置容器为垂直布局:使用 lv_obj_set_flex_flow 设置容器

Verybot之OpenCV应用二:霍夫变换查找圆

其实我是想通过这个程序来测试一下,OpenCV在Verybot上跑得怎么样,霍夫变换的原理就不多说了,下面是程序: #include "cv.h"#include "highgui.h"#include "stdio.h"int main(int argc, char** argv){cvNamedWindow("vedio",0);CvCapture* capture;i

HTML5自定义属性对象Dataset

原文转自HTML5自定义属性对象Dataset简介 一、html5 自定义属性介绍 之前翻译的“你必须知道的28个HTML5特征、窍门和技术”一文中对于HTML5中自定义合法属性data-已经做过些介绍,就是在HTML5中我们可以使用data-前缀设置我们需要的自定义属性,来进行一些数据的存放,例如我们要在一个文字按钮上存放相对应的id: <a href="javascript:" d

[论文笔记]LLM.int8(): 8-bit Matrix Multiplication for Transformers at Scale

引言 今天带来第一篇量化论文LLM.int8(): 8-bit Matrix Multiplication for Transformers at Scale笔记。 为了简单,下文中以翻译的口吻记录,比如替换"作者"为"我们"。 大语言模型已被广泛采用,但推理时需要大量的GPU内存。我们开发了一种Int8矩阵乘法的过程,用于Transformer中的前馈和注意力投影层,这可以将推理所需