Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

docs: Add unsafe_update_figure doc #1058

Open
wants to merge 6 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions plugins/plotly-express/docs/sidebar.json
Original file line number Diff line number Diff line change
Expand Up @@ -133,6 +133,10 @@
{
"label": "Multiple axes",
"path": "multiple-axes.md"
},
{
"label": "`unsafe_update_figure` Chart Customization",
"path": "unsafe_update_figure.md"
}
]
}
Expand Down
107 changes: 107 additions & 0 deletions plugins/plotly-express/docs/unsafe-update-figure.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
# `unsafe_update_figure` Chart Customization

To customize a chart in a way that is not directly supported by Deephaven Plotly Express (`dx`), use the `unsafe_update_figure` parameter.
A Plotly [`Figure`](https://plotly.com/python/figure-structure/) object backs every `dx` chart, and `unsafe_update_figure` allows you to directly modify this object.
bmingles marked this conversation as resolved.
Show resolved Hide resolved

> [!WARNING]
> Update figure is marked "unsafe" because some modifications can entirely break your figure, and care must be taken.
> `dx` maps `Table` columns to an index of a trace within `Figure.data` which will break if the trace order changes. Do not remove traces. Add new traces at the end of the list.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would directly state why it is labelled "unsafe" at the start of this warning.

Update figure is marked "unsafe" because some modifications can entirely break your figure, and care must be taken to not ...

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

Copy link
Contributor

@bmingles bmingles Dec 16, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It might be helpful to explicitly call out the relationship of unsafe_update_figure to the plotly docs. I found it very confusing searching for update_figure in the plotly docs since I wrongfully assumed unsafe_update_figure was a wrapper for something with a similar name. I found the figure reference doc (https://plotly.com/python/reference/) to be a little more helpful in helping me find what I was looking for. It has more direct figure.update_xxx examples throughout.

I left a suggested change above.


`unsafe_update_figure` accepts a function that takes a Plotly `Figure` object as input and optionally returns a modified `Figure` object. If a `Figure` is not returned, it is assumed that the input `Figure` has been modified in place.

## Examples

### Bar Line

Add a line to bars in a bar plot with `update_traces`.

```python
import deephaven.plot.express as dx

tips = dx.data.tips()


def update(figure):
# Add a gray line to the bars
figure.update_traces(marker_line_width=3, marker_line_color="gray")


bar_lined_plot = dx.bar(tips, x="Day", unsafe_update_figure=update)
```

### Legend Location

Change the location of the legend to the bottom of the plot by updating the layout.

```python
import deephaven.plot.express as dx

tips = dx.data.tips()


def update(figure):
# Update the layout to move the legend to the bottom
# y is negative to move the legend outside the plot area
figure.update_layout(
legend=dict(orientation="h", yanchor="bottom", y=-0.3, xanchor="left", x=0.3)
)


legend_bottom_plot = dx.scatter(
tips, x="TotalBill", y="Tip", color="Day", unsafe_update_figure=update
)
```

### Vertical Line

Add a vertical line to a plot with `add_vline`.

```python
import deephaven.plot.express as dx

tips = dx.data.tips()


def update(figure):
# Add a dashed orange vertical line at x=20
figure.add_vline(x=20, line_width=3, line_dash="dash", line_color="orange")


scatter_vline_plot = dx.scatter(
tips, x="TotalBill", y="Tip", unsafe_update_figure=update
)
```

### Between Line Fill

Fill the area between lines in a line plot with `fill="tonexty"`.

```python
import deephaven.plot.express as dx

my_table = dx.data.stocks()

# subset data for just DOG transactions and add upper and lower bounds
dog_prices = my_table.where("Sym = `DOG`").update_view(
["UpperPrice = Price + 5", "LowerPrice = Price - 5"]
)


def update(figure):
# tonexty fills the area between the trace and the previous trace in the list
figure.update_traces(
fill="tonexty", fillcolor="rgba(123,67,0,0.3)", selector={"name": "LowerPrice"}
)
figure.update_traces(
fill="tonexty", fillcolor="rgba(123,67,0,0.3)", selector={"name": "Price"}
)


# Order matters for y since the fill is between the trace and the previous trace in the list
filled_line_plot = dx.line(
dog_prices,
x="Timestamp",
y=["UpperPrice", "Price", "LowerPrice"],
unsafe_update_figure=update,
)
```
Loading