Skip to content

Commit 5e7a3ee

Browse files
committed
Changes:
* removed 'auto' option for options.round, the reason is the high duration of getting the path bounding box at initialization; * the default `origin` option value is no longer 'auto' but [0, 0, 0]; * conversion tools no longer make use of checks (EG: isPathArray) to remove bottlenecks at constructor initialization; * fixed issue with `pathToRelative` where additional MoveTo segments didn't get converted to relative #40 ; * fixed a small issue with `pathToAbsolute` with additional relative `m` segments not properly processed; * updated splitPath to work with relative arrays * removed `pathFactory` (an elegant solution but not very fast) in favor of the below mentioned; * removed `replaceArc` it's no longer needed; * added new tools for each segment type for calculating length, and bbox as well as finding a point at length; * fixed a small issue with Arc segments where getting point at given length was also finding points inside the shape and not in stroke; * further performance improvements #44; * updated tests; * updated docs / demo: added path total length, point at length and bbox; * updated dependencies.
1 parent a7d6372 commit 5e7a3ee

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

48 files changed

+2501
-1849
lines changed

README.md

+59-3
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
[![NPM Version](https://img.shields.io/npm/v/svg-path-commander.svg)](https://www.npmjs.com/package/svg-path-commander)
55
[![NPM Downloads](https://img.shields.io/npm/dm/svg-path-commander.svg)](http://npm-stat.com/charts.html?svg-path-commander)
66
[![jsDeliver](https://img.shields.io/jsdelivr/npm/hw/svg-path-commander)](https://www.jsdelivr.com/package/npm/svg-path-commander)
7-
[![typescript version](https://img.shields.io/badge/typescript-5.6.2-brightgreen)](https://www.typescriptlang.org/)
7+
[![typescript version](https://img.shields.io/badge/typescript-5.6.3-brightgreen)](https://www.typescriptlang.org/)
88
[![prettier version](https://img.shields.io/badge/prettier-2.8.8-brightgreen)](https://prettier.io/)
99
[![eslint version](https://img.shields.io/badge/eslint-8.57.1-brightgreen)](https://github.com/eslint)
1010
[![vitest version](https://img.shields.io/badge/vitest-2.1.2-brightgreen)](https://vitest.dev/)
@@ -75,7 +75,7 @@ const transform = {
7575
rotate: 15, // Z axis rotation
7676
scale: 0.75, // uniform scale on X, Y, Z axis
7777
skew: 15, // skew 15deg on the X axis
78-
origin: [15, 0] // if not specified, it will calculate a bounding box to determine a proper `transform-origin`
78+
origin: [15, 0] // if not specified, it will use the default origin value [0, 0]
7979
}
8080
const transformed2DPathString = new SVGPathCommander(path).transform(transform).toString();
8181
@@ -90,6 +90,21 @@ const transform = {
9090
const transformed3DPathString = new SVGPathCommander(path).transform(transform).toString();
9191
```
9292
93+
Access the `bbox` instance property to apply a consistent `transform-origin`:
94+
```js
95+
// apply a 3D transformation with a consistent origin
96+
const transformed3DPath = new SVGPathCommander(path);
97+
const { cx, cy, cz } = transformed3DPath.bbox;
98+
const transform = {
99+
translate: [15, 15, 15], // `[15, 15]` would apply a 2D translation, and only `15` for X axis translation
100+
rotate: [15, 15, 15], // or only "15" for 2D rotation on Z axis
101+
scale: [0.7, 0.75, 0.8], // or only "0.7" for 2D scale on all X, Y, Z axis
102+
skew: [15, 15], // or only "15" for the X axis
103+
origin: [cx, cy, cz] // the origin
104+
}
105+
const transformed3DPathString = transformed3DPath.transform(transform).toString();
106+
```
107+
93108
SVGPathCommander comes with a full range of additional static methods, here's how to normalize a path:
94109
```js
95110
const path = 'M0 0 H50';
@@ -112,13 +127,53 @@ const myPathString = SVGPathCommander.pathToString([['M', 0, 0], ['L', 50, 0]]);
112127
// result => 'M0 0 L50 0'
113128
```
114129
115-
116130
Check a path string validity:
117131
```js
118132
SVGPathCommander.isValidPath(path);
119133
// result => boolean
120134
```
121135
136+
Check if path is a certain kind of `PathArray`:
137+
```js
138+
SVGPathCommander.isAbsoluteArray([['M', 0, 0], ['L', 50, 0]]);
139+
// result => true
140+
```
141+
142+
Create a custom function to apply a 3D transformation using static methods:
143+
```ts
144+
import { parsePathString, getPathBBox, transformPath, pathToString } from 'svg-path-commander';
145+
146+
function myTransformFn(pathInput: string | PathArray, transformObject: TransformObject) {
147+
const path = parsePathString(pathInput);
148+
const { cx, cy, cz } = getPathBBox(path);
149+
150+
return pathToString(
151+
transformPath(path, {
152+
...transformObject, origin: [cx, cy, cz]
153+
})
154+
)
155+
}
156+
```
157+
In extreme cases where performance is paramount, you can consider the parent SVG `viewBox` attribute to extract a bounding box required for a consistent transform origin.
158+
159+
```ts
160+
// const svgViewBox = document.getElementById('my-svg').getAttribute('viewBox');
161+
const viewBox = '0 0 24 24';
162+
163+
const [x, y, width, height] = viewBox.split(/\s/).map(Number);
164+
const origin = [
165+
x + width / 2, // CX
166+
y + height / 2, // CY
167+
Math.max(width, height) + Math.min(width, height) / 2, // CZ
168+
];
169+
170+
// use this origin for your shape transformation
171+
const myNewString = new SVGPathCommander('M0 0 H50')
172+
.transform({ rotate: [35, 0, 0], origin })
173+
.toString();
174+
```
175+
176+
122177
Convert a shape to `<path>` and transfer all non-specific attributes
123178
```js
124179
const myCircle = document.getElementById('myCircle');
@@ -213,6 +268,7 @@ For developer guidelines, and a complete list of static methods, head over to th
213268
* James Halliday for his excelent [point-at-length](https://github.com/substack/point-at-length)
214269
* Eric Eastwood for his excelent [svg-curve-lib](https://github.com/MadLittleMods/svg-curve-lib)
215270
* PhET Interactive Simulations for their [kite](https://github.com/phetsims/kite)
271+
* [herrstrietzel](https://github.com/herrstrietzel) for his awesome [svg-pathdata-getbbox](https://github.com/herrstrietzel/svg-pathdata-getbbox)
216272
217273
# License
218274
**SVGPathCommander** is released under [MIT Licence](https://github.com/thednp/svg-path-commander/blob/master/LICENSE).

dist/svg-path-commander.cjs

+1-1
Large diffs are not rendered by default.

dist/svg-path-commander.cjs.map

+1-1
Large diffs are not rendered by default.

dist/svg-path-commander.d.ts

+113-25
Original file line numberDiff line numberDiff line change
@@ -141,7 +141,7 @@ export type LengthFactory = {
141141
};
142142
};
143143
export type Options = {
144-
round: "auto" | "off" | number;
144+
round: "off" | number;
145145
origin: number[];
146146
};
147147
export type PathTransform = {
@@ -368,6 +368,82 @@ export type PointTuple = [
368368
number,
369369
number
370370
];
371+
export type DerivedPoint = Point & {
372+
t: number;
373+
};
374+
export type QuadPoints = [
375+
Point,
376+
Point,
377+
Point,
378+
Point,
379+
Point,
380+
Point
381+
];
382+
export type CubicPoints = [
383+
Point,
384+
Point,
385+
Point,
386+
Point,
387+
Point,
388+
Point,
389+
Point,
390+
Point
391+
];
392+
export type DerivedQuadPoints = [
393+
DerivedPoint,
394+
DerivedPoint,
395+
DerivedPoint,
396+
DerivedPoint,
397+
DerivedPoint,
398+
DerivedPoint
399+
];
400+
export type DerivedCubicPoints = [
401+
DerivedPoint,
402+
DerivedPoint,
403+
DerivedPoint,
404+
DerivedPoint,
405+
DerivedPoint,
406+
DerivedPoint,
407+
DerivedPoint,
408+
DerivedPoint
409+
];
410+
export type QuadCoordinates = [
411+
number,
412+
number,
413+
number,
414+
number,
415+
number,
416+
number
417+
];
418+
export type CubicCoordinates = [
419+
number,
420+
number,
421+
number,
422+
number,
423+
number,
424+
number,
425+
number,
426+
number
427+
];
428+
export type ArcCoordinates = [
429+
number,
430+
number,
431+
number,
432+
number,
433+
number,
434+
number,
435+
number,
436+
number,
437+
number
438+
];
439+
export type LineCoordinates = [
440+
number,
441+
number,
442+
number,
443+
number
444+
];
445+
export type DeriveCallback = (t: number) => Point;
446+
export type IteratorCallback = (segment: PathSegment, params: ParserParams, index: number) => PathSegment;
371447
/**
372448
* Creates a new SVGPathCommander instance with the following properties:
373449
* * segments: `pathArray`
@@ -381,31 +457,24 @@ export type PointTuple = [
381457
declare class SVGPathCommander {
382458
static CSSMatrix: typeof CSSMatrix$1;
383459
static getSVGMatrix: (transform: TransformObjectValues) => CSSMatrix$1;
384-
static getPathBBox: (path: PathArray | string) => PathBBox;
460+
static getPathBBox: (pathInput: PathArray | string) => {
461+
x: number;
462+
y: number;
463+
width: number;
464+
height: number;
465+
x2: number;
466+
y2: number;
467+
cx: number;
468+
cy: number;
469+
cz: number;
470+
};
385471
static getPathArea: (path: PathArray) => number;
386472
static getTotalLength: (pathInput: string | PathArray) => number;
387473
static getDrawDirection: (path: string | PathArray) => boolean;
388-
static getPointAtLength: (pathInput: string | PathArray, distance: number) => {
474+
static getPointAtLength: (pathInput: string | PathArray, distance?: number) => {
389475
x: number;
390476
y: number;
391477
};
392-
static pathFactory: (pathInput: string | PathArray, distance?: number) => {
393-
point: {
394-
x: number;
395-
y: number;
396-
};
397-
length: number;
398-
readonly bbox: {
399-
min: {
400-
x: number;
401-
y: number;
402-
};
403-
max: {
404-
x: number;
405-
y: number;
406-
};
407-
};
408-
};
409478
static getPropertiesAtLength: (pathInput: string | PathArray, distance?: number) => SegmentProperties;
410479
static getPropertiesAtPoint: (pathInput: string | PathArray, point: {
411480
x: number;
@@ -440,16 +509,15 @@ declare class SVGPathCommander {
440509
static parsePathString: (pathInput: string | PathArray) => PathArray;
441510
static roundPath: (path: PathArray, roundOption?: number | "off") => PathArray;
442511
static splitPath: (pathInput: PathArray) => PathArray[];
443-
static splitCubic: (pts: number[]) => [
512+
static splitCubic: (pts: number[], ratio?: number) => [
444513
CubicSegment,
445514
CubicSegment
446515
];
447-
static replaceArc: (pathInput: PathArray | string) => PathArray;
448516
static optimizePath: (pathInput: PathArray, round: "off" | number) => PathArray;
449517
static reverseCurve: (path: CurveArray) => CurveArray;
450518
static reversePath: (pathInput: PathArray) => PathArray;
451519
static normalizePath: (pathInput: string | PathArray) => NormalArray;
452-
static transformPath: (path: string | PathArray, transform?: Partial<TransformObject>) => PathArray;
520+
static transformPath: (pathInput: PathArray | string, transform?: Partial<TransformObject>) => PathArray;
453521
static pathToAbsolute: (pathInput: string | PathArray) => AbsoluteArray;
454522
static pathToRelative: (pathInput: string | PathArray) => RelativeArray;
455523
static pathToCurve: (pathInput: string | PathArray) => CurveArray;
@@ -467,15 +535,35 @@ declare class SVGPathCommander {
467535
* @param config instance options
468536
*/
469537
constructor(pathValue: string, config?: Partial<Options>);
470-
get bbox(): PathBBox;
538+
get bbox(): {
539+
x: number;
540+
y: number;
541+
width: number;
542+
height: number;
543+
x2: number;
544+
y2: number;
545+
cx: number;
546+
cy: number;
547+
cz: number;
548+
};
471549
get length(): number;
472550
/**
473551
* Returns the path bounding box, equivalent to native `path.getBBox()`.
474552
*
475553
* @public
476554
* @returns the pathBBox
477555
*/
478-
getBBox(): PathBBox;
556+
getBBox(): {
557+
x: number;
558+
y: number;
559+
width: number;
560+
height: number;
561+
x2: number;
562+
y2: number;
563+
cx: number;
564+
cy: number;
565+
cz: number;
566+
};
479567
/**
480568
* Returns the total path length, equivalent to native `path.getTotalLength()`.
481569
*

dist/svg-path-commander.js

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

dist/svg-path-commander.js.map

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)