32
32
from mplfinance import _styles
33
33
34
34
from mplfinance ._arg_validators import _check_and_prepare_data , _mav_validator
35
- from mplfinance ._arg_validators import _get_valid_plot_types
35
+ from mplfinance ._arg_validators import _get_valid_plot_types , _fill_between_validator
36
36
from mplfinance ._arg_validators import _process_kwargs , _validate_vkwargs_dict
37
37
from mplfinance ._arg_validators import _kwarg_not_implemented , _bypass_kwarg_validation
38
38
from mplfinance ._arg_validators import _hlines_validator , _vlines_validator
@@ -304,9 +304,7 @@ def _valid_plot_kwargs():
304
304
'Description' : 'fill between specification as y-value, or sequence of' +
305
305
' y-values, or dict containing key "y1" plus any additional' +
306
306
' kwargs for `fill_between()`' ,
307
- 'Validator' : lambda value : _num_or_seq_of_num (value ) or
308
- (isinstance (value ,dict ) and 'y1' in value and
309
- _num_or_seq_of_num (value ['y1' ])) },
307
+ 'Validator' : _fill_between_validator },
310
308
311
309
'tight_layout' : { 'Default' : False ,
312
310
'Description' : 'True|False implement tight layout (minimal padding around Figure)' +
@@ -687,7 +685,7 @@ def plot( data, **kwargs ):
687
685
aptype = apdict ['type' ]
688
686
if aptype == 'ohlc' or aptype == 'candle' :
689
687
ax = _addplot_collections (panid ,panels ,apdict ,xdates ,config )
690
- _addplot_apply_supplements (ax ,apdict )
688
+ _addplot_apply_supplements (ax ,apdict , xdates )
691
689
else :
692
690
apdata = apdict ['data' ]
693
691
if isinstance (apdata ,list ) and not isinstance (apdata [0 ],(float ,int )):
@@ -700,24 +698,28 @@ def plot( data, **kwargs ):
700
698
for column in apdata :
701
699
ydata = apdata .loc [:,column ] if havedf else column
702
700
ax = _addplot_columns (panid ,panels ,ydata ,apdict ,xdates ,config )
703
- _addplot_apply_supplements (ax ,apdict )
701
+ _addplot_apply_supplements (ax ,apdict , xdates )
704
702
705
703
# fill_between is NOT supported for external_axes_mode
706
704
# (caller can easily call ax.fill_between() themselves).
707
705
if config ['fill_between' ] is not None and not external_axes_mode :
708
- fb = config ['fill_between' ]
709
- panid = config ['main_panel' ]
710
- if isinstance (fb ,dict ):
706
+ fblist = copy .deepcopy (config ['fill_between' ])
707
+ if _num_or_seq_of_num (fblist ):
708
+ fblist = [dict (y1 = fblist ),]
709
+ elif isinstance (fblist ,dict ):
710
+ fblist = [fblist ,]
711
+ if not _list_of_dict (fblist ):
712
+ raise TypeError ('Bad type for `fill_between` specifier.' )
713
+ for fb in fblist :
711
714
if 'x' in fb :
712
715
raise ValueError ('fill_between dict may not contain `x`' )
716
+ panid = config ['main_panel' ]
713
717
if 'panel' in fb :
714
718
panid = fb ['panel' ]
715
719
del fb ['panel' ]
716
- else :
717
- fb = dict (y1 = fb )
718
- fb ['x' ] = xdates
719
- ax = panels .at [panid ,'axes' ][0 ]
720
- ax .fill_between (** fb )
720
+ fb ['x' ] = xdates # add 'x' to caller's fb dict
721
+ ax = panels .at [panid ,'axes' ][0 ]
722
+ ax .fill_between (** fb )
721
723
722
724
# put the primary axis on one side,
723
725
# and the twinx() on the "other" side:
@@ -1045,7 +1047,7 @@ def _addplot_columns(panid,panels,ydata,apdict,xdates,config):
1045
1047
1046
1048
return ax
1047
1049
1048
- def _addplot_apply_supplements (ax ,apdict ):
1050
+ def _addplot_apply_supplements (ax ,apdict , xdates ):
1049
1051
if (apdict ['ylabel' ] is not None ):
1050
1052
ax .set_ylabel (apdict ['ylabel' ])
1051
1053
if apdict ['ylim' ] is not None :
@@ -1059,6 +1061,20 @@ def _addplot_apply_supplements(ax,apdict):
1059
1061
ax .set_yscale (yscale ,** ysd )
1060
1062
elif isinstance (ysd ,str ):
1061
1063
ax .set_yscale (ysd )
1064
+ # added by Wen
1065
+ if "fill_between" in apdict and apdict ['fill_between' ] is not None :
1066
+ # deep copy because mplfinance code sometimes modifies the fill_between dict
1067
+ fblist = copy .deepcopy (apdict ['fill_between' ])
1068
+ if isinstance (fblist ,dict ):
1069
+ fblist = [fblist ,]
1070
+ if _list_of_dict (fblist ):
1071
+ for fb in fblist :
1072
+ if 'x' in fb :
1073
+ raise ValueError ('fill_between dict may not contain `x`' )
1074
+ fb ['x' ] = xdates # add 'x' to caller's fb dict
1075
+ ax .fill_between (** fb )
1076
+ else :
1077
+ raise ValueError ('Invalid addplot fill_between: must be `dict` or `list of dict`' )
1062
1078
1063
1079
def _set_ylabels_side (ax_pri ,ax_sec ,primary_on_right ):
1064
1080
# put the primary axis on one side,
@@ -1234,6 +1250,10 @@ def _valid_addplot_kwargs():
1234
1250
" style\' s marketcolors). For addplot `type='ohlc'`" +
1235
1251
" and type='candle'" ,
1236
1252
'Validator' : lambda value : _is_marketcolor_object (value ) },
1253
+ 'fill_between' : { 'Default' : None , # added by Wen
1254
+ 'Description' : " fill region" ,
1255
+ 'Validator' : _fill_between_validator },
1256
+
1237
1257
}
1238
1258
1239
1259
_validate_vkwargs_dict (vkwargs )
0 commit comments