"""
Defines the Param pane which converts Parameterized classes into a
set of widgets.
"""
from __future__ import annotations
import asyncio
import inspect
import itertools
import json
import os
import sys
import textwrap
import types
from collections import OrderedDict, defaultdict, namedtuple
from collections.abc import Callable
from contextlib import contextmanager
from functools import partial
from typing import (
TYPE_CHECKING, Any, ClassVar, Generator, List, Mapping, Optional, Tuple,
Type,
)
import param
from param.parameterized import classlist, discard_events, iscoroutinefunction
from .config import config
from .io import state
from .layout import (
Column, Panel, Row, Spacer, Tabs,
)
from .pane.base import PaneBase, ReplacementPane
from .reactive import Reactive
from .util import (
abbreviated_repr, eval_function, full_groupby, fullpath, get_method_owner,
is_parameterized, param_name, recursive_parameterized,
)
from .viewable import Layoutable, Viewable
from .widgets import (
ArrayInput, Button, Checkbox, ColorPicker, DataFrame, DatePicker,
DateRangeSlider, DatetimeInput, DatetimeRangeSlider, DiscreteSlider,
FileSelector, FloatInput, FloatSlider, IntInput, IntSlider, LiteralInput,
MultiSelect, RangeSlider, Select, StaticText, Tabulator, TextInput, Toggle,
Widget,
)
from .widgets.button import _ButtonBase
if TYPE_CHECKING:
from bokeh.document import Document
from bokeh.model import Model
from pyviz_comms import Comm
def SingleFileSelector(pobj: param.Parameter) -> Type[Widget]:
"""
Determines whether to use a TextInput or Select widget for FileSelector
"""
if pobj.path:
return Select
else:
return TextInput
def LiteralInputTyped(pobj: param.Parameter) -> Type[Widget]:
if isinstance(pobj, param.Tuple):
return type(str('TupleInput'), (LiteralInput,), {'type': tuple})
elif isinstance(pobj, param.Number):
return type(str('NumberInput'), (LiteralInput,), {'type': (int, float)})
elif isinstance(pobj, param.Dict):
return type(str('DictInput'), (LiteralInput,), {'type': dict})
elif isinstance(pobj, param.List):
return type(str('ListInput'), (LiteralInput,), {'type': list})
return LiteralInput
def DataFrameWidget(pobj: param.DataFrame) -> Type[Widget]:
if 'panel.models.tabulator' in sys.modules:
return Tabulator
else:
return DataFrame
[docs]@contextmanager
def set_values(*parameterizeds, **param_values):
"""
Temporarily sets parameter values to the specified values on all
supplied Parameterized objects.
Arguments
---------
parameterizeds: tuple(param.Parameterized)
Any number of parameterized objects.
param_values: dict
A dictionary of parameter names and temporary values.
"""
old = []
for parameterized in parameterizeds:
old_values = {p: getattr(parameterized, p) for p in param_values}
old.append((parameterized, old_values))
parameterized.param.update(**param_values)
try:
yield
finally:
for parameterized, old_values in old:
parameterized.param.update(**old_values)
[docs]class Param(PaneBase):
"""
Param panes render a Parameterized class to a set of widgets which
are linked to the parameter values on the class.
"""
display_threshold = param.Number(default=0, precedence=-10, doc="""
Parameters with precedence below this value are not displayed.""")
default_layout = param.ClassSelector(default=Column, class_=Panel,
is_instance=False)
default_precedence = param.Number(default=1e-8, precedence=-10, doc="""
Precedence value to use for parameters with no declared
precedence. By default, zero predecence is available for
forcing some parameters to the top of the list, and other
values above the default_precedence values can be used to sort
or group parameters arbitrarily.""")
expand = param.Boolean(default=False, doc="""
Whether parameterized subobjects are expanded or collapsed on
instantiation.""")
expand_button = param.Boolean(default=None, doc="""
Whether to add buttons to expand and collapse sub-objects.""")
expand_layout = param.Parameter(default=Column, doc="""
Layout to expand sub-objects into.""")
height = param.Integer(default=None, bounds=(0, None), doc="""
Height of widgetbox the parameter widgets are displayed in.""")
hide_constant = param.Boolean(default=False, doc="""
Whether to hide widgets of constant parameters.""")
initializer = param.Callable(default=None, doc="""
User-supplied function that will be called on initialization,
usually to update the default Parameter values of the
underlying parameterized object.""")
name = param.String(default='', doc="""
Title of the pane.""")
parameters = param.List(default=[], allow_None=True, doc="""
If set this serves as a whitelist of parameters to display on
the supplied Parameterized object.""")
show_labels = param.Boolean(default=True, doc="""
Whether to show labels for each widget""")
show_name = param.Boolean(default=True, doc="""
Whether to show the parameterized object's name""")
sort = param.ClassSelector(default=False, class_=(bool, Callable), doc="""
If True the widgets will be sorted alphabetically by label.
If a callable is provided it will be used to sort the Parameters,
for example lambda x: x[1].label[::-1] will sort by the reversed
label.""")
width = param.Integer(default=None, allow_None=True, bounds=(0, None), doc="""
Width of widgetbox the parameter widgets are displayed in.""")
widgets = param.Dict(doc="""
Dictionary of widget overrides, mapping from parameter name
to widget class.""")
mapping: ClassVar[Mapping[param.Parameter, Widget | Callable[[param.Parameter], Widget]]] = {
param.Action: Button,
param.Array: ArrayInput,
param.Boolean: Checkbox,
param.CalendarDate: DatePicker,
param.Color: ColorPicker,
param.Date: DatetimeInput,
param.DateRange: DatetimeRangeSlider,
param.CalendarDateRange: DateRangeSlider,
param.DataFrame: DataFrameWidget,
param.Dict: LiteralInputTyped,
param.FileSelector: SingleFileSelector,
param.Filename: TextInput,
param.Foldername: TextInput,
param.Integer: IntSlider,
param.List: LiteralInputTyped,
param.MultiFileSelector: FileSelector,
param.ListSelector: MultiSelect,
param.Number: FloatSlider,
param.ObjectSelector: Select,
param.Parameter: LiteralInputTyped,
param.Range: RangeSlider,
param.Selector: Select,
param.String: TextInput,
}
if hasattr(param, 'Event'):
mapping[param.Event] = Button
_ignored_refs: ClassVar[Tuple[str]] = ('object',)
_linkable_properties: ClassVar[Tuple[str]] = ()
_rerender_params: ClassVar[List[str]] = []
_unpack: ClassVar[bool] = True
def __init__(self, object=None, **params):
if isinstance(object, param.Parameter):
if 'show_name' not in params:
params['show_name'] = False
params['parameters'] = [object.name]
object = object.owner
if isinstance(object, param.parameterized.Parameters):
object = object.cls if object.self is None else object.self
if 'parameters' not in params and object is not None:
params['parameters'] = [p for p in object.param if p != 'name']
self._explicit_parameters = False
else:
self._explicit_parameters = object is not None
if object and 'name' not in params:
params['name'] = param_name(object.name)
super().__init__(object, **params)
self._updating = []
# Construct Layout
kwargs = {p: v for p, v in self.param.values().items()
if p in Layoutable.param and v is not None}
self._widget_box = self.default_layout(**kwargs)
layout = self.expand_layout
if isinstance(layout, Panel):
self._expand_layout = layout
self.layout = self._widget_box
elif isinstance(self._widget_box, layout):
self.layout = self._expand_layout = self._widget_box
elif isinstance(layout, type) and issubclass(layout, Panel):
self.layout = self._expand_layout = layout(self._widget_box, **kwargs)
else:
raise ValueError('expand_layout expected to be a panel.layout.Panel'
'type or instance, found %s type.' %
type(layout).__name__)
self.param.watch(self._update_widgets, [
'object', 'parameters', 'name', 'display_threshold', 'expand_button',
'expand', 'expand_layout', 'widgets', 'show_labels', 'show_name',
'hide_constant'])
self._update_widgets()
def __repr__(self, depth=0):
cls = type(self).__name__
obj_cls = type(self.object).__name__
params = [] if self.object is None else list(self.object.param)
parameters = [k for k in params if k != 'name']
params = []
for p, v in sorted(self.param.values().items()):
if v == self.param[p].default: continue
elif v is None: continue
elif isinstance(v, str) and v == '': continue
elif p == 'object' or (p == 'name' and (v.startswith(obj_cls) or v.startswith(cls))): continue
elif p == 'parameters' and v == parameters: continue
try:
params.append('%s=%s' % (p, abbreviated_repr(v)))
except RuntimeError:
params.append('%s=%s' % (p, '...'))
obj = 'None' if self.object is None else '%s' % type(self.object).__name__
template = '{cls}({obj}, {params})' if params else '{cls}({obj})'
return template.format(cls=cls, params=', '.join(params), obj=obj)
#----------------------------------------------------------------
# Callback API
#----------------------------------------------------------------
@property
def _synced_params(self):
ignored_params = ['default_layout', 'loading', 'background']
return [p for p in Layoutable.param if p not in ignored_params]
def _update_widgets(self, *events):
parameters = []
for event in sorted(events, key=lambda x: x.name):
if event.name == 'object':
if isinstance(event.new, param.parameterized.Parameters):
# Setting object will trigger this method a second time
self.object = event.new.cls if event.new.self is None else event.new.self
return
if self._explicit_parameters:
parameters = self.parameters
elif event.new is None:
parameters = []
else:
parameters = [p for p in event.new.param if p != 'name']
if event.new is not None:
self.name = param_name(event.new.name)
if event.name == 'parameters':
if event.new is None:
self._explicit_parameters = False
if self.object is not None:
parameters = [p for p in self.object.param if p != 'name']
else:
self._explicit_parameters = True
parameters = [] if event.new == [] else event.new
if parameters != [] and parameters != self.parameters:
# Setting parameters will trigger this method a second time
self.parameters = parameters
return
for cb in list(self._internal_callbacks):
if cb.inst in self._widget_box.objects:
cb.inst.param.unwatch(cb)
self._internal_callbacks.remove(cb)
# Construct widgets
if self.object is None:
self._widgets = {}
else:
self._widgets = self._get_widgets()
alias = {'_title': 'name'}
widgets = [widget for p, widget in self._widgets.items()
if (self.object.param[alias.get(p, p)].precedence is None)
or (self.object.param[alias.get(p, p)].precedence >= self.display_threshold)]
self._widget_box.objects = widgets
if not (self.expand_button == False and not self.expand):
self._link_subobjects()
def _link_subobjects(self):
for pname, widget in self._widgets.items():
widgets = [widget] if isinstance(widget, Widget) else widget
if not any(is_parameterized(getattr(w, 'value', None)) or
any(is_parameterized(o) for o in getattr(w, 'options', []))
for w in widgets):
continue
if (isinstance(widgets, Row) and isinstance(widgets[1], Toggle)):
selector, toggle = (widgets[0], widgets[1])
else:
selector, toggle = (widget, None)
def toggle_pane(change, parameter=pname):
"Adds or removes subpanel from layout"
parameterized = getattr(self.object, parameter)
existing = [p for p in self._expand_layout.objects
if isinstance(p, Param) and
p.object in recursive_parameterized(parameterized)]
if not change.new:
self._expand_layout[:] = [
e for e in self._expand_layout.objects
if e not in existing
]
elif change.new:
kwargs = {k: v for k, v in self.param.values().items()
if k not in ['name', 'object', 'parameters']}
pane = Param(parameterized, name=parameterized.name,
**kwargs)
if isinstance(self._expand_layout, Tabs):
title = self.object.param[pname].label
pane = (title, pane)
self._expand_layout.append(pane)
def update_pane(change, parameter=pname):
"Adds or removes subpanel from layout"
layout = self._expand_layout
existing = [p for p in layout.objects if isinstance(p, Param)
and p.object is change.old]
if toggle:
toggle.disabled = not is_parameterized(change.new)
if not existing:
return
elif is_parameterized(change.new):
parameterized = change.new
kwargs = {k: v for k, v in self.param.values().items()
if k not in ['name', 'object', 'parameters']}
pane = Param(parameterized, name=parameterized.name,
**kwargs)
layout[layout.objects.index(existing[0])] = pane
else:
layout.remove(existing[0])
watchers = [selector.param.watch(update_pane, 'value')]
if toggle:
watchers.append(toggle.param.watch(toggle_pane, 'value'))
self._internal_callbacks += watchers
if self.expand:
if self.expand_button:
toggle.value = True
else:
toggle_pane(namedtuple('Change', 'new')(True))
@property
def _ordered_params(self):
params = [(p, pobj) for p, pobj in self.object.param.objects('existing').items()
if p in self.parameters or p == 'name']
if self.sort:
if callable(self.sort):
key_fn = self.sort
else:
key_fn = lambda x: x[1].label
sorted_params = sorted(params, key=key_fn)
sorted_params = [el[0] for el in sorted_params if (el[0] != 'name' or el[0] in self.parameters)]
return sorted_params
key_fn = lambda x: x[1].precedence if x[1].precedence is not None else self.default_precedence
sorted_precedence = sorted(params, key=key_fn)
filtered = [(k, p) for k, p in sorted_precedence]
groups = itertools.groupby(filtered, key=key_fn)
# Params preserve definition order in Python 3.6+
ordered_groups = [list(grp) for (_, grp) in groups]
ordered_params = [el[0] for group in ordered_groups for el in group
if (el[0] != 'name' or el[0] in self.parameters)]
return ordered_params
#----------------------------------------------------------------
# Model API
#----------------------------------------------------------------
def _rerender(self):
precedence = lambda k: self.object.param['name' if k == '_title' else k].precedence
params = self._ordered_params
if self.show_name:
params.insert(0, '_title')
widgets = []
for k in params:
if precedence(k) is None or precedence(k) >= self.display_threshold:
widgets.append(self._widgets[k])
self._widget_box.objects = widgets
def _rerender_widget(self, p_name):
watchers = []
for w in self._internal_callbacks:
if w.inst is self._widgets[p_name]:
w.inst.param.unwatch(w)
else:
watchers.append(w)
self._widgets[p_name] = self.widget(p_name)
self._rerender()
def _get_widgets(self):
"""Return name,widget boxes for all parameters (i.e., a property sheet)"""
# Format name specially
if self.expand_layout is Tabs:
widgets = []
elif self.show_name:
widgets = [('_title', StaticText(value='<b>{0}</b>'.format(self.name)))]
else:
widgets = []
widgets += [(pname, self.widget(pname)) for pname in self._ordered_params]
return OrderedDict(widgets)
def _get_model(
self, doc: Document, root: Optional[Model] = None,
parent: Optional[Model] = None, comm: Optional[Comm] = None
) -> Model:
model = self.layout._get_model(doc, root, parent, comm)
self._models[root.ref['id']] = (model, parent)
return model
def _cleanup(self, root: Model | None = None) -> None:
self.layout._cleanup(root)
super()._cleanup(root)
#----------------------------------------------------------------
# Public API
#----------------------------------------------------------------
[docs] @classmethod
def applies(cls, obj: Any) -> float | bool | None:
if isinstance(obj, param.parameterized.Parameters):
return 0.8
elif (is_parameterized(obj) or (isinstance(obj, param.Parameter) and obj.owner is not None)):
return 0.1
return False
@classmethod
def widget_type(cls, pobj):
ptype = type(pobj)
for t in classlist(ptype)[::-1]:
if t not in cls.mapping:
continue
wtype = cls.mapping[t]
if isinstance(wtype, types.FunctionType):
return wtype(pobj)
return wtype
[docs] def get_root(
self, doc: Optional[Document] = None, comm: Comm | None = None,
preprocess: bool = True
) -> Model:
root = super().get_root(doc, comm, preprocess)
ref = root.ref['id']
self._models[ref] = (root, None)
return root
[docs] def select(self, selector=None):
"""
Iterates over the Viewable and any potential children in the
applying the Selector.
Arguments
---------
selector: type or callable or None
The selector allows selecting a subset of Viewables by
declaring a type or callable function to filter by.
Returns
-------
viewables: list(Viewable)
"""
return super().select(selector) + self.layout.select(selector)
[docs]class ParamMethod(ReplacementPane):
"""
ParamMethod panes wrap methods on parameterized classes and
rerenders the plot when any of the method's parameters change. By
default ParamMethod will watch all parameters on the class owning
the method or can be restricted to certain parameters by annotating
the method using the param.depends decorator. The method may
return any object which itself can be rendered as a Pane.
"""
defer_load = param.Boolean(default=config.defer_load, doc="""
Whether to defer load until after the page is rendered.
Can be set as parameter or by setting panel.config.defer_load.""")
lazy = param.Boolean(default=False, doc="""
Whether to lazily evaluate the contents of the object
only when it is required for rendering.""")
loading_indicator = param.Boolean(default=config.loading_indicator, doc="""
Whether to show a loading indicator while the pane is updating.
Can be set as parameter or by setting panel.config.loading_indicator.""")
def __init__(self, object=None, **params):
super().__init__(object, **params)
self._async_task = None
self._evaled = not (self.lazy or self.defer_load)
self._link_object_params()
if object is not None:
self._validate_object()
if not self.defer_load:
self._replace_pane()
@param.depends('object', watch=True)
def _validate_object(self):
dependencies = getattr(self.object, '_dinfo', None)
if not dependencies or not dependencies.get('watch'):
return
fn_type = 'method' if type(self) is ParamMethod else 'function'
self.param.warning(
f"The {fn_type} supplied for Panel to display was declared "
f"with `watch=True`, which will cause the {fn_type} to be "
"called twice for any change in a dependent Parameter. "
"`watch` should be False when Panel is responsible for "
f"displaying the result of the {fn_type} call, while "
f"`watch=True` should be reserved for {fn_type}s that work "
"via side-effects, e.g. by modifying internal state of a "
"class or global state in an application's namespace."
)
#----------------------------------------------------------------
# Callback API
#----------------------------------------------------------------
@classmethod
def eval(self, function):
return eval_function(function)
async def _eval_async(self, awaitable):
if self._async_task:
self._async_task.cancel()
self._async_task = task = asyncio.current_task()
curdoc = state.curdoc
has_context = bool(curdoc.session_context) if curdoc else False
if has_context:
curdoc.on_session_destroyed(lambda context: task.cancel())
try:
if isinstance(awaitable, types.AsyncGeneratorType):
async for new_obj in awaitable:
self._update_inner(new_obj)
else:
self._update_inner(await awaitable)
except Exception as e:
if not curdoc or (has_context and curdoc.session_context):
raise e
finally:
self._async_task = None
self._inner_layout.loading = False
def _replace_pane(self, *args, force=False):
deferred = self.defer_load and not state.loaded
if not self._inner_layout.loading:
self._inner_layout.loading = bool(self.loading_indicator or deferred)
self._evaled |= force or not (self.lazy or deferred)
if not self._evaled:
return
try:
if self.object is None:
new_object = Spacer()
else:
new_object = self.eval(self.object)
if inspect.isawaitable(new_object) or isinstance(new_object, types.AsyncGeneratorType):
param.parameterized.async_executor(partial(self._eval_async, new_object))
return
elif isinstance(new_object, Generator):
for new_obj in new_object:
self._update_inner(new_obj)
else:
self._update_inner(new_object)
finally:
self._inner_layout.loading = False
def _update_pane(self, *events):
callbacks = []
for watcher in self._internal_callbacks:
obj = watcher.inst if watcher.inst is None else watcher.cls
if obj is self:
callbacks.append(watcher)
continue
obj.param.unwatch(watcher)
self._internal_callbacks = callbacks
self._link_object_params()
self._replace_pane()
def _link_object_params(self):
parameterized = get_method_owner(self.object)
params = parameterized.param.method_dependencies(self.object.__name__)
deps = params
def update_pane(*events):
# Update nested dependencies if parameterized object events
if any(is_parameterized(event.new) for event in events):
new_deps = parameterized.param.method_dependencies(self.object.__name__)
for p in list(deps):
if p in new_deps: continue
watchers = self._internal_callbacks
for w in list(watchers):
if (w.inst is p.inst and w.cls is p.cls and
p.name in w.parameter_names):
obj = p.cls if p.inst is None else p.inst
obj.param.unwatch(w)
watchers.remove(w)
deps.remove(p)
new_deps = [dep for dep in new_deps if dep not in deps]
for _, params in full_groupby(new_deps, lambda x: (x.inst or x.cls, x.what)):
p = params[0]
pobj = p.cls if p.inst is None else p.inst
ps = [_p.name for _p in params]
watcher = pobj.param.watch(update_pane, ps, p.what)
self._internal_callbacks.append(watcher)
for p in params:
deps.append(p)
self._replace_pane()
for _, params in full_groupby(params, lambda x: (x.inst or x.cls, x.what)):
p = params[0]
pobj = (p.inst or p.cls)
ps = [_p.name for _p in params]
if isinstance(pobj, Reactive) and self.loading_indicator:
props = {p: 'loading' for p in ps if p in pobj._linkable_params}
if props:
pobj.jslink(self._inner_layout, **props)
watcher = pobj.param.watch(update_pane, ps, p.what)
self._internal_callbacks.append(watcher)
def _get_model(
self, doc: Document, root: Optional[Model] = None,
parent: Optional[Model] = None, comm: Optional[Comm] = None
) -> Model:
if not self._evaled:
deferred = self.defer_load and not state.loaded
if deferred:
state.onload(partial(self._replace_pane, force=True))
self._replace_pane(force=not deferred)
return super()._get_model(doc, root, parent, comm)
#----------------------------------------------------------------
# Public API
#----------------------------------------------------------------
[docs] @classmethod
def applies(cls, obj: Any) -> float | bool | None:
return inspect.ismethod(obj) and isinstance(get_method_owner(obj), param.Parameterized)
@param.depends(config.param.defer_load, watch=True)
def _update_defer_load_default(default_value):
ParamMethod.param.defer_load.default = default_value
@param.depends(config.param.loading_indicator, watch=True)
def _update_loading_indicator_default(default_value):
ParamMethod.param.loading_indicator.default = default_value
[docs]class ParamFunction(ParamMethod):
"""
ParamFunction panes wrap functions decorated with the param.depends
decorator and rerenders the output when any of the function's
dependencies change. This allows building reactive components into
a Panel which depend on other parameters, e.g. tying the value of
a widget to some other output.
"""
priority: ClassVar[float | bool | None] = 0.6
_applies_kw: ClassVar[bool] = True
def _link_object_params(self):
deps = getattr(self.object, '_dinfo', {})
dep_params = list(deps.get('dependencies', [])) + list(deps.get('kw', {}).values())
if not dep_params and not self.lazy and not self.defer_load and not iscoroutinefunction(self.object):
fn = getattr(self.object, '__bound_function__', self.object)
fn_name = getattr(fn, '__name__', repr(self.object))
self.param.warning(
f"The function {fn_name!r} does not have any dependencies "
"and will never update. Are you sure you did not intend "
"to depend on or bind a parameter or widget to this function? "
"If not simply call the function before passing it to Panel. "
"Otherwise, when passing a parameter as an argument, "
"ensure you pass at least one parameter and reference the "
"actual parameter object not the current value, i.e. use "
"object.param.parameter not object.parameter."
)
grouped = defaultdict(list)
for dep in dep_params:
grouped[id(dep.owner)].append(dep)
for group in grouped.values():
pobj = group[0].owner
watcher = pobj.param.watch(self._replace_pane, [dep.name for dep in group])
if isinstance(pobj, Reactive) and self.loading_indicator:
props = {dep.name: 'loading' for dep in group
if dep.name in pobj._linkable_params}
if props:
pobj.jslink(self._inner_layout, **props)
self._internal_callbacks.append(watcher)
#----------------------------------------------------------------
# Public API
#----------------------------------------------------------------
[docs] @classmethod
def applies(cls, obj: Any, **kwargs) -> float | bool | None:
if isinstance(obj, types.FunctionType):
if hasattr(obj, '_dinfo'):
return True
if (
kwargs.get('defer_load') or
(cls.param.defer_load.default is None and config.defer_load) or
iscoroutinefunction(obj)
):
return True
return None
return False
class JSONInit(param.Parameterized):
"""
Callable that can be passed to Widgets.initializer to set Parameter
values using JSON. There are three approaches that may be used:
1. If the json_file argument is specified, this takes precedence.
2. The JSON file path can be specified via an environment variable.
3. The JSON can be read directly from an environment variable.
Here is an easy example of setting such an environment variable on
the commandline:
PARAM_JSON_INIT='{"p1":5}' jupyter notebook
This addresses any JSONInit instances that are inspecting the
default environment variable called PARAM_JSON_INIT, instructing it to set
the 'p1' parameter to 5.
"""
varname = param.String(default='PARAM_JSON_INIT', doc="""
The name of the environment variable containing the JSON
specification.""")
target = param.String(default=None, doc="""
Optional key in the JSON specification dictionary containing the
desired parameter values.""")
json_file = param.String(default=None, doc="""
Optional path to a JSON file containing the parameter settings.""")
def __call__(self, parameterized):
warnobj = param.main.param if isinstance(parameterized, type) else parameterized.param
param_class = (parameterized if isinstance(parameterized, type)
else parameterized.__class__)
target = self.target if self.target is not None else param_class.__name__
env_var = os.environ.get(self.varname, None)
if env_var is None and self.json_file is None: return
if self.json_file or env_var.endswith('.json'):
try:
fname = self.json_file if self.json_file else env_var
with open(fullpath(fname), 'r') as f:
spec = json.load(f)
except Exception:
warnobj.warning('Could not load JSON file %r' % spec)
else:
spec = json.loads(env_var)
if not isinstance(spec, dict):
warnobj.warning('JSON parameter specification must be a dictionary.')
return
if target in spec:
params = spec[target]
else:
params = spec
for name, value in params.items():
try:
parameterized.param.update(**{name:value})
except ValueError as e:
warnobj.warning(str(e))
def link_param_method(root_view, root_model):
"""
This preprocessor jslinks ParamMethod loading parameters to any
widgets generated from those parameters ensuring that the loading
indicator is enabled client side.
"""
methods = root_view.select(lambda p: isinstance(p, ParamMethod) and p.loading_indicator)
widgets = root_view.select(lambda w: isinstance(w, Widget) and getattr(w, '_param_pane', None) is not None)
for widget in widgets:
for method in methods:
for cb in method._internal_callbacks:
pobj = cb.cls if cb.inst is None else cb.inst
if widget._param_pane.object is pobj and widget._param_name in cb.parameter_names:
if isinstance(widget, DiscreteSlider):
w = widget._slider
else:
w = widget
if 'value' in w._linkable_params:
w.jslink(method._inner_layout, value='loading')
Viewable._preprocessing_hooks.insert(0, link_param_method)
__all__= (
"Param",
"ParamFunction",
"ParamMethod",
"set_values"
)