-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathtransform.ts
70 lines (57 loc) · 1.72 KB
/
transform.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
import {Attributes, setAttributes} from './elements.ts';
import {AbstractTransformableTo} from './transformable.ts';
export class Transform extends AbstractTransformableTo<Transform> {
protected constructor(readonly svgTransform: string) {
super();
}
static fromSVGTransform(svgTransform: string) {
return new Transform(svgTransform);
}
static readonly IDENTITY = new Transform("");
transform(tf: Transform) {
return new Transform([tf.svgTransform, this.svgTransform].filter(Boolean).join(" "));
}
asAttributes(): Attributes {
return {transform: this.svgTransform || undefined};
}
toString() {
return `transform="${this.svgTransform}"`;
}
}
/**
* A convenience tool for creating transforms. Usage example:
*
* Tf.rotate(20).translateX(10)
*/
export const Tf = Transform.IDENTITY;
export function transformedToString(object: unknown, tf: Transform) {
if (tf.svgTransform)
return `${tf.svgTransform}*${object}`;
return `${object}`;
}
function simplifyTf<
A extends string,
S extends SVGElement & Record<A, SVGAnimatedTransformList>,
>(
element: SVGElement,
subclass: {prototype: S, new(): S},
attribute: A,
) {
if (element instanceof subclass) {
const tfList = element[attribute].baseVal;
if (tfList.length > 1) {
const matrix = tfList.consolidate()?.matrix;
if (matrix)
setAttributes(element, {
[attribute]: Tf.matrix([
matrix.a, matrix.b, matrix.c, matrix.d, matrix.e, matrix.f,
]).svgTransform,
});
}
}
}
export function simplifyTransform(element: SVGElement) {
simplifyTf(element, SVGGradientElement, "gradientTransform");
simplifyTf(element, SVGGraphicsElement, "transform");
return element;
}