0

当有一个函数生成指向 DynamicMap 的点的 NdOverlay 时,我遇到了一些非常奇怪的事情,其中​​该函数与面板小部件相关联(我认为面板小部件并不重要)。

下面的代码是一个产生预期行为的工作示例。每当您更改小部件值时,都会生成一个新图,其中包含两组重叠的点,具有不同的颜色和各自的图例条目。图片如下代码所示。

a_widget = pn.widgets.Select(name='A', options=[1,2,3,4])
b_widget = pn.widgets.IntSlider(name='B', start=10, end=20, value=10)
widget_box = pn.WidgetBox(a_widget, b_widget, align='center')

@pn.depends(a=a_widget.param.value, b=b_widget.param.value)
def get_points(a, b):
    return hv.NdOverlay({x: hv.Points(np.random.rand(10,10)) for x in range(1,3)})

points = hv.DynamicMap(get_points)

pn.Row(widget_box, points)

图片

下面显示的第二个示例旨在证明在某些情况下,您可能只想简单地返回一个空图,并且我在此示例中所做的方式与此示例中的方式相同:http:/ /holoviews.org/gallery/demos/bokeh/box_draw_roi_editor.html#bokeh-gallery-box-draw-roi-editor 当 时,该代码的结果是一个空图a == 1,但当a具有除 以外1的值时,结果相当奇怪,如代码下方的图像所示。

  1. 这些点都有相同的颜色
  2. 例如,在更改滑块时,某些点会被冻结并且永远不会更改,而上述工作示例中并非如此。
a_widget = pn.widgets.Select(name='A', options=[1,2,3,4])
b_widget = pn.widgets.IntSlider(name='B', start=10, end=20, value=10)
widget_box = pn.WidgetBox(a_widget, b_widget, align='center')

@pn.depends(a=a_widget.param.value, b=b_widget.param.value)
def get_points(a, b):
    if a == 1:
        return hv.NdOverlay({None: hv.Points([])})
    else:
        return hv.NdOverlay({x: hv.Points(np.random.rand(10,10)) for x in range(1,3)})

points = hv.DynamicMap(get_points)

pn.Row(widget_box, points)

图片

4

1 回答 1

1

虽然我无法解决观察到的问题NdOverlay,但可以借助Overlay.

由于b_widget您的代码中从未使用过,为了简单起见,我将其删除。

a_widget = pn.widgets.Select(name='A', options=[1,2,3,4])
widget_box = pn.WidgetBox(a_widget, align='center')

@pn.depends(a=a_widget.param.value)
def get_points(a):
    images = []
    if a == 3:
        images.append(hv.Points(np.random.rand(10,10), label='None'))
    else:
        for x in range(1,3):
            images.append(hv.Points(np.random.rand(10,10), label=str(x)))
    return hv.Overlay(images)

points = hv.DynamicMap(get_points)

pn.Row(widget_box, points)

NdOverlay文档中描述的如何使用 NdOverlay 的方式与您的方法不同,这可能是观察到问题的原因。


无论如何,为了缩小代码的哪一部分对观察到的问题负责,我删除了所有不需要重现它的代码。

为清楚起见,我重命名了 的值a,并且我还确保提供了 的起始值a

结果在测试代码时,if--else语句并不重要,所以我也删除了它。

为了确保变量的行为符合预期,我添加了一些print- 语句。

这给出了以下最小的可重现示例:

a_widget = pn.widgets.Select(name='A', value='Test', options=['Test','Test1', 'Test2'])

@pn.depends(a=a_widget.param.value)
def get_points(a):
    dict_ = {}
    dict_[str(a)] = hv.Points(np.random.rand(10,10))

    print(dict_)
    overlay = hv.NdOverlay(dict_)
    print(overlay)

    return overlay

points = hv.DynamicMap(get_points)

# using the server approach here to see the outpout of the
# print-statements
app = pn.Row(a_widget, points)
app.app() 

运行此代码并在选择小部件中选择不同的选项时,事实证明,一旦选择了其中一个选项,该选项就Test不会更新。Test1Test3

当我们像这样更改第一行中的默认值时

a_widget = pn.widgets.Select(name='A', value='Test2', options=['Test','Test1', 'Test2'])

现在Test2没有正确更新。

所以看起来这是一个DynamicMap使用问题NdOverlay。所以我建议你将此问题报告给开发人员(如果尚未完成),要么等待新版本发布,要么使用不同的方法(例如,如上所示)。

于 2019-12-05T20:33:23.603 回答