Matrix Transformation example
This animation is interactive. When all the transformations have been applied, you can drag both the vector and the vector.
import 'package:manim_web/manim.dart';
class MatrixTransformationScene extends Scene {
late Array transformationMatrix;
late NumberPlane plane;
late NumberPlane fadedPlane;
late NumberPlane originalPlane;
late Dot i;
late Dot j;
late Dot originalI;
late Dot originalJ;
late Arrow arrowI;
late Arrow arrowJ;
late MathTex texI;
late MathTex texJ;
late MathTex texMatrix;
double planeScale = 1.5;
@override
FutureOr<void> preload() {
MathTex.preload(r'\hat{\imath}');
MathTex.preload(r'\hat{\jmath}');
}
@override
Future construct() async {
transformationMatrix = Array.identity(2);
await createPlane();
await createControls();
await addTexMatrix();
await transformPlane([
[0, -1],
[1, 0]
]); // rotation on 90 degrees
await transformPlane([
[1, 0.5],
[0, 1]
]); // sheer along x
makeInteractive();
await continueRendering();
}
Future createPlane() async {
var planeRadius = max(FRAME_X_RADIUS, FRAME_Y_RADIUS);
originalPlane = NumberPlane(
xMin: -planeRadius * planeScale,
xMax: planeRadius * planeScale,
yMin: -planeRadius * planeScale,
yMax: planeRadius * planeScale,
makeSmoothAfterApplyingFunctions: false);
plane = originalPlane.copy();
await play(AnimationGroup([
for (var submob in plane.getFamily())
if (submob.hasPoints()) ShowCreation(submob),
], lagRatio: 0.01));
}
Future createControls() async {
originalI = Dot(originalPlane.c2p(RIGHT))..setColor(color: RED_C);
originalJ = Dot(originalPlane.c2p(UP))..setColor(color: GREEN_C);
var originPoint = originalPlane.c2p(ORIGIN);
i = originalI.copy();
j = originalJ.copy();
arrowI = Arrow(start: originPoint, end: i.getPos(), buff: 0)
..setColor(color: RED_C);
arrowJ = Arrow(start: originPoint, end: j.getPos(), buff: 0)
..setColor(color: GREEN_C);
addTex();
addToFront([i, j, arrowI, arrowJ, texI, texJ]);
await play(AnimationGroup([
for (var mob in [i, j, arrowI, arrowJ]) ShowCreation(mob),
for (var tex in [texI, texJ]) FadeIn(tex)
], lagRatio: 0.1));
}
void addTex() {
texI = MathTex(r'\hat{\imath}')
..setColor(color: RED_C)
..addUpdater(
(m, dt) => m..nextToMobject(i, direction: UL, buffer: SMALL_BUFFER));
texJ = MathTex(r'\hat{\jmath}')
..setColor(color: GREEN_C)
..addUpdater(
(m, dt) => m..nextToMobject(j, direction: UL, buffer: SMALL_BUFFER));
}
Future addTexMatrix() async {
// Tex substring isolation isn't possible yet
// TODO Change that
// texMatrix = MathTex('''
// \\begin{bmatrix}
// {{1.00}} & {{0.00}} \\\\
// {{0.00}} & {{1.00}}
// \\end{bmatrix}
// ''');
// texMatrix.addUpdater((_, dt) => texMatrix
// ..content = '''
// \\begin{bmatrix}
// ${transformationMatrix[Tuple2(0, 0)]} & ${transformationMatrix[Tuple2(0, 1)]} \\\\
// ${transformationMatrix[Tuple2(1, 0)]} & ${transformationMatrix[Tuple2(1, 1)]}
// \\end{bmatrix}
// ''');
// await play(FadeIn(texMatrix));
}
Future transformPlane(List<List<double>> values) async {
var matrix = Array(values: values);
// assert values is a 2x2 matrix
assert(listEqual(matrix.shape.toList(), [2, 2]));
var p2 = plane.copy()..applyMatrix(matrix);
transformationMatrix = matrix.matMul(transformationMatrix);
var iPos = originalPlane.c2p(RIGHT).matMul(transformationMatrix);
var jPos = originalPlane.c2p(UP).matMul(transformationMatrix);
// update the position of i and j
var i2 = i.copy()..moveToPoint(iPos);
var j2 = j.copy()..moveToPoint(jPos);
var originPoint = originalPlane.c2p(ORIGIN);
var newArrowI = Arrow(start: originPoint, end: iPos, buff: 0)
..setColor(color: RED_C);
var newArrowJ = Arrow(start: originPoint, end: jPos, buff: 0)
..setColor(color: GREEN_C);
await playMany([
Transform(i, target: i2),
Transform(j, target: j2),
Transform(arrowI, target: newArrowI),
Transform(arrowJ, target: newArrowJ),
Transform(plane, target: p2),
]);
}
void makeInteractive() {
add([DraggableMobject(mob: i), DraggableMobject(mob: j)]);
var planeUpdater = (Mobject mob, double dt) {
var iPos = originalPlane.p2c(i.getCenter());
var jPos = originalPlane.p2c(j.getCenter());
var newPlane = originalPlane.copy()
..applyMatrix(Array(values: [
[iPos.x, jPos.x],
[iPos.y, jPos.y]
]));
var originPoint = originalPlane.c2p(ORIGIN);
var newArrowI = Arrow(start: originPoint, end: iPos, buff: 0)
..setColor(color: RED_C);
var newArrowJ = Arrow(start: originPoint, end: jPos, buff: 0)
..setColor(color: GREEN_C);
plane.become(newPlane);
arrowI.become(newArrowI);
arrowJ.become(newArrowJ);
return mob;
};
i.addUpdater(planeUpdater);
j.addUpdater(planeUpdater);
}
}
Vector3 listToVector(List<double> l) =>
l.length == 2 ? Vector3(l[0], l[1], 0) : Vector3(l[0], l[1], l[2]);