Skip to content

Commit 7ab0c90

Browse files
authored
Handle large collections of subplots without validation error (#1091)
* Fix for GH1031, large subplot grids resulted in validation error. * Removed internal usage of deprecated classes.
1 parent d670bb0 commit 7ab0c90

File tree

3 files changed

+41
-8
lines changed

3 files changed

+41
-8
lines changed

plotly/matplotlylib/renderer.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -530,7 +530,7 @@ def draw_text(self, **props):
530530
if not align:
531531
align = props['style']['halign'] # mpl default
532532
if 'annotations' not in self.plotly_fig['layout']:
533-
self.plotly_fig['layout']['annotations'] = go.Annotations()
533+
self.plotly_fig['layout']['annotations'] = []
534534
if props['text_type'] == 'xlabel':
535535
self.msg += " Text object is an xlabel\n"
536536
self.draw_xlabel(**props)
@@ -576,7 +576,7 @@ def draw_text(self, **props):
576576
yref = 'paper'
577577
xanchor = props['style']['halign'] # no difference here!
578578
yanchor = mpltools.convert_va(props['style']['valign'])
579-
annotation = go.Annotation(
579+
annotation = go.layout.Annotation(
580580
text=(str(props['text']) if
581581
isinstance(props['text'], six.string_types) else
582582
props['text']),
@@ -631,7 +631,7 @@ def draw_title(self, **props):
631631
'position'])
632632
x, y = mpltools.display_to_paper(x_px, y_px,
633633
self.plotly_fig['layout'])
634-
annotation = go.Annotation(
634+
annotation = go.layout.Annotation(
635635
text=props['text'],
636636
font=go.layout.annotation.Font(
637637
color=props['style']['color'],

plotly/tests/test_core/test_tools/test_make_subplots.py

+27
Original file line numberDiff line numberDiff line change
@@ -2128,3 +2128,30 @@ def test_subplot_titles_insets(self):
21282128
fig = tls.make_subplots(insets=[{'cell': (1, 1), 'l': 0.7, 'b': 0.3}],
21292129
subplot_titles=("", 'Inset'))
21302130
self.assertEqual(fig, expected)
2131+
2132+
def test_large_columns_no_errors(self):
2133+
"""
2134+
Test that creating subplots with a large number of columns, and
2135+
zero vertical spacing doesn't result in domain values that are out
2136+
of range.
2137+
2138+
Here is the error that was reported in GH1031
2139+
2140+
ValueError:
2141+
Invalid value of type 'builtins.float' received for the
2142+
'domain[1]' property of layout.yaxis
2143+
Received value: 1.0000000000000007
2144+
2145+
The 'domain[1]' property is a number and may be specified as:
2146+
- An int or float in the interval [0, 1]
2147+
2148+
"""
2149+
v_space = 0.0
2150+
2151+
# 2D
2152+
fig = tls.make_subplots(100, 1, vertical_spacing=v_space)
2153+
2154+
# 3D
2155+
fig = tls.make_subplots(100, 1,
2156+
vertical_spacing=v_space,
2157+
specs=[[{'is_3d': True}] for _ in range(100)])

plotly/tools.py

+11-5
Original file line numberDiff line numberDiff line change
@@ -568,9 +568,9 @@ def get_subplots(rows=1, columns=1, print_grid=False, **kwargs):
568568
y_start = (plot_height + vertical_spacing) * rrr
569569
y_end = y_start + plot_height
570570

571-
xaxis = graph_objs.XAxis(domain=[x_start, x_end], anchor=x_anchor)
571+
xaxis = dict(domain=[x_start, x_end], anchor=x_anchor)
572572
fig['layout'][xaxis_name] = xaxis
573-
yaxis = graph_objs.YAxis(domain=[y_start, y_end], anchor=y_anchor)
573+
yaxis = dict(domain=[y_start, y_end], anchor=y_anchor)
574574
fig['layout'][yaxis_name] = yaxis
575575
plot_num += 1
576576

@@ -1080,7 +1080,11 @@ def _get_anchors(r, c, x_cnt, y_cnt, shared_xaxes, shared_yaxes):
10801080
# Function pasting x/y domains in layout object (2d case)
10811081
def _add_domain(layout, x_or_y, label, domain, anchor, position):
10821082
name = label[0] + 'axis' + label[1:]
1083-
axis = {'domain': domain}
1083+
1084+
# Clamp domain elements between [0, 1].
1085+
# This is only needed to combat numerical precision errors
1086+
# See GH1031
1087+
axis = {'domain': [max(0.0, domain[0]), min(1.0, domain[1])]}
10841088
if anchor:
10851089
axis['anchor'] = anchor
10861090
if isinstance(position, float):
@@ -1090,7 +1094,9 @@ def _add_domain(layout, x_or_y, label, domain, anchor, position):
10901094

10911095
# Function pasting x/y domains in layout object (3d case)
10921096
def _add_domain_is_3d(layout, s_label, x_domain, y_domain):
1093-
scene = graph_objs.Scene(domain={'x': x_domain, 'y': y_domain})
1097+
scene = dict(
1098+
domain={'x': [max(0.0, x_domain[0]), min(1.0, x_domain[1])],
1099+
'y': [max(0.0, y_domain[0]), min(1.0, y_domain[1])]})
10941100
layout[s_label] = scene
10951101

10961102
x_cnt = y_cnt = s_cnt = 1 # subplot axis/scene counters
@@ -1339,7 +1345,7 @@ def _pad(s, cell_len=cell_len):
13391345
'yref': 'paper',
13401346
'text': subplot_titles[index],
13411347
'showarrow': False,
1342-
'font': graph_objs.Font(size=16),
1348+
'font': dict(size=16),
13431349
'xanchor': 'center',
13441350
'yanchor': 'bottom'
13451351
})

0 commit comments

Comments
 (0)