Skip to content

Commit ac94063

Browse files
committed
Initial commit
1 parent 6903e33 commit ac94063

File tree

3 files changed

+122
-0
lines changed

3 files changed

+122
-0
lines changed

docs/api.rst

+9
Original file line numberDiff line numberDiff line change
@@ -115,6 +115,15 @@ Gridspec classes
115115
:toctree: api
116116

117117

118+
Artist subclass
119+
===============
120+
121+
.. automodule:: proplot.artist
122+
123+
.. automodsumm:: proplot.artist
124+
:toctree: api
125+
126+
118127
Miscellaneous tools
119128
===================
120129

proplot/__init__.py

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
from .figure import * # noqa: F401 F403
2020
from .ui import * # noqa: F401 F403
2121
from .demos import * # noqa: F401 F403
22+
from .artist import * # noqa: F401 F403
2223

2324
# SCM versioning
2425
name = 'proplot'

proplot/artist.py

+112
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,112 @@
1+
#!/usr/bin/env python3
2+
"""
3+
Create a metaclass
4+
"""
5+
import matplotlib.artist as martist
6+
import matplotlib.lines as mlines
7+
8+
__all__ = []
9+
10+
PROPS_PROTECTED = [
11+
'figure', 'axes', 'pickradius',
12+
]
13+
14+
15+
def _gen_subclasses(cls):
16+
"""
17+
Generate all of a class's sublcasses with recursion.
18+
"""
19+
try:
20+
for subclass in cls.__subclasses__():
21+
yield subclass
22+
for subclass in _gen_subclasses(subclass):
23+
yield subclass
24+
except TypeError:
25+
return
26+
27+
28+
def _gen_properties(cls):
29+
"""
30+
Generate property definitions for every
31+
Get proper
32+
"""
33+
for attr in dir(cls):
34+
obj = getattr(cls, attr)
35+
if callable(obj) and attr[:4] == 'get_':
36+
getter = obj
37+
name = attr[4:]
38+
if hasattr(cls, name):
39+
continue
40+
elif name in PROPS_PROTECTED:
41+
continue
42+
args = [getter] # property args
43+
setter = getattr(cls, 'set_' + name, None)
44+
if callable(setter):
45+
args.append(setter)
46+
yield name, property(*args, doc=getter.__doc__)
47+
48+
49+
# Approach #1: Apply properties to all artists
50+
for cls in _gen_subclasses(martist.Artist):
51+
for name, prop in _gen_properties(cls):
52+
setattr(cls, name, prop)
53+
print(f'Added properties to {cls.__name__}.')
54+
55+
56+
# Approach #2: Requires importing proplot before matplotlib.
57+
# This will not work for ProPlot but may be in matplotlib's future.
58+
class Propertifier(type): # create new class definitions
59+
"""
60+
Artist metaclass.
61+
"""
62+
def __init__(cls, *args, **kwargs):
63+
"""
64+
Add property getters and setters to all matplotlib artist subclasses.
65+
"""
66+
super().__init__(*args, **kwargs)
67+
for name, property in _gen_properties(cls):
68+
setattr(cls, name, property)
69+
print(f'Added properties to {cls.__name__}.')
70+
71+
72+
class Artist(martist.Artist, metaclass=Propertifier):
73+
"""
74+
New base class for all matplotlib artists that
75+
uses `Propertifier` as the metaclass.
76+
"""
77+
78+
79+
# Apply monkey patch
80+
import matplotlib.artist as martist
81+
martist.Artist = Artist
82+
83+
84+
# Testing metaclasses
85+
class MetaClass(type):
86+
def __init__(cls, *args, **kwargs):
87+
print('Brand new class!', cls)
88+
super().__init__(*args, **kwargs)
89+
90+
91+
A = MetaClass('A', (object,), {})
92+
B = MetaClass('B', (A,), {})
93+
94+
95+
# Testing Artist
96+
class Test(Artist):
97+
def set_foo(self, foo):
98+
print('set foo!', foo)
99+
100+
def get_foo(self):
101+
print('get foo!')
102+
103+
def get_bar(self):
104+
print('get bar!')
105+
106+
107+
class Line2D(Artist, mlines.Line2D):
108+
pass
109+
110+
111+
test = Test()
112+
line = Line2D([0], [0])

0 commit comments

Comments
 (0)