"""Defines valid style options, validation and utilities"""importnumpyasnpfrombokeh.core.propertiesimport(Angle,Color,DashPattern,FontSize,MarkerType,Percent,Size,)try:frommatplotlibimportcm,colorsexceptImportError:cm,colors=None,Nonefrom...core.optionsimportabbreviated_exceptionfrom...core.utilimportarraylike_typesfrom...util.transformimportdimfrom..utilimportCOLOR_ALIASES,RGB_HEX_REGEX,rgb2hex# Define shared style properties for bokeh plotsproperty_prefixes=['selection','nonselection','muted','hover']base_properties=['visible','muted']line_properties=['line_color','line_alpha','color','alpha','line_width','line_join','line_cap','line_dash']line_properties+=[f'{prefix}_{prop}'forpropinline_propertiesforprefixinproperty_prefixes]fill_properties=['fill_color','fill_alpha']fill_properties+=[f'{prefix}_{prop}'forpropinfill_propertiesforprefixinproperty_prefixes]text_properties=['text_font','text_font_size','text_font_style','text_color','text_alpha','text_align','text_baseline']legend_dimensions=['label_standoff','label_width','label_height','glyph_width','glyph_height','legend_padding','legend_spacing','click_policy']# Conversion between matplotlib and bokeh markersmarkers={'+':{'marker':'cross'},'x':{'marker':'cross','angle':np.pi/4},'s':{'marker':'square'},'d':{'marker':'diamond'},'^':{'marker':'triangle','angle':0},'>':{'marker':'triangle','angle':-np.pi/2},'v':{'marker':'triangle','angle':np.pi},'<':{'marker':'triangle','angle':np.pi/2},'1':{'marker':'triangle','angle':0},'2':{'marker':'triangle','angle':-np.pi/2},'3':{'marker':'triangle','angle':np.pi},'4':{'marker':'triangle','angle':np.pi/2},'o':{'marker':'circle'},'*':{'marker':'asterisk'},}
[docs]defmpl_to_bokeh(properties):""" Utility to process style properties converting any matplotlib specific options to their nearest bokeh equivalent. """new_properties={}fork,vinproperties.items():ifisinstance(v,dict):new_properties[k]=velifk=='s':new_properties['size']=velifk=='marker':new_properties.update(markers.get(v,{'marker':v}))elif(k=='color'ork.endswith('_color'))andnotisinstance(v,(dict,dim)):withabbreviated_exception():v=COLOR_ALIASES.get(v,v)ifisinstance(v,tuple):withabbreviated_exception():v=rgb2hex(v)new_properties[k]=velse:new_properties[k]=vnew_properties.pop('cmap',None)returnnew_properties
[docs]defvalidate(style,value,scalar=False):""" Validates a style and associated value. Arguments --------- style: str The style to validate (e.g. 'color', 'size' or 'marker') value: The style value to validate scalar: bool Returns ------- valid: boolean or None If validation is supported returns boolean, otherwise None """validator=get_validator(style)ifvalidatorisNone:returnNoneifisinstance(value,arraylike_types+(list,)):ifscalar:returnFalsereturnall(validator(v)forvinvalue)returnvalidator(value)
# Utilities
[docs]defrgba_tuple(rgba):""" Ensures RGB(A) tuples in the range 0-1 are scaled to 0-255. """ifisinstance(rgba,tuple):returntuple(int(c*255)ifi<3elsecfori,cinenumerate(rgba))else:returnCOLOR_ALIASES.get(rgba,rgba)
[docs]defexpand_batched_style(style,opts,mapping,nvals):""" Computes styles applied to a batched plot by iterating over the supplied list of style options and expanding any options found in the supplied style dictionary returning a data and mapping defining the data that should be added to the ColumnDataSource. """opts=sorted(opts,key=lambdax:xin['color','alpha'])applied_styles=set(mapping)style_data,style_mapping={},{}foroptinopts:if'color'inopt:alias='color'elif'alpha'inopt:alias='alpha'else:alias=Noneifoptnotinstyleoroptinmapping:continueelifopt==alias:ifaliasinapplied_styles:continueelif'line_'+aliasinapplied_styles:if'fill_'+aliasnotinopts:continueopt='fill_'+aliasval=style[alias]elif'fill_'+aliasinapplied_styles:opt='line_'+aliasval=style[alias]else:val=style[alias]else:val=style[opt]style_mapping[opt]={'field':opt}applied_styles.add(opt)if'color'inoptandisinstance(val,tuple):val=rgb2hex(val)style_data[opt]=[val]*nvalsreturnstyle_data,style_mapping