Tabbed plot extension for matplotlib using the Qt backend
Project description
AbracaTABra
This repository is basically a matplotlib extension using the Qt backend to create plot windows with groups of tabs, where the contents of each tab is a matplotlib figure.
This package is essentially a replacement for pyplot; it creates and manages figures separately from pyplot, so calling pyplot.show() or pyplot.pause() will not do anything with windows created from this package.
This package provides the functions show_all_windows() and update_all_windows(delay_seconds), which are very similar in behavior to show() and pause(interval), respectively, from pyplot.
Also, abracatabra() is a more fun equivalent to show_all_windows()...you should try it out!
Dependencies
- matplotlib
- One of the following Qt bindings for Python (this is the order matplotlib looks for them):
- PyQt6
- PySide6 (preferred option)
- PyQt5
- PySide2
Installation
This will install the package as well as matplotlib, if it isn't installed:
pip install abracatabra
Qt bindings are an optional dependency of the package. A Python Qt package is required for functionality, but there is no good way to have a default optional dependency with pip...so you have to install separately or manually specify one of the following optional dependencies:
- [qt-pyside6]
- [qt-pyqt6]
For example, run this to install PySide6 along with this package:
pip install "abracatabra[qt-pyside6]"
NOTE: Optional dependency installations are not provided for PyQt5 or PySide2 because QT 5 has reached end of life and it is not recommended to use them.
Usage
import numpy as np
import abracatabra as tabby
window1 = tabby.TabbedPlotWindow(window_id="README example", ncols=2)
window2 = tabby.TabbedPlotWindow(size=(500, 400))
# data
t = np.arange(0, 10, 0.001)
ysin = np.sin(t)
ycos = np.cos(t)
f = window1.add_figure_tab("sin", col=0)
ax = f.add_subplot()
(line1,) = ax.plot(t, ysin, "--")
ax.set_xlabel("time")
ax.set_ylabel("sin(t)")
ax.set_title("Plot of sin(t)")
f = window1.add_figure_tab("time", col=1)
ax = f.add_subplot()
ax.plot(t, t)
ax.set_xlabel("time")
ax.set_ylabel("t")
ax.set_title("Plot of t")
window1.apply_tight_layout()
f = window2.add_figure_tab("cos")
ax = f.add_subplot()
(line2,) = ax.plot(t, ycos, "--")
ax.set_xlabel("time")
ax.set_ylabel("cos(t)")
ax.set_title("Plot of cos(t)")
f = window2.add_figure_tab("time")
ax = f.add_subplot()
ax.plot(t, t)
ax.set_xlabel("time")
ax.set_ylabel("t")
ax.set_title("Plot of t", fontsize=20)
window2.apply_tight_layout()
### animate
## option 1
dt = 0.1
for k in range(100):
t += dt
ysin = np.sin(t)
line1.set_ydata(ysin)
ycos = np.cos(t)
line2.set_ydata(ycos)
# For timing to be accurate, you would have to calculate how long it took to
# run the previous 5 lines and subtract that from dt
tabby.update_all_windows(dt)
# You would need this to keep windows open if not in an interactive environment
# tabby.show_all_windows(block=True)
## option 2: use animation callbacks
print("Same thing, but using animation callbacks now")
def update_sin(frame: int):
time = t + frame * dt
line1.set_ydata(np.sin(time))
def update_cos(frame: int):
time = t + frame * dt
line2.set_ydata(np.cos(time))
window1.tab_groups[0, 0].get_tab("sin").register_animation_callback(update_sin)
window2.tab_groups[0, 0].get_tab("cos").register_animation_callback(update_cos)
# This method accounts for how long it takes to call the animation callbacks
# so that the time between frames is closer to the specified time step. Also,
# if a tab is not active (visible), it will not call the animation callback
# for that tab, which can save a lot of time if you have many tabs.
tabby.animate_all_windows(frames=100, ts=dt, print_timing=True, hold=False)
tabby.abracatabra() # keep windows open if not in an interactive environment
Example using blitting
import numpy as np
import abracatabra
blit = True
window = abracatabra.TabbedPlotWindow(autohide_tabs=True)
fig = window.add_figure_tab("robot arm animation", include_toolbar=False, blit=blit)
ax = fig.add_subplot()
# background elements
fig.tight_layout()
ax.set_aspect("equal", "box")
length = 1.0
lim = 1.25 * length
ax.axis((-lim, lim, -lim, lim))
(baseline,) = ax.plot([0, length], [0, 0], "k--")
# draw and save background for fast rendering
fig.canvas.draw()
background = fig.canvas.copy_from_bbox(ax.bbox)
# moving elements
def get_arm_endpoints(theta):
x = np.array([0, length * np.cos(theta)])
y = np.array([0, length * np.sin(theta)])
return x, y
time = np.linspace(0, 10, 501)
theta_hist = np.sin(time)
x, y = get_arm_endpoints(theta_hist[0])
(arm_line,) = ax.plot(x, y, linewidth=5, color="blue")
# animate
def animation_step(idx: int):
theta = theta_hist[idx]
x, y = get_arm_endpoints(theta)
arm_line.set_xdata(x)
arm_line.set_ydata(y)
if blit:
fig.canvas.restore_region(background)
ax.draw_artist(arm_line)
dt = time[1] - time[0]
window.register_animation_callback(animation_step, "robot arm animation")
abracatabra.animate_all_windows(frames=len(theta_hist), ts=dt, print_timing=True)
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file abracatabra-1.4.0.tar.gz.
File metadata
- Download URL: abracatabra-1.4.0.tar.gz
- Upload date:
- Size: 45.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8190217f9662ad45154b52863ec387b34958d84be6a74755958482f67439de5b
|
|
| MD5 |
fcd6868fd8704750ac5e518948840ae2
|
|
| BLAKE2b-256 |
9d5071233cd3036933e509fe5574bbe1e14aa4369c638eec3f4ade35ede7c02b
|
File details
Details for the file abracatabra-1.4.0-py3-none-any.whl.
File metadata
- Download URL: abracatabra-1.4.0-py3-none-any.whl
- Upload date:
- Size: 56.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
283790850055149f350a35eb29b53789ae1899fa4ef2830412a13fad45dba0ed
|
|
| MD5 |
0f4be8ed041571065281319e53a05a3e
|
|
| BLAKE2b-256 |
da0f276015273f2fd4aa06614c02644566bc2dbc8d8ebe8150ec77face7ed3ca
|