Metadata-Version: 2.4
Name: uplt-py
Version: 0.8.2
Summary: Unified API and style for Python plotting libraries
Project-URL: Repository, https://github.com/makarovdi/uplt
Project-URL: Issues, https://github.com/makarovdi/uplt/issues
Project-URL: Changelog, https://github.com/makarovdi/uplt/blob/main/CHANGELOG.md
Author-email: Dmitry Makarov <makarovdmv@gmail.com>
License-Expression: BSD-3-Clause
License-File: LICENSE
Classifier: Intended Audience :: Developers
Classifier: Intended Audience :: Science/Research
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Topic :: Scientific/Engineering :: Visualization
Requires-Python: >=3.10
Requires-Dist: numpy<3.0,>=1.21
Requires-Dist: pillow<11.0,>=10.3.0
Requires-Dist: scipy<2.0,>=1.11
Provides-Extra: all
Requires-Dist: kaleido; extra == 'all'
Requires-Dist: matplotlib<4.0,>=3.7; extra == 'all'
Requires-Dist: plotly<6.0,>=5.16; extra == 'all'
Provides-Extra: matplot
Requires-Dist: matplotlib<4.0,>=3.7; extra == 'matplot'
Provides-Extra: plotly
Requires-Dist: kaleido; extra == 'plotly'
Requires-Dist: plotly<6.0,>=5.16; extra == 'plotly'
Description-Content-Type: text/markdown

<h1 align="center">
    <img src="https://media.githubusercontent.com/media/makarovdi/uplt/refs/heads/develop/logo.png", width=200>
</h1>

<br>

[![python](https://img.shields.io/badge/Python-3.10-blue?logo=python&logoColor=white)](https://docs.python.org/3/whatsnew/3.10.html)
[![license](https://img.shields.io/badge/License-BSD%203--Clause-green)](https://choosealicense.com/licenses/mit/)
[![jupyter](https://img.shields.io/badge/Jupyter-Lab-F37626.svg?style=flat&logo=Jupyter)](https://jupyterlab.readthedocs.io/en/stable)

Unified API and style for Python plotting libraries.

## Usage

<table style="border-collapse: collapse; border-style: hidden;">

<tr>
    <td><b>plotly</b></td>
    <td><b>matplotlib</b></td>
</tr>

<tr>
<td>

```python
import numpy as np
import uplt

x = np.linspace(0, np.pi*4, num=100)
phi = np.pi/4

fig = uplt.figure('plotly')
fig.plot(x, np.sin(x - 0*phi), name='#1')
fig.plot(x, np.sin(x - 1*phi), name='#2')
fig.plot(x, np.sin(x - 2*phi), name='#3')
fig.plot(x, np.sin(x - 3*phi), name='#4')
fig.xlabel('X').ylabel('Y')
fig.legend().show()
```

</td>

<td>

```python
import numpy as np
import uplt

x = np.linspace(0, np.pi*4, num=100)
phi = np.pi/4

fig = uplt.figure('matplot')
fig.plot(x, np.sin(x - 0*phi), name='#1')
fig.plot(x, np.sin(x - 1*phi), name='#2')
fig.plot(x, np.sin(x - 2*phi), name='#3')
fig.plot(x, np.sin(x - 3*phi), name='#4')
fig.xlabel('X').ylabel('Y')
fig.legend().show()
```

</td>
</tr>

<tr>
<td>

<picture align="center">
    <img src="https://media.githubusercontent.com/media/makarovdi/uplt/refs/heads/main/gallery/asset/plotly5-example.png">
</picture>

</td>

<td>

<picture align="center">
    <img src="https://media.githubusercontent.com/media/makarovdi/uplt/refs/heads/main/gallery/asset/mpl-example.png">
</picture>

</td>

</tr>
</table>

> 💡 See [gallery](https://github.com/makarovdi/uplt/blob/main/gallery/gallery.md) for more examples.  

> 💡 The `uplot` alias is available and can be used interchangeably with `uplt`.

## Install

Recent stable version (without any plotting library):
```bash
pip install uplt-py
```
To automatically install all optional dependencies (matplotlib, plotly, ...):
```bash
pip install "uplt-py[all]"
```

If you need only `matplotlib` support:
```bash
pip install "uplt-py[matplot]"
```
> 💡  Replace `[matplot]` with `[plotly]` for plotly-only installation


## Plotting Libs - Pros & Cons

### [Matplotlib](https://matplotlib.org/)

🟢 Highly configurable.  
🟢 Good documentation and a lot of ready-to-use recipes (e.g. on StackOverflow).  
🟡 Common API (MATLAB legacy).  


🔴 Limited interactivity (especially for Jupyter).  
🔴 API, behavior and parameter names are inconsistent (e.g. plt.xlim and axis.set_xlim).  
🔴 Slow and limited 3D rendering.  


### [Plotly](https://plotly.com/python/)

🟢 Very good interactivity.  
🟢 Native compatibility with Jupyter.  
🟢 Possibility to save interactive plot (html-file).  
🟢 Fast and interactive 3D plot.  

🔴 Not well documented (a lot of parameters, small amount of examples).  
🔴 High memory consumption (limited number of plots in Jupyter).  
🔴 Some expected API functions are missing (e.g. imshow).  
🔴 3D and 2D axis parameters are not unified (e.g. layout.xaxis doesn't work for 3D).  

## Functions

| Function                                                            | Description                                                                                                                                                   |
| :------------------------------------------------------------------ | :------------------------------------------------------------------------------------------------------------------------------------------------------------ |
| `plot(x, y, z)` <br/> `plot(obj)`                                   | Plot 2D or 3D line. <br/>Line plot for custom class (supported by a plugin).                                                                                  |
| `scatter(x, y, z)` <br/> `scatter(obj)`                             | Scatter plot for 2D or 3D data points. <br/> Scatter plot for custom class (supported by a plugin).                                                           |
| `surface3d(x, y, z)`                                                | Plot a surface in 3D space where the color scale corresponds to the z-values.                                                                                 |
| `bar(x, y)`                                                         | Create a bar plot.                                                                                                                                            |
| `imshow(image)`                                                     | Display an image.                                                                                                                                             |
| `hline(y)` <br/> `vline(x)`                                         | Plot horizontal or vertical line. `2D only`                                                                                                                   |
| `title(text)`                                                       | Set the title of the figure.                                                                                                                                  |
| `legend(show)`                                                      | Show or hide the legend on the figure.                                                                                                                        |
| `grid(show)`                                                        | Show or hide the grid on the figure.                                                                                                                          |
| `xlabel(text)` <br/> `ylabel(text)` <br/> `zlabel(text)`            | Set the label for the x, y, z-axis.                                                                                                                           |
| `xlim(min, max)` <br/> `ylim(min, max)` <br/> `zlim(min, max)`      | Set limits for the x, y, z-axis.                                                                                                                              |
| `xscale()` <br/> `yscale()`                                         | Set scale for the x, y-axis: 'linear' or 'log'.                                                                                                               |
| `current_color()` <br/> `scroll_color(count)` <br/> `reset_color()` | Get the color which will be used for the next plot. <br/> Scroll a list of predefined colors for plots. <br/> Set the current color to the start of the list. |
| `axis_aspect(mode)`                                                 | Set the aspect ratio of the axis.                                                                                                                             |
| `as_image()`                                                        | Get the figure as a NumPy array.                                                                                                                              |
| `save(filename)`                                                    | Save the figure to a file.                                                                                                                                    |
| `close()`                                                           | Close the figure. Free allocated resources.                                                                                                                   |
| `show(block)`                                                       | Display the figure.                                                                                                                                           |


## Extending


### Plugin

The plugin system allows extending `uplt` for visualizing custom objects.
For example, the `DataFrame` plugin enables this code:
```python
import uplt
import pandas as pd

car_crashes = pd.read_csv(
    'https://raw.githubusercontent.com/mwaskom/seaborn-data/master/car_crashes.csv'
)

fig = uplt.figure()
fig.plot(car_crashes[['total', 'speeding', 'alcohol', 'no_previous']])
fig.show()
```
<picture align="left">
    <img src='https://media.githubusercontent.com/media/makarovdi/uplt/refs/heads/main/gallery/asset/plugin.png' width='480'>
</picture>


To implement the plugin, you can follow this structure:
```python
import numpy as np
import pandas as pd

import uplt.plugin as plugin


class DataFramePlugin(plugin.IPlotPlugin):

    def extract_data(self, obj: pd.DataFrame) -> list[plugin.PlotData]:
        data = []
        for name in obj.columns:
            if not np.issubdtype(obj.dtypes[name], np.number): continue
            y = obj[name].values
            x = np.arange(len(y))
            data.append(plugin.PlotData(x=x, y=y, name=name.replace('_', ' ').title()))
        return data

plugin.register(pd.DataFrame, handler=DataFramePlugin())
```

> 💡 Check `test/plugin.py` for a more advanced plugin example.

### Engine

Adding a new plotting library is straightforward. Implement two interfaces `IPlotEngine` and `IFigure`:
```python
import uplt
from uplt import IPlotEngine, IFigure

class MyEngine(IPlotEngine):
    ...
    def figure(self, ...) -> MyFigure: ...

class MyFigure(IFigure):
    def plot(self, ...): ...
    def scatter(self, ...): ...
    ...

# register the engine
uplt.engine.register(MyEngine(), name='test')
```
Then use it in the regular way:
```python
import uplt

fig = uplt.figure(engine='test')
fig.plot(...)
fig.show()
```

## Dependencies

- `Python` ≥ 3.10
- `NumPy` ≥ 1.21 `v2.0 supported`
- `pillow` ≥ 10.3

### Optional
- `matplotlib` ≥ 3.7
- `plotly` ≥  5.17


## License

This software is licensed under the `BSD-3-Clause` license.
See the [LICENSE](https://github.com/makarovdi/uplt/blob/main/LICENSE) file for details.

## TODO

Check the plan for new features [here](https://github.com/makarovdi/uplt/blob/develop/TODO.md).
