从 Matrix 解构出 Translate/Scale/Rotate(平移/缩放/旋转)

在 XAML 中,我们对一个 UIElement 进行一个 RenderTransform 是再常见不过的事情了,我们可以从众多叠加的 TransformGroup 瞬间得到一个 Matrix 表示整个变换的综合变换矩阵,然而反过来却不好做——从变换矩阵中反向得到变换分量。


首先明确的是,各种 TranslateTransformScaleTransformRotateTransformMatrix 具有唯一确定的解,然而反向转换却是有无穷多个解的。于是如果我们要得到一个解,我们需要给定一个条件,然后得到这个条件下的其中一个解。

准备工作

为了写出一个通用的变换方法来,我准备了一个测试控件,并为它随意填写一个变换:

<Border x:Name="DisplayShape" Background="#FF1B6CB0" Width="200" Height="100">
    <UIElement.RenderTransform>
        <TransformGroup>
            <ScaleTransform ScaleX="0.8" ScaleY="2"/>
            <SkewTransform/>
            <RotateTransform Angle="-48.366"/>
            <TranslateTransform x:Name="TranslateTransform" X="85" Y="160"/>
        </TransformGroup>
    </UIElement.RenderTransform>
    <TextBlock Foreground="{media:LuminanceForeground TargetName=DisplayShape}" Text="walterlv" HorizontalAlignment="Center" VerticalAlignment="Center"/>
</Border>

LuminanceForeground 的作用可参见我的另一篇文章:计算能在任何背景色上清晰显示的前景色


▲ 一个随便应用了一个变换的控件

我们将从这个控件中取得变换矩阵 Matrix,然后计算出变换分量的一个解,应用到新的控件上:

<Rectangle x:Name="TraceShape" Width="200" Height="100" Stroke="#FFE2620A" StrokeThickness="4"/>


▲ 我们希望计算一组变换分量以便让这个框追踪变换了的控件

于是,我们写下了测试代码:

private void OnLoaded(object sender, RoutedEventArgs args)
{
    var matrix = DisplayShape.RenderTransform.Value;
    var (scaling, rotation, translation) = ExtractMatrix(matrix);
    var group = new TransformGroup();
    group.Children.Add(new ScaleTransform {ScaleX = scaling.X, ScaleY = scaling.Y});
    group.Children.Add(new RotateTransform {Angle = rotation});
    group.Children.Add(new TranslateTransform {X = translation.X, Y = translation.Y});
    TraceShape.RenderTransform = group;
}

private (Vector Scaling, double Rotation, Vector Translation) ExtractMatrix(Matrix matrix)
{
    // 我们希望在这里写出一个方法,以便得到三个变换分量。
}

OnLoaded 是为了让代码运行起来,而 ExtractMatrix 才是我们的核心——将变换分量解构出来。


思路和初步成果

我们的思路是创造一个单位矩形,让它应用这个变换,然后测量变换后矩形的宽高变化,角度变化和位置变化。由于直接使用 Rect 类型时无法表示旋转后的矩形,所以我们直接使用四个顶点来计算,于是我们写出如下代码:

private (Vector Scaling, double Rotation, Vector Translation) ExtractMatrix(Matrix matrix)
{
    var unitPoints = new[] {new Point(0, 0), new Point(1, 0), new Point(1, 1), new Point(0, 1)};
    var transformedPoints = unitPoints.Select(matrix.Transform).ToArray();
    var scaling = new Vector(
        (transformedPoints[1] - transformedPoints[0]).Length,
        (transformedPoints[3] - transformedPoints[0]).Length);
    var rotation = Vector.AngleBetween(new Vector(1, 0), transformedPoints[1] - transformedPoints[0]);
    var translation = transformedPoints[0] - unitPoints[0];
    return (scaling, rotation, translation);
}

运行后,我们发现追踪框已经与原始控件完全贴合,说明计算正确。


▲ 追踪框完全贴合


可以灵活应用计算结果

不过如果真要在产品中做追踪框,肯定不能像上图那样被严重拉伸。所以,我们把缩放分量去掉,换成尺寸变化:

private void OnLoaded(object sender, RoutedEventArgs args)
{
    var matrix = DisplayShape.RenderTransform.Value;
    var (scaling, rotation, translation) = ExtractMatrix(matrix);
    var group = new TransformGroup();
    TraceShape.Width = DisplayShape.ActualWidth * scaling.X;
    TraceShape.Height = DisplayShape.ActualHeight * scaling.Y;
    group.Children.Add(new RotateTransform {Angle = rotation});
    group.Children.Add(new TranslateTransform {X = translation.X, Y = translation.Y});
    TraceShape.RenderTransform = group;
}

以上代码中,ScaleTransform 已经被去掉,取而代之的是宽高的设置。


▲ 没有被拉伸的追踪框


更通用的方法

以上虽然达到了目的,不过实际应用中可能会有更多的限制,例如:

首先来解决变换中心的通用性问题。

我们将变换中心设为 (0.5, 0.5)

<Rectangle x:Name="TraceShape" Width="200" Height="100" Stroke="#FFE2620A" StrokeThickness="4" RenderTransformOrigin="0.5 0.5"/>

于是,追踪框不知道飞到哪里去了……


▲ 改变了变换中心

这时,我们需要将变换中心导致的额外平移量考虑在内。

如果 S 表示所求变换的缩放分量,R 表示所求变换的旋转分量,T 表示所求变换的平移分量;M 表示需要模拟的目标矩阵。那么,S 将可以通过缩放比和参数指定的缩放中心唯一确定;R 将可以通过旋转角度和参数指定的旋转中心唯一确定;T 不能确定,是我们要求的。

由于我们按照缩放->旋转->平移的顺序模拟 M,所以:

\[SRT=M\]

即:

\[T=S^{-1}R^{-1}M\]

所以,我们在上面的之前成果的代码上再做些额外的处理,加上以上公式的推导结果:

public static (Vector Scaling, double Rotation, Vector Translation) MatrixToGroup(Matrix matrix, CenterSpecification specifyCenter = null)
{
    // 生成一个单位矩形(0, 0, 1, 1),计算单位矩形经矩阵变换后形成的带旋转的矩形。
    // 于是,我们将可以通过比较这两个矩形中点的数据来求出一个解。
    var unitPoints = new[] {new Point(0, 0), new Point(1, 0), new Point(1, 1), new Point(0, 1)};
    var transformedPoints = unitPoints.Select(matrix.Transform).ToArray();

    // 测试单位矩形宽高的长度变化量,以求出缩放比(作为参数 specifyCenter 中变换中心的计算参考)。
    var scaling = new Vector((transformedPoints[1] - transformedPoints[0]).Length, (transformedPoints[3] - transformedPoints[0]).Length);
    // 测试单位向量的旋转变化量,以求出旋转角度。
    var rotation = Vector.AngleBetween(new Vector(1, 0), transformedPoints[1] - transformedPoints[0]);
    var translation = transformedPoints[0] - unitPoints[0];

    // 如果指定了变换分量的变换中心点。
    if (specifyCenter != null)
    {
        // 那么,就获取指定的变换中心点(缩放中心和旋转中心)。
        var (scalingCenter, rotationCenter) = specifyCenter(scaling);

        // 如果 S 表示所求变换的缩放分量,R 表示所求变换的旋转分量,T 表示所求变换的平移分量;M 表示传入的目标矩阵。
        // 那么,S 将可以通过缩放比和参数指定的缩放中心唯一确定;R 将可以通过旋转角度和参数指定的旋转中心唯一确定。
        // S = scaleMatrix; R = rotateMatrix.
        var scaleMatrix = Matrix.Identity;
        scaleMatrix.ScaleAt(scaling.X, scaling.Y, scalingCenter.X, scalingCenter.Y);
        var rotateMatrix = Matrix.Identity;
        rotateMatrix.RotateAt(rotation, rotationCenter.X, rotationCenter.Y);

        // T 是不确定的,它会受到 S 和 T 的影响;但确定等式 SRT=M,即 T=S^{-1}R^{-1}M。
        // T = translateMatrix; M = matrix.
        scaleMatrix.Invert();
        rotateMatrix.Invert();
        var translateMatrix = Matrix.Multiply(rotateMatrix, scaleMatrix);
        translateMatrix = Matrix.Multiply(translateMatrix, matrix);

        // 用考虑了变换中心的平移量覆盖总的平移分量。
        translation = new Vector(translateMatrix.OffsetX, translateMatrix.OffsetY);
    }

    // 按缩放、旋转、平移来返回变换分量。
    return (scaling, rotation, translation);
}

本来第二个参数是可以用 Func 的,但那样的意义解释起来太费劲,所以改成了委托的定义:

/// <summary>
/// 为 <see cref="MatrixToGroup"/> 方法提供变换中心的指定方法。
/// </summary>
/// <param name="scalingFactor">先进行缩放后进行旋转时,旋转中心的计算可能需要考虑前面缩放后的坐标。此参数可以得知缩放比。</param>
/// <returns>绝对坐标的缩放中心和旋转中心。</returns>
public delegate (Point ScalingCenter, Point RotationCenter) CenterSpecification(Vector scalingFactor);

这时我们就可以得到我们想要的 TransformGroup,而且 RenderTransformOrigin 随便设:

private void OnLoaded(object sender, RoutedEventArgs args)
{
    var matrix = DisplayShape.RenderTransform.Value;
    var (scaling, rotation, translation) = TransformMatrix.MatrixToGroup(matrix,
        scalingFactor => (new Point(), new Point(
            DisplayShape.ActualWidth * scalingFactor.X / 2,
            DisplayShape.ActualHeight * scalingFactor.Y / 2)));
    TraceShape.RenderTransform = ScaleAtZeroRotateAtCenter(scaling, rotation, translation, DisplayShape.RenderSize, TraceShape.RenderTransformOrigin);
}

public static TransformGroup ScaleAtZeroRotateAtCenter(Vector scaling, double rotation, Vector translation, Size originalSize, Point renderTransformOrigin = default(Point))
{
    var group = new TransformGroup();
    var scaleTransform = new ScaleTransform
    {
        ScaleX = scaling.X,
        ScaleY = scaling.Y,
        CenterX = -originalSize.Width * renderTransformOrigin.X,
        CenterY = -originalSize.Height * renderTransformOrigin.Y,
    };
    var rotateTransform = new RotateTransform
    {
        Angle = rotation,
        CenterX = originalSize.Width * (scaling.X / 2 - renderTransformOrigin.X),
        CenterY = originalSize.Height * (scaling.Y / 2 - renderTransformOrigin.Y),
    };
    group.Children.Add(scaleTransform);
    group.Children.Add(rotateTransform);
    group.Children.Add(new TranslateTransform {X = translation.X, Y = translation.Y});
    return group;
}

考虑到前面可以灵活地运用得到的变换分量,我们现在也这么用:

private void OnLoaded(object sender, RoutedEventArgs args)
{
    var matrix = DisplayShape.RenderTransform.Value;
    var (scaling, rotation, translation) = TransformMatrix.MatrixToGroup(matrix,
        scalingFactor => (new Point(), new Point(
            DisplayShape.ActualWidth * scalingFactor.X / 2,
            DisplayShape.ActualHeight * scalingFactor.Y / 2)));
    TraceShape.Width = DisplayShape.ActualWidth * scaling.X;
    TraceShape.Height = DisplayShape.ActualHeight * scaling.Y;
    TraceShape.RenderTransform = NoScaleButRotateAtOrigin(
        rotation, translation, DisplayShape.RenderSize);
}

public static TransformGroup NoScaleButRotateAtOrigin(double rotation, Vector translation, Size originalSize)
{
    var group = new TransformGroup();
    group.Children.Add(new RotateTransform {Angle = rotation});
    group.Children.Add(new TranslateTransform {X = translation.X, Y = translation.Y});
    return group;
}

我们的 RenderTransformOrigin 是随意设的,效果也像下图一样稳定可用。为了直观,我把两种用法放到了一起比较:

▲ 设置了 RenderTransformOrigin 依然有用

本文会经常更新,请阅读原文: https://blog.walterlv.com/post/extract-translation-scaling-rotation-from-matrix.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

如果你想持续阅读我的最新博客,请点击 RSS 订阅,或者前往 CSDN 关注我的主页

知识共享许可协议 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 吕毅 (包含链接: https://blog.walterlv.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 (walter.lv@qq.com)