了解Android Matrix转换

ImageView

原文 了解Android Matrix转换

很多年前,在学校我学习了矩阵。 我记不太清楚了,但我记得的是在想,“但是......你对这些知识做了什么呢?”

快进几年,我开始作为Android开发人员工作,不得不使用ImageViewscaleType - 如果你看过所有可能的类型,你已经注意到其中一个是matrix 。 多年来,我一直避开它,使用其他规模类型或解决问题。 然而几周前我正在开发一种设计,其中组件的背景图像应该与视图的左上角对齐,而不执行任何缩放,如下所示:

ImageView

所以我继续添加了一个ImageView并再次浏览了所有scaleType - 希望有一个我错过的类型,但是在我尝试使用scaleType="matrix"它们都没有完全正确,它完全符合我的要求。 但为什么这有效呢? 它实际上做了什么?

所以我看了一下Matrix文档:

Matrix类包含一个3x3矩阵,用于转换坐标

嗯......不是很有帮助。 幸运的是,我并不是唯一一个不懂的人, Arnaud Bos写了一篇很棒的文章 ,详细解释了背后的数学(警告:如果你打算阅读它 - 可能需要一杯咖啡(或者两杯)的时间)。 如果你在那篇文章的中途迷路了,我不能责怪你 - 这很复杂,但好消息是你不必理解数学就能用matrix(虽然它很有帮助。

我们怎样才能使用Matrix

正如我之前提到的,我们必须在ImageView上设置scaleType="matrix" 。 但要真正能够使用它,我们必须在代码中设置imageMatrix

imageView.imageMatrix = Matrix().apply {
    // perform transformations
}

现在我们有了这个 - 我们可以用它做什么? 矩阵支持一系列不同的变换,如translatescalerotateskew 。 如果这些听起来很熟悉,因为它们(大多数)与View,动画或画布上的相同。

您会发现,对于每个操作,都有一setpre版本。 稍后我会谈到这一点,但是现在我们只使用set版本。

所以,我们能做些什么?

Translating (移动)

ImageView

设置translation意味着将图像移动到其他位置。 您所要做的就是使用Matrix上所需的xy坐标调用setTranslate

val dWidth = imageView.drawable.intrinsicWidth
val dHeight = imageView.drawable.intrinsicHeight

val vWidth = imageView.measuredWidth
val vHeight = imageView.measuredHeight
setTranslate(
    round((vWidth - dWidth) * 0.5f),
    round((vHeight - dHeight) * 0.5f)
)

在这个例子中,我们只是将drawable置于View中心,这导致与在ImageView上设置scaleType="center"相同的行为。 那么让我们来看看ImageView如何做到这一点:

mDrawMatrix.setTranslate(Math.round((vwidth - dwidth) * 0.5f),
                         Math.round((vheight - dheight) * 0.5f));

它完全一样! 所以我们不知道它已经使用了矩阵变换。

Scaling (缩放)

ImageView

Scaling(正如您可能已经猜到的那样)定义了图像的大小。 您可以定义两个值 - 一个用于x轴,另一个用于y轴。 但是,通过缩放,您还可以设置轴心点。

枢轴点定义转换将保持不变的点。 默认情况下它是0, 0 - 左上角 - 意味着图像将向右和向下延伸,左上角保持不变 - 就像左边的上面的gif一样。

如果要从中心缩放图像(如右侧的gif),可以将轴设置为图像的中心,如下所示:

setScale(0.5f, 0.5f, dWidth / 2f, dHeight / 2f)

这将使图像缩放到其大小的一半,其中心的枢轴点。 如果你只想从左上角缩放它,你可以省略最后两个参数,如下所示:

setScale(0.5f, 0.5f)

但是你可以用缩放做更多的事情。 如果提供负比例值,则基本上可以围绕轴(或两个)镜像图像。 相当漂亮!

ImageView

Rotation (旋转)

ImageView

你猜对了! 通过旋转,您可以旋转图像。 在这里,我们提供了我们想要旋转的角度,以及一个类似于比例的可选枢轴点。

  setRotate(45f,dWidth / 2f,dHeight / 2f) 

将图像围绕图像中心旋转45度。 如果将它旋转-45度,它将向左旋转。

Skewing (倾斜)

imageView

倾斜可能是你以前没有听说过的转变。 倾斜将沿轴(或两个)拉伸您的图像,就像上面的GIF一样。 我们来看一个例子:

  setSkew(1f,0f,dWidth / 2f,dHeight / 2f) 

这会使图像在x轴(以及中心点周围)偏斜1,这是图像的宽度,导致图像倾斜45度,就像上面的gif一样。

应用多个转换

我们现在可以translatescalerotateskew图像,但是如果我们想要将它们组合起来呢? 显而易见的事情可能是连续调用多个set方法。 但是,这只会应用最后一次转换 - 所有以前的转换都将被覆盖。 这是因为set方法基本上重置了Matrix

但正如我之前提到的,还有每个转换的post版本。 通过使用这些,我们可以应用多个变换,并真正利用matrix的魔力。

但是pre的区别是什么? 对于第一次转换,使用三种版本中的哪一种没有区别,但对于任何未来的转换,它可以产生很大的不同。

假设我们想要将图像转换为视图的中心并将其缩放到一半大小。 这两个版本将产生预期的效果:

val drawableLeft = round((vWidth - dWidth) * 0.5f)
val drawableTop = round((vHeight - dHeight) * 0.5f)
// Version 1
setTranslate(drawableLeft, drawableTop)
val (viewCenterX, viewCenterY) = vWidth / 2f to vHeight / 2f
postScale(0.5f, 0.5f, viewCenterX, viewCenterY)
// Version 2
setTranslate(drawableLeft, drawableTop)
val (drawableCenterX, drawableCenterY) = dWidth / 2f to dHeight / 2f
preScale(0.5f, 0.5f, drawableCenterX, drawableCenterY)

请注意,在第一个版本中,我们使用postScale和视图的中心,而在第二个版本中,我们使用preScaledrawable的中心。

使用postScale ,将在translate后应用比例转换。 由于图像已经在视图中居中,我们必须使用视图的中心点作为枢轴。

val (viewCenterX, viewCenterY) = vWidth / 2f to vHeight / 2f
postScale(0.5f, 0.5f, viewCenterX, viewCenterY)

所以从头开始回顾这个例子 - 为什么应用scaleType="matrix"只是起作用? 使用默认martix,比例将为1,平移,旋转和倾斜将为0,因此图像将在左上角绘制。 所以它完全符合我的需要!

下次您必须以默认scaleType不起作用的方式布局图像时 - 尝试使用Matrix

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

推荐阅读更多精彩内容