Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ianmackenzie/elm-geometry
2D/3D geometry package for Elm
https://github.com/ianmackenzie/elm-geometry
cad coordinate-systems curve elm geometry hacktoberfest point transformations vector
Last synced: 1 day ago
JSON representation
2D/3D geometry package for Elm
- Host: GitHub
- URL: https://github.com/ianmackenzie/elm-geometry
- Owner: ianmackenzie
- License: mpl-2.0
- Created: 2016-08-10T02:37:26.000Z (over 8 years ago)
- Default Branch: main
- Last Pushed: 2024-07-27T00:35:06.000Z (7 months ago)
- Last Synced: 2025-02-11T12:17:41.658Z (9 days ago)
- Topics: cad, coordinate-systems, curve, elm, geometry, hacktoberfest, point, transformations, vector
- Language: Elm
- Homepage:
- Size: 4.43 MB
- Stars: 184
- Watchers: 12
- Forks: 26
- Open Issues: 47
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
- Code of conduct: CODE_OF_CONDUCT.md
- Authors: AUTHORS
Awesome Lists containing this project
- awesome-ccamel - ianmackenzie/elm-geometry - 2D/3D geometry package for Elm (Elm)
README
# elm-geometry
_Release notes for all versions of `elm-geometry` are available
[here](https://github.com/ianmackenzie/elm-geometry/releases)._⚠️ **NOTE**: Currently you probably want to use `elm-geometry` 3.11 instead
if you're using `elm-3d-scene`, since `elm-3d-scene` has not yet been
updated to use `elm-geometry` 4.0. The next release of `elm-3d-scene`
will be updated to use `elm-geometry` 4.0.`elm-geometry` is an [Elm](http://elm-lang.org) package for working with 2D and
3D geometry. It provides a wide variety of geometric data types such as points,
vectors, arcs, spline curves and coordinate frames, along with functions for
transforming and combining them in many different ways. You can:- Rotate points around axes in 3D
- Mirror triangles across 3D planes
- Project 3D geometry into 2D sketch planes
- Measure distances and angles between different objects
- Convert objects between different coordinate systems
- Compose complex 2D/3D transformations
- ...and much more!## Table of contents
- [Overview](#overview)
- [Units and coordinate systems](#units-and-coordinate-systems)
- [Units](#units)
- [Coordinate systems](#coordinate-systems)
- [Conversions](#conversions)
- [Installation](#installation)
- [Using the package](#using-the-package)
- [Documentation](#documentation)
- [Related packages](#related-packages)
- [Climate action](#climate-action)
- [Questions and feedback](#questions-and-feedback)## Overview
`elm-geometry` includes a wide variety of data types: points, vectors, directions...
data:image/s3,"s3://crabby-images/8b8a5/8b8a5c5ae15bcbb619cb0d70a506830fced24389" alt="Point2d"
data:image/s3,"s3://crabby-images/eb40a/eb40ac87f8c9cbea3e0ee61634d35e7b58d2469b" alt="Point3d"
data:image/s3,"s3://crabby-images/9f8df/9f8df0f2372249b54672f5f9ea91998b55c1cd51" alt="Vector2d"
data:image/s3,"s3://crabby-images/e4f3f/e4f3fb2a8c5e25b54fd4daf0f00a46b0e9e12a90" alt="Vector3d"
data:image/s3,"s3://crabby-images/783f5/783f5631ff6c32c7babfcb34df26a8482594e19a" alt="Direction2d"
data:image/s3,"s3://crabby-images/2a917/2a917a24ec85db8fe2d900bc40694df67556e9c6" alt="Direction3d"...line segments, triangles, bounding boxes...
data:image/s3,"s3://crabby-images/7e02d/7e02d64a4e047cf680b6d5ea04c08d7f3ce9fbda" alt="LineSegment2d"
data:image/s3,"s3://crabby-images/c9bff/c9bff7fc49229b27262eadc0f92239b73439fa6b" alt="LineSegment3d"
data:image/s3,"s3://crabby-images/77009/7700972d1388090efb3e11fb12ab6e67ea5bf36e" alt="Triangle2d"
data:image/s3,"s3://crabby-images/72351/723513baeb1e7ff33db62f7544b0a012d2488007" alt="Triangle3d"
data:image/s3,"s3://crabby-images/e6403/e64039006715614157f74dac1bf8052bd7bdad10" alt="BoundingBox2d"
data:image/s3,"s3://crabby-images/a06c4/a06c4af6e47cbd2104eacab7f1f660086cb69cd7" alt="BoundingBox3d"...polylines, polygons, quadratic and cubic splines...
data:image/s3,"s3://crabby-images/fb47f/fb47ffb0df09651c44500e6976c9ee77ea594d38" alt="Polyline2d"
data:image/s3,"s3://crabby-images/a333b/a333b4f158b78f743be5a13f0bef22392b86bb76" alt="Polyline3d"
data:image/s3,"s3://crabby-images/b51ca/b51ca4e4848fcb99f6c40ddeed58db54b36f719a" alt="Polygon2d"
data:image/s3,"s3://crabby-images/06e70/06e70a0d606e119763c831d61f033a035954916c" alt="QuadraticSpline2d"
data:image/s3,"s3://crabby-images/a1ff1/a1ff14547ed8f4ad433f6e049eab045db153a586" alt="QuadraticSpline3d"
data:image/s3,"s3://crabby-images/f3611/f361153b4ec2aecd235de0e6211356b4163f3cfe" alt="CubicSpline2d"
data:image/s3,"s3://crabby-images/2d48f/2d48f7e2ecd29883204ce8305fe6984b879a37ed" alt="CubicSpline3d"...circles, arcs, ellipses and elliptical arcs...
data:image/s3,"s3://crabby-images/12d21/12d2161a988a8ddeaa091af8c300c71e52c94c05" alt="Circle2d"
data:image/s3,"s3://crabby-images/55fd0/55fd0bf7cc15e26c49a9fd62e22f9f84be3b82b2" alt="Circle3d"
data:image/s3,"s3://crabby-images/6caae/6caaef1dc1159ef1d21982696b1307d4df6baf42" alt="Arc2d"
data:image/s3,"s3://crabby-images/3a4dd/3a4dd66b199b9cdb10c6e4b2bf6e4cf126a19407" alt="Arc3d"
data:image/s3,"s3://crabby-images/db0c8/db0c8c666ee1c1357f2fed8f09fbd32fd4ed0b26" alt="Ellipse2d"
data:image/s3,"s3://crabby-images/f487a/f487ad3274858164c4878e930edffb9cd77a8081" alt="EllipticalArc2d"...plus axes, planes, and various forms of 2D/3D coordinate systems:
data:image/s3,"s3://crabby-images/94c4b/94c4b4e8f052395a4459ea5365242bd70787c262" alt="Axis2d"
data:image/s3,"s3://crabby-images/0f95e/0f95eb1779d5f8c22a12112aecf3c2875474922a" alt="Axis3d"
data:image/s3,"s3://crabby-images/ed219/ed219b280b14850c17d589b435d432a1f9453d2d" alt="Plane3d"
data:image/s3,"s3://crabby-images/665af/665af7527bae299cbeda027e71c80a4441d06cd0" alt="Frame2d"
data:image/s3,"s3://crabby-images/12d01/12d01e00207f1b55e6442ac77b089ce998d2b602" alt="Frame3d"
data:image/s3,"s3://crabby-images/bfb75/bfb756bbcba27164e0f3ba4743f06d29dce4ca5c" alt="SketchPlane3d"A large range of geometric functionality is included, such as various forms of
constructors...```elm
Point3d.xyz
(Length.meters 2)
(Length.meters 4)
(Length.meters 5)
-- OR --
Point3d.meters 2 4 5Direction2d.fromAngle (Angle.degrees 30)
-- OR --
Direction2d.degrees 30Point3d.midpoint p1 p2
Vector2d.withLength (Length.feet 3) Direction2d.y
Triangle2d.fromVertices ( p1, p2, p3 )
-- OR --
Triangle2d.from p1 p2 p3Plane3d.throughPoints p1 p2 p3
Axis3d.through Point3d.origin Direction3d.z
Arc2d.from p1 p2 (Angle.degrees 90)
QuadraticSpline3d.fromControlPoints p1 p2 p3
CubicSpline2d.fromEndpoints
startPoint
startDerivative
endPoint
endDerivative
```...point/vector arithmetic...
```elm
v1 |> Vector3d.plus v2-- the vector from the point p1 to the point p2
Vector2d.from p1 p2v1 |> Vector3d.cross v2
Vector2d.length vector
-- distance of a point from the origin
point |> Point2d.distanceFrom Point2d.origin
```...and 2D/3D transformations:
```elm
vector |> Vector2d.rotateBy anglepoint |> Point2d.rotateAround Point2d.origin angle
point |> Point3d.mirrorAcross Plane3d.xy
vector |> Vector3d.projectionIn Direction3d.z
triangle |> Triangle3d.rotateAround Axis3d.x angle
lineSegment
|> LineSegment3d.mirrorAcross Plane3d.yz
|> LineSegment3d.projectOnto Plane3d.xyPlane3d.xy |> Plane3d.offsetBy (Length.meters 3)
```## Units and coordinate systems
Most types in `elm-geometry` include two [phantom type parameters](https://blog.ilias.xyz/5-things-you-didnt-know-about-elm-7bdb67b1b3cd#ea40)
that allow compile-time tracking of both what units that geometry is in (usually
either meters for real-world geometry, or pixels for on-screen geometry) and
what coordinate system the geometry is defined in. For example, you might use a```elm
Point2d Pixels YUpCoordinates
```to represent a point on the screen that is defined in Y-up coordinates (from
the lower-left corner of an SVG drawing, for example) as opposed to Y-down
coordinates from the top left corner of the screen.### Units
`elm-geometry` uses the `Quantity` type from [`elm-units`](https://package.elm-lang.org/packages/ianmackenzie/elm-units/latest) to track/convert the units associated with
numeric values such as point coordinates, vector components, lengths, distances
and angles. Internally, `elm-units` converts everything to [SI](https://en.wikipedia.org/wiki/International_System_of_Units)
units, so```elm
Point2d.inches 10 20
```and
```elm
Point2d.centimeters 25.4 50.8
```are equivalent. Tracking units at compile time prevents mixing and matching
different types of geometry; for example,```elm
Point2d.xy (Length.meters 3) (Length.meters 4)
```and
```elm
Point2d.xy (Pixels.pixels 200) (Pixels.pixels 300)
```have completely different units, so the compiler can catch nonsensical
operations like trying to find the distance from the first point to the second.### Coordinate systems
2D/3D geometry is often represented using X/Y/Z coordinates. As a result, in
addition to tracking which units are used, `elm-geometry` also lets you add type
annotations to specify what _coordinate system_ particular geometry is defined
in. For example, we might declare a `TopLeftCoordinates` type and then add a
type annotation to a `point` asserting that it is defined in coordinates
relative to the top-left corner of the screen:```elm
{-| A coordinate system where (0, 0) is the top left corner
of the screen, positive X is to the right, and positive Y
is down.
-}
type TopLeftCoordinates =
TopLeftCoordinatespoint : Point2d Pixels TopLeftCoordinates
point =
Point2d.pixels 200 300
```Note that the `TopLeftCoordinates` type we declared gives us a convenient place
to document exactly how that coordinate system is defined. This combination now
gives us some nice type safety - the compiler will tell us if we try to mix two
points that have different units or are defined in different coordinate systems.## Installation
Assuming you have [installed Elm](https://guide.elm-lang.org/install.html) and
started a new project, you'll want to run```text
elm install ianmackenzie/elm-geometry
elm install ianmackenzie/elm-units
```in a command prompt inside your project directory. Note that even though
`elm-units` is a dependency of `elm-geometry`, you'll still need to explicitly
install it so that you can import modules like [`Quantity`](https://package.elm-lang.org/packages/ianmackenzie/elm-units/latest/Quantity)
and [`Length`](https://package.elm-lang.org/packages/ianmackenzie/elm-units/latest/Length)
in your own code (which will be needed in basically any code that uses
`elm-geometry`.)## Using the package
By itself, `elm-geometry` only performs abstract geometric operations like
measurements (distances, areas), checks (containment, intersection) and
transformations (scaling, rotation, translation, mirroring). See the [related
packages](#related-packages) section below for links to some packages that build
on top of `elm-geometry` to perform 2D drawing, 3D rendering, physics simulation
etc.In general when using `elm-geometry`, you'll need to import a module for every
different data type that you want to work with; there is no "main" module. For
example, to calculate the distance between two 2D points, you would import the
`Point2d` module and write something like:```elm
module Main exposing (main)import Html exposing (Html)
import Length -- from elm-units, see 'Installation'
import Point2dmain : Html msg
main =
let
firstPoint =
Point2d.meters 1 2secondPoint =
Point2d.meters 3 4distanceInCentimeters =
Point2d.distanceFrom firstPoint secondPoint
|> Length.inCentimeters
in
Html.text <|
"Distance: "
++ String.fromInt (round distanceInCentimeters)
++ " cm"
```which should end up displaying "Distance: 283 cm".
Note that it was necessary to also import the [`Length`](https://package.elm-lang.org/packages/ianmackenzie/elm-units/latest/Length)
module from [`elm-units`](https://package.elm-lang.org/packages/ianmackenzie/elm-units/latest/),
since the [`Point2d.distanceFrom`](https://package.elm-lang.org/packages/ianmackenzie/elm-geometry/latest/Point2d#distanceFrom)
function returns a [`Quantity Float units`](https://package.elm-lang.org/packages/ianmackenzie/elm-units/latest/Quantity#Quantity),
not a plain `Float`. In general, in addition to `elm-geometry` modules, you'll
likely need to import either the `Length` or [`Pixels`](https://package.elm-lang.org/packages/ianmackenzie/elm-units/latest/Pixels)
modules from `elm-units` (depending on whether you're working in real-world or
on-screen units) to work with any individual values returned by `elm-geometry`
functions (distances, areas, point coordinates, vector components, etc.).## Documentation
[Full API documentation](https://package.elm-lang.org/packages/ianmackenzie/elm-geometry/latest)
is available for each module. Most modules are associated with a particular data
type (for example, the [`Point3d`](https://package.elm-lang.org/packages/ianmackenzie/elm-geometry/latest/Point3d)
module contains functions for creating and manipulating `Point3d` values).## Related packages
There are several other Elm packages related to `elm-geometry`:
- For drawing in 2D, check out [`elm-geometry-svg`](https://package.elm-lang.org/packages/ianmackenzie/elm-geometry-svg/latest/)
- For 3D graphics, check out [`elm-3d-scene`](https://package.elm-lang.org/packages/ianmackenzie/elm-3d-scene/latest/)
for a high-level approach or [`elm-geometry-linear-algebra-interop`](https://package.elm-lang.org/packages/ianmackenzie/elm-geometry-linear-algebra-interop/latest/)
and [`elm-3d-camera`](https://package.elm-lang.org/packages/ianmackenzie/elm-3d-camera/latest/)
for working with WebGL directly
- For physics-based simulations/games, [`elm-physics`](https://package.elm-lang.org/packages/w0rm/elm-physics/latest/)
is based on `elm-geometry` and provides a 3D physics engine including
collisions, gravity, and constraints (joints)
- The [`elm-1d-parameter`](https://package.elm-lang.org/packages/ianmackenzie/elm-1d-parameter/latest/)
package is both used internally by `elm-geometry`, and is useful to combine
with functions like [`Point2d.interpolateFrom`](https://package.elm-lang.org/packages/ianmackenzie/elm-geometry/latest/Point2d#interpolateFrom)
to generate evenly-spaced values
- Functions like [`Polygon2d.triangulate`](https://package.elm-lang.org/packages/ianmackenzie/elm-geometry/latest/Polygon2d#triangulate)
return their results as a `TriangularMesh` value from [`elm-triangular-mesh`](https://package.elm-lang.org/packages/ianmackenzie/elm-triangular-mesh/latest/)I'm hopeful that in the future there will be packages that build on
`elm-geometry` to do non-graphical things like 3D printing or CNC machining!## Climate action
I would like for the projects I work on to be as helpful as possible in
addressing the climate crisis. If- you are working on a project that helps address the climate crisis (clean
energy, public transit, reforestation, sustainable agriculture etc.) either as
an individual, as part of an non-profit organization or even as part of a
for-profit company, and
- there is a new feature you would find helpful for that work (or a bug you need
fixed) in any of my open-source projects, thenplease [open a new issue](https://github.com/ianmackenzie/elm-geometry/issues),
describe briefly what you're working on and I will treat that issue as high
priority.## Questions and feedback
Please [open a new issue](https://github.com/ianmackenzie/elm-geometry/issues)
if you run into a bug, if any documentation is missing/incorrect/confusing, or
if there's a new feature that you would find useful. For general questions about
using `elm-geometry`, the best place is probably the **#geometry** channel on
the friendly [Elm Slack](http://elmlang.herokuapp.com/):data:image/s3,"s3://crabby-images/83887/83887ffb008a65c2df20b10dc8b78672cbc0f1e5" alt="Elm Slack #geometry channel conversation"
You can also try:
- Sending me (**@ianmackenzie**) a message on Slack - even if you don't have any
particular questions right now, it would be great to know what you're hoping
to do with the package!
- Posting to the [Elm Discourse](https://discourse.elm-lang.org/) forumsYou can also find me on Twitter ([@ianemackenzie](https://twitter.com/ianemackenzie)),
where I occasionally post `elm-geometry`-related stuff like demos or new
releases. Have fun, and don't be afraid to ask for help!