Metadata-Version: 2.4
Name: expressive-shapes
Version: 0.1.0
Summary: Expressive shape morphing and animations
Keywords: geometry,bezier,polygon,morph,shape,rounded-polygon,material-design,cairo
Author: Aman Valiyakath
Author-email: Aman Valiyakath <notifyamanvs@gmail.com>
License-Expression: GPL-3.0-only
License-File: LICENSE
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: GNU General Public License v3 (GPLv3)
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Typing :: Typed
Classifier: Topic :: Multimedia :: Graphics
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Requires-Python: >=3.9
Project-URL: Homepage, https://github.com/amansxcalibur/expressive-shapes
Project-URL: Source, https://github.com/amansxcalibur/expressive-shapes
Project-URL: Issues, https://github.com/amansxcalibur/expressive-shapes/issues
Project-URL: Changelog, https://github.com/amansxcalibur/expressive-shapes/blob/main/CHANGELOG.md
Description-Content-Type: text/markdown

# expressive-shapes

A pure-Python library for creating, rounding, and morphing polygons using cubic Bezier curves. Inspired by Android's [Material Design shape system](https://developer.android.com/develop/ui/compose/graphics/draw/shapes), brought to Python.

<p align="center">
  <img src="https://amansxcalibur.github.io/zenith-resources/cairo-shapes/demo.gif" width="320" alt="Shape morphing demo"/>
</p>
<p align="center"><i>"I like the dots and shapes, and I think they like me too"</i></p>

## Features

- **Rounded polygons** -- Create polygons with per-vertex corner rounding and smoothing, matching Material Design 3 specifications
- **Shape morphing** -- Smoothly interpolate between any two shapes using feature-aware Bezier matching
- **30+ built-in presets** -- Circle, star, heart, clover, ghost, pixel shapes, and more
- **Renderer-agnostic** -- Outputs cubic Bezier curves; render with Cairo, SVG, Canvas, or any path-based renderer
- **Zero dependencies** -- Pure Python, no external packages required

## Installation

```bash
pip install expressive-shapes
```

## Quick Start

### Create a rounded polygon

Vertices are defined as a flat list of `[x, y, x, y, ...]`. The built-in presets use unit coordinates (0.0--1.0), which keeps shapes resolution-independent -- you scale at render time. But you're free to use whatever coordinate space fits your setup (pixel coords, world units, etc.).

```python
from expressive_shapes import RoundedPolygon, CornerRounding

# unit-coordinate triangle (0.0-1.0)
poly = RoundedPolygon.create(
    vertices=[0.5, 0.1, 0.9, 0.9, 0.1, 0.9],
    rounding=CornerRounding(radius=0.08, smoothing=0.6),
)

# get cubic Bezier curves to render with any graphics library
for curve in poly.get_all_curves():
    print(curve.p0, curve.p1, curve.p2, curve.p3)
```

### Morph between two shapes

```python
from expressive_shapes import RoundedPolygon, CornerRounding
from expressive_shapes.morph.bezier_morph import Morph

poly_a = RoundedPolygon.create(
    vertices=[0.5, 0.1, 0.9, 0.9, 0.1, 0.9],  # triangle
    rounding=CornerRounding(radius=0.07, smoothing=0.0),
)
poly_b = RoundedPolygon.create(
    vertices=[0.2, 0.2, 0.8, 0.2, 0.8, 0.8, 0.2, 0.8],  # square
    rounding=CornerRounding(radius=0.07, smoothing=0.0),
)

# match features between the two shapes (done once)
matched = Morph.match(poly_a, poly_b)

# interpolate at any progress value between 0.0 and 1.0
curves = Morph.as_cubics(matched, progress=0.5)  # halfway morph
```

### Use built-in shape presets

Presets are defined in unit coordinates. Scale them to your target size when creating the polygon, or let your renderer handle it via a transform.

```python
from expressive_shapes import RoundedPolygon
from expressive_shapes.shapes.shape_presets import star, heart, circle

def preset_to_polygon(preset):
    # convert a unit-coordinate preset to a RoundedPolygon
    verts = []
    per_vertex = []
    for (ux, uy), rounding in preset:
        verts.extend([ux, uy])
        per_vertex.append(rounding)
    return RoundedPolygon.create(vertices=verts, per_vertex_rounding=per_vertex)

star_poly = preset_to_polygon(star)
heart_poly = preset_to_polygon(heart)
```

### Render with Cairo

Since shapes are in unit coordinates, use `ctx.scale()` to map them to your output size. This is the typical approach for Cairo/SVG/Canvas-style renderers, but you can just as easily pre-multiply your vertices if your pipeline expects pixel coordinates.

```python
import cairo
from expressive_shapes.morph.bezier_morph import Morph

# create poly_a and poly_b in unit coords as above
matched = Morph.match(poly_a, poly_b)
curves = Morph.as_cubics(matched, progress=0.35)

size = 500
surface = cairo.ImageSurface(cairo.FORMAT_ARGB32, size, size)
ctx = cairo.Context(surface)

# scale unit coords (0.0-1.0) up to the output size
ctx.scale(size, size)

ctx.move_to(curves[0].p0.x, curves[0].p0.y)
for c in curves:
    ctx.curve_to(c.p1.x, c.p1.y, c.p2.x, c.p2.y, c.p3.x, c.p3.y)
ctx.close_path()

ctx.set_source_rgb(0.24, 0.52, 0.93)
ctx.fill()

surface.write_to_png("morph.png")
```

## API Overview

| Class            | Description                                                                                |
| ---------------- | ------------------------------------------------------------------------------------------ |
| `RoundedPolygon` | A polygon with per-vertex rounding, represented as cubic Bezier features                   |
| `CornerRounding` | Controls corner `radius` and `smoothing` (0.0 = circular arc, 1.0 = fully smoothed)        |
| `Morph`          | Feature-aware shape matching and interpolation                                             |
| `Point`          | 2D point with arithmetic operations and interpolation                                      |
| `Cubic`          | A cubic Bezier segment (p0, p1, p2, p3)                                                    |
| `Feature`        | A polygon segment -- either a "corner" or an "edge", containing one or more `Cubic` curves |

## Shape Presets

The `shapes.shape_presets` module includes 30+ ready-to-use shapes defined in unit coordinates (0.0-1.0):

`circle`, `square`, `triangle`, `diamond`, `pentagon`, `star`, `heart`, `arrow`, `shield`, `pill`, `arch`, `semicircle`, `oval`, `fan`, `gem`, `clamshell`, `cookie_8`, `cookie_12`, `four_leaf_clover`, `boom`, `ghost_ish`, `bun`, `pixel_circle`, `pixel_triangle`, `puffy_square`, `puffy_diamond`, `concave_rectangle`, `slanted`, and more.

Each preset is a list of `((x, y), CornerRounding)` tuples that can be scaled to any size.

## Debugging

Debug output is available behind a flag:

```python
import expressive_shapes
expressive_shapes.DEBUG = True
```

This enables detailed output for shape matching, feature alignment, and the morph walking algorithm.

## Requirements

- Python >= 3.9
- No runtime dependencies

## License

[GPL-3.0](LICENSE)
