Python Plotly Express : How to conditionally fill an area plot?

标签 python plotly plotly-express

我想绘制时间序列面积图,其中正 (>= 0) 值用一种颜色填充,负 (< 0) 值用另一种颜色填充。

以这个例子为例:

import pandas as pd
import numpy as np
import plotly.express as px

df = pd.read_csv('https://raw.githubusercontent.com/plotly/datasets/master/finance-charts-apple.csv').assign(
    PnL = lambda x: x['AAPL.Close'] - 100
)
px.area(
    data_frame = df,
    x = 'Date',
    y = 'PnL',
    width = 500,
    height = 300
)

figure_1

我希望 PnL 低于 0 的部分填充为红色。

所以这就是我的尝试:

import pandas as pd
import numpy as np
import plotly.express as px

df = pd.read_csv('https://raw.githubusercontent.com/plotly/datasets/master/finance-charts-apple.csv').assign(
    PnL = lambda x: x['AAPL.Close'] - 100
)
df['sign'] = np.where(df['PnL'] >= 0, 'positive', 'negative')
px.area(
    data_frame = df,
    x = 'Date',
    y = 'PnL',
    color = 'sign',
    color_discrete_map = {
        'positive': 'steelblue',
        'negative': 'crimson'
    },
    width = 500,
    height = 300
)

但这给了我:

figure_2

这不是我要找的。执行此操作的最佳方法是什么?

最佳答案

这是我在有耐心的时候能做的最好的事情:

import plotly.graph_objects as go

mask = df['PnL'] >= 0
df['PnL_above'] = np.where(mask, df['PnL'], 0)
df['PnL_below'] = np.where(mask, 0, df['PnL'])

fig = go.Figure()
fig.add_trace(go.Scatter(x=df['Date'], y=df['PnL_above'], fill='tozeroy'))
fig.add_trace(go.Scatter(x=df['Date'], y=df['PnL_below'], fill='tozeroy'))

结果:

显然不理想,但可以帮助您完成大部分工作。两条迹线相交处有一些轻微的伪影,显然当值为零时线条颜色仍然可见。

通过在两条轨迹中添加mode='none',可以去掉线条,只渲染填充区域:

关于Python Plotly Express : How to conditionally fill an area plot?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69039149/

相关文章:

python - 在绘图条形图中更改颜色

python - 添加下拉菜单以绘制表达树状图

python - 如何在交互式绘图时摆脱最大递归深度错误?

Python3 & Opencv3 & Multiprocessing 引发系统错误

python - plotly 无法导入名称异常

python - Plotly:使用 plotly express line 时如何在 hoverinfo 中格式化日期?

python - plotly express facet plot 中的单轴标题

python - 在Python中从列表中的每个元素减去自身的有效方法

python - Session.run() 如何选择要运行的子图

python - 如何在 django 中的 python 脚本中接收表单输入?