I am trying to highlight timeperiods in Plotly. I seems like the best way is to use Shapes, like this, but in the real world you do not want to add each shape manually like in the example url. I am thinking a for loop is the best solution, but open for other (less computational) suggestions.
my script look like this:
np.random.seed(12345)
rows = 20
x = pd.Series(np.random.randn(rows),index=pd.date_range('1/1/2020', periods=rows)).cumsum()
df = pd.DataFrame({"index": x})
# add column showing what to shade
df["signal"] = df['index'] < 5
# plot index and highlight periods with Rectangle Shapes in Plotly
fig = px.line(df, x=df.index, y="index")
for row in df.iterrows():
if df['signal'] == False:
ply_shapes['shape_' + str(i)]=go.layout.Shape(type="rect",
x0=df.dato[i-1],
y0=0,
x1=df.dato[i],
y1=2000,
opacity=0.5,
layer="below"
)
lst_shapes=list(ply_shapes.values())
fig.update_layout(shapes=lst_shapes)
fig.show()
But this returns:
ValueError: The truth value of a Series is ambiguous. Use a.empty, a.bool(), a.item(), a.any() or a.all().
There are several reasons why your code snippet will not work.
1.
iterrows()
returns an iterator containing the index of each row and the data in each row as a seres. To use this, you would have to replacewith (for example):
Your error is raised bacause your are assigning a series of
True, False
withif df['signal'] == False:
instead of a single value likeif row[1]['signal'
will do.But only fixing this will not help you. At least not within the confinements of your snippet, because:
2.
dato
does not exist in your sample dataframe.Similar questions have been asked and answered before. But since the solution will be similar for very different sounding questions, I've decided to make a custom solution for your use case as well.
The best approach will in either case depend heavily on how you identify and assign your highlighted periods within your time series. The following example will use random data and identify periods to highlight depending on a threshold. Just like in your question.
My suggestion will boil down to a function
highLights()
that will take a a plotly figure, a pandas series and a threshold as input, as well as a few other details (take a look at the docstring).The
highLights()
function with some example input:Plot
Complete code: