Skip to content

Commit fc64079

Browse files
bar base and timeline
1 parent f37fd3c commit fc64079

File tree

8 files changed

+224
-141
lines changed

8 files changed

+224
-141
lines changed

CHANGELOG.md

+3-1
Original file line numberDiff line numberDiff line change
@@ -2,13 +2,14 @@
22
All notable changes to this project will be documented in this file.
33
This project adheres to [Semantic Versioning](http://semver.org/).
44

5-
65
## [4.9.0] - unreleased
76

87
### Added
98

109
- `px.NO_COLOR` constant to override wide-form color assignment in Plotly Express ([#2614](https://github.com/plotly/plotly.py/pull/2614))
1110
- `facet_row_spacing` and `facet_col_spacing` added to Plotly Express cartesian 2d functions ([#2614](https://github.com/plotly/plotly.py/pull/2614))
11+
- `base` added to Plotly Express `bar` and `bar_polar` functions
12+
- `plotly.express.timeline()` added as an official alternative to `plotly.figure_factories.create_gantt()`
1213

1314
### Fixed
1415

@@ -21,6 +22,7 @@ This project adheres to [Semantic Versioning](http://semver.org/).
2122

2223

2324

25+
2426
## [4.8.2] - 2020-06-26
2527

2628
### Updated

doc/python/gantt.md

+74-68
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,8 @@ jupyter:
55
text_representation:
66
extension: .md
77
format_name: markdown
8-
format_version: '1.1'
9-
jupytext_version: 1.1.1
8+
format_version: '1.2'
9+
jupytext_version: 1.4.2
1010
kernelspec:
1111
display_name: Python 3
1212
language: python
@@ -20,7 +20,7 @@ jupyter:
2020
name: python
2121
nbconvert_exporter: python
2222
pygments_lexer: ipython3
23-
version: 3.6.7
23+
version: 3.7.7
2424
plotly:
2525
description: How to make Gantt Charts in Python with Plotly. Gantt Charts use
2626
horizontal bars to represent the start and end times of tasks.
@@ -37,109 +37,100 @@ jupyter:
3737
A [Gantt chart](https://en.wikipedia.org/wiki/Gantt_chart) is a type of bar chart that illustrates a project schedule. The chart lists the tasks to be performed on the vertical axis, and time intervals on the horizontal axis. The width of the horizontal bars in the graph shows the duration of each activity.
3838

3939

40-
Gantt charts can be made using a [figure factory](/python/figure-factories/) as detailed in this page. See also the [bar charts examples](https://plotly.com/python/bar-charts/).
40+
### Gantt Charts and Timelines with plotly.express
4141

42+
[Plotly Express](/python/plotly-express/) is the easy-to-use, high-level interface to Plotly, which [operates on a variety of types of data](/python/px-arguments/) and produces [easy-to-style figures](/python/styling-plotly-express/). With `px.timeline` (*introduced in version 4.9*) each data point is represented as a horizontal bar with a start and end point specified as dates.
4243

43-
#### Simple Gantt Chart
44+
The `px.timeline` function by default sets the X-axis to be of `type=date`, so it can be configured like any [time-series chart](/python/time-series/).
45+
46+
Plotly Express also supports a [general-purpose `px.bar` function for bar charts](/python/bar-charts/).
4447

4548
```python
46-
import plotly.figure_factory as ff
49+
import plotly.express as px
50+
import pandas as pd
4751

48-
df = [dict(Task="Job A", Start='2009-01-01', Finish='2009-02-28'),
49-
dict(Task="Job B", Start='2009-03-05', Finish='2009-04-15'),
50-
dict(Task="Job C", Start='2009-02-20', Finish='2009-05-30')]
52+
df = pd.DataFrame([
53+
dict(Task="Job A", Start='2009-01-01', Finish='2009-02-28'),
54+
dict(Task="Job B", Start='2009-03-05', Finish='2009-04-15'),
55+
dict(Task="Job C", Start='2009-02-20', Finish='2009-05-30')
56+
])
5157

52-
fig = ff.create_gantt(df)
58+
fig = px.timeline(df, x_start="Start", x_end="Finish", y="Task")
59+
fig.update_yaxes(autorange="reversed") # otherwise tasks are listed from the bottom up
5360
fig.show()
5461
```
5562

56-
#### Index by Numeric Variable
63+
`px.timeline` supports [discrete color](/python/discrete-color/) as above, or [continuous color](/python/colorscales/) as follows.
5764

5865
```python
59-
import plotly.figure_factory as ff
66+
import plotly.express as px
67+
import pandas as pd
6068

61-
df = [dict(Task="Job A", Start='2009-01-01', Finish='2009-02-28', Complete=10),
62-
dict(Task="Job B", Start='2008-12-05', Finish='2009-04-15', Complete=60),
63-
dict(Task="Job C", Start='2009-02-20', Finish='2009-05-30', Complete=95)]
69+
df = pd.DataFrame([
70+
dict(Task="Job A", Start='2009-01-01', Finish='2009-02-28', Resource="Alex"),
71+
dict(Task="Job B", Start='2009-03-05', Finish='2009-04-15', Resource="Alex"),
72+
dict(Task="Job C", Start='2009-02-20', Finish='2009-05-30', Resource="Max")
73+
])
6474

65-
fig = ff.create_gantt(df, colors='Viridis', index_col='Complete', show_colorbar=True)
75+
fig = px.timeline(df, x_start="Start", x_end="Finish", y="Task", color="Resource")
76+
fig.update_yaxes(autorange="reversed")
6677
fig.show()
6778
```
6879

69-
#### Index by String Variable
70-
7180
```python
72-
import plotly.figure_factory as ff
73-
74-
df = [dict(Task="Job A", Start='2009-01-01', Finish='2009-02-01', Resource='Apple'),
75-
dict(Task="Job B", Start='2009-03-05', Finish='2009-04-15', Resource='Grape'),
76-
dict(Task="Job C", Start='2009-04-20', Finish='2009-09-30', Resource='Banana')]
81+
import plotly.express as px
82+
import pandas as pd
7783

78-
colors = ['#7a0504', (0.2, 0.7, 0.3), 'rgb(210, 60, 180)']
84+
df = pd.DataFrame([
85+
dict(Task="Job A", Start='2009-01-01', Finish='2009-02-28', Completion_pct=50),
86+
dict(Task="Job B", Start='2009-03-05', Finish='2009-04-15', Completion_pct=25),
87+
dict(Task="Job C", Start='2009-02-20', Finish='2009-05-30', Completion_pct=75)
88+
])
7989

80-
fig = ff.create_gantt(df, colors=colors, index_col='Resource', reverse_colors=True,
81-
show_colorbar=True)
90+
fig = px.timeline(df, x_start="Start", x_end="Finish", y="Task", color="Completion_pct")
91+
fig.update_yaxes(autorange="reversed")
8292
fig.show()
8393
```
8494

85-
#### Use a Dictionary for Colors
95+
It is also possible to have multiple bars on the same horizontal line, say by resource:
8696

87-
```python
88-
import plotly.figure_factory as ff
97+
*Note*: When setting `color` to the same value as `y`, `autorange` should not be set to `reverse`, so as to list the value of the Y axis in the same order as the legend entries.
8998

90-
df = [dict(Task="Job A", Start='2016-01-01', Finish='2016-01-02', Resource='Apple'),
91-
dict(Task="Job B", Start='2016-01-02', Finish='2016-01-04', Resource='Grape'),
92-
dict(Task="Job C", Start='2016-01-02', Finish='2016-01-03', Resource='Banana')]
99+
```python
100+
import plotly.express as px
101+
import pandas as pd
93102

94-
colors = dict(Apple='rgb(220, 0, 0)', Grape='rgb(170, 14, 200)', Banana=(1, 0.9, 0.16))
103+
df = pd.DataFrame([
104+
dict(Task="Job A", Start='2009-01-01', Finish='2009-02-28', Resource="Alex"),
105+
dict(Task="Job B", Start='2009-03-05', Finish='2009-04-15', Resource="Alex"),
106+
dict(Task="Job C", Start='2009-02-20', Finish='2009-05-30', Resource="Max")
107+
])
95108

96-
fig = ff.create_gantt(df, colors=colors, index_col='Resource', show_colorbar=True)
109+
fig = px.timeline(df, x_start="Start", x_end="Finish", y="Resource", color="Resource")
97110
fig.show()
98111
```
99112

100-
#### Use a Pandas Dataframe
113+
#### Deprecated Figure Factory
114+
115+
Prior to the introduction of `plotly.express.timeline()` in version 4.9, the recommended way to make Gantt charts was to use the now-deprecated `create_gantt()` [figure factory](/python/figure-factories/), as follows:
101116

102117
```python
103118
import plotly.figure_factory as ff
104119

105-
import pandas as pd
106-
df = pd.read_csv('https://raw.githubusercontent.com/plotly/datasets/master/gantt_example.csv')
120+
df = [dict(Task="Job A", Start='2009-01-01', Finish='2009-02-28'),
121+
dict(Task="Job B", Start='2009-03-05', Finish='2009-04-15'),
122+
dict(Task="Job C", Start='2009-02-20', Finish='2009-05-30')]
107123

108-
fig = ff.create_gantt(df, colors=['#333F44', '#93e4c1'], index_col='Complete',
109-
show_colorbar=True, bar_width=0.2, showgrid_x=True, showgrid_y=True)
124+
fig = ff.create_gantt(df)
110125
fig.show()
111126
```
112127

113-
#### Using Hours and Minutes in Times
114-
115-
```python
116-
import plotly.figure_factory as ff
128+
<!-- #region -->
129+
#### Group Tasks Together
117130

118-
df = [
119-
dict(Task='Morning Sleep', Start='2016-01-01', Finish='2016-01-01 6:00:00', Resource='Sleep'),
120-
dict(Task='Breakfast', Start='2016-01-01 7:00:00', Finish='2016-01-01 7:30:00', Resource='Food'),
121-
dict(Task='Work', Start='2016-01-01 9:00:00', Finish='2016-01-01 11:25:00', Resource='Brain'),
122-
dict(Task='Break', Start='2016-01-01 11:30:00', Finish='2016-01-01 12:00:00', Resource='Rest'),
123-
dict(Task='Lunch', Start='2016-01-01 12:00:00', Finish='2016-01-01 13:00:00', Resource='Food'),
124-
dict(Task='Work', Start='2016-01-01 13:00:00', Finish='2016-01-01 17:00:00', Resource='Brain'),
125-
dict(Task='Exercise', Start='2016-01-01 17:30:00', Finish='2016-01-01 18:30:00', Resource='Cardio'),
126-
dict(Task='Post Workout Rest', Start='2016-01-01 18:30:00', Finish='2016-01-01 19:00:00', Resource='Rest'),
127-
dict(Task='Dinner', Start='2016-01-01 19:00:00', Finish='2016-01-01 20:00:00', Resource='Food'),
128-
dict(Task='Evening Sleep', Start='2016-01-01 21:00:00', Finish='2016-01-01 23:59:00', Resource='Sleep')
129-
]
130-
131-
colors = dict(Cardio = 'rgb(46, 137, 205)',
132-
Food = 'rgb(114, 44, 121)',
133-
Sleep = 'rgb(198, 47, 105)',
134-
Brain = 'rgb(58, 149, 136)',
135-
Rest = 'rgb(107, 127, 135)')
136-
137-
fig = ff.create_gantt(df, colors=colors, index_col='Resource', title='Daily Schedule',
138-
show_colorbar=True, bar_width=0.8, showgrid_x=True, showgrid_y=True)
139-
fig.show()
140-
```
141131

142-
#### Group Tasks Together
132+
The following example shows how to use the now-deprecated `create_gantt()` [figure factory](/python/figure-factories/) to color tasks by a numeric variable.
133+
<!-- #endregion -->
143134

144135
```python
145136
import plotly.figure_factory as ff
@@ -162,7 +153,22 @@ fig = ff.create_gantt(df, colors=colors, index_col='Resource', show_colorbar=Tru
162153
fig.show()
163154
```
164155

156+
#### Color by Numeric Variable
157+
158+
The following example shows how to use the now-deprecated `create_gantt()` [figure factory](/python/figure-factories/) to color tasks by a numeric variable.
159+
160+
```python
161+
import plotly.figure_factory as ff
162+
163+
df = [dict(Task="Job A", Start='2009-01-01', Finish='2009-02-28', Complete=10),
164+
dict(Task="Job B", Start='2008-12-05', Finish='2009-04-15', Complete=60),
165+
dict(Task="Job C", Start='2009-02-20', Finish='2009-05-30', Complete=95)]
166+
167+
fig = ff.create_gantt(df, colors='Viridis', index_col='Complete', show_colorbar=True)
168+
fig.show()
169+
```
170+
165171
#### Reference
166172

167173

168-
For more info on `ff.create_gantt()`, see the [full function reference](https://plotly.com/python-api-reference/generated/plotly.figure_factory.create_gantt.html)
174+
For more info on `ff.create_gantt()`, see the [full function reference](https://plotly.com/python-api-reference/generated/plotly.figure_factory.create_gantt.html)

packages/python/plotly/plotly/express/__init__.py

+2
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
line_geo,
2929
area,
3030
bar,
31+
timeline,
3132
bar_polar,
3233
violin,
3334
box,
@@ -81,6 +82,7 @@
8182
"parallel_categories",
8283
"area",
8384
"bar",
85+
"timeline",
8486
"bar_polar",
8587
"violin",
8688
"box",

packages/python/plotly/plotly/express/_chart_types.py

+49
Original file line numberDiff line numberDiff line change
@@ -317,6 +317,7 @@ def bar(
317317
hover_data=None,
318318
custom_data=None,
319319
text=None,
320+
base=None,
320321
error_x=None,
321322
error_x_minus=None,
322323
error_y=None,
@@ -357,6 +358,53 @@ def bar(
357358
bar.__doc__ = make_docstring(bar, append_dict=_cartesian_append_dict)
358359

359360

361+
def timeline(
362+
data_frame=None,
363+
x_start=None,
364+
x_end=None,
365+
y=None,
366+
color=None,
367+
facet_row=None,
368+
facet_col=None,
369+
facet_col_wrap=0,
370+
facet_row_spacing=None,
371+
facet_col_spacing=None,
372+
hover_name=None,
373+
hover_data=None,
374+
custom_data=None,
375+
text=None,
376+
animation_frame=None,
377+
animation_group=None,
378+
category_orders={},
379+
labels={},
380+
color_discrete_sequence=None,
381+
color_discrete_map={},
382+
color_continuous_scale=None,
383+
range_color=None,
384+
color_continuous_midpoint=None,
385+
opacity=None,
386+
range_x=None,
387+
range_y=None,
388+
title=None,
389+
template=None,
390+
width=None,
391+
height=None,
392+
):
393+
"""
394+
In a timeline plot, each row of `data_frame` is represented as a rectangular
395+
mark on an x axis of type `date`, spanning from `x_start` to `x_end`.
396+
"""
397+
return make_figure(
398+
args=locals(),
399+
constructor="timeline",
400+
trace_patch=dict(textposition="auto", orientation="h"),
401+
layout_patch=dict(barmode="overlay"),
402+
)
403+
404+
405+
timeline.__doc__ = make_docstring(timeline)
406+
407+
360408
def histogram(
361409
data_frame=None,
362410
x=None,
@@ -847,6 +895,7 @@ def bar_polar(
847895
hover_name=None,
848896
hover_data=None,
849897
custom_data=None,
898+
base=None,
850899
animation_frame=None,
851900
animation_group=None,
852901
category_orders={},

packages/python/plotly/plotly/express/_core.py

+34-2
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@
1919

2020
# Declare all supported attributes, across all plot types
2121
direct_attrables = (
22-
["x", "y", "z", "a", "b", "c", "r", "theta", "size"]
22+
["base", "x", "y", "z", "a", "b", "c", "r", "theta", "size", "x_start", "x_end"]
2323
+ ["hover_name", "text", "names", "values", "parents", "wide_cross"]
2424
+ ["ids", "error_x", "error_x_minus", "error_y", "error_y_minus", "error_z"]
2525
+ ["error_z_minus", "lat", "lon", "locations", "animation_group"]
@@ -610,7 +610,8 @@ def configure_cartesian_axes(args, fig, orders):
610610
# Set x-axis titles and axis options in the bottom-most row
611611
x_title = get_decorated_label(args, args["x"], "x")
612612
for xaxis in fig.select_xaxes(row=1):
613-
xaxis.update(title_text=x_title)
613+
if "is_timeline" not in args:
614+
xaxis.update(title_text=x_title)
614615
set_cartesian_axis_opts(args, xaxis, "x", orders)
615616

616617
# Configure axis type across all x-axes
@@ -621,6 +622,9 @@ def configure_cartesian_axes(args, fig, orders):
621622
if "log_y" in args and args["log_y"]:
622623
fig.update_yaxes(type="log")
623624

625+
if "is_timeline" in args:
626+
fig.update_xaxes(type="date")
627+
624628
return fig.layout
625629

626630

@@ -1599,6 +1603,31 @@ def aggfunc_continuous(x):
15991603
return args
16001604

16011605

1606+
def process_dataframe_timeline(args):
1607+
"""
1608+
Massage input for bar traces for px.timeline()
1609+
"""
1610+
args["is_timeline"] = True
1611+
if args["x_start"] is None or args["x_end"] is None:
1612+
raise ValueError("Both x_start and x_end are required")
1613+
1614+
try:
1615+
x_start = pd.to_datetime(args["data_frame"][args["x_start"]])
1616+
x_end = pd.to_datetime(args["data_frame"][args["x_end"]])
1617+
except (ValueError, TypeError):
1618+
raise TypeError(
1619+
"Both x_start and x_end must refer to data convertible to datetimes."
1620+
)
1621+
1622+
# note that we are not adding any columns to the data frame here, so no risk of overwrite
1623+
args["data_frame"][args["x_end"]] = (x_end - x_start).astype("timedelta64[ms]")
1624+
args["x"] = args["x_end"]
1625+
del args["x_end"]
1626+
args["base"] = args["x_start"]
1627+
del args["x_start"]
1628+
return args
1629+
1630+
16021631
def infer_config(args, constructor, trace_patch, layout_patch):
16031632
attrs = [k for k in direct_attrables + array_attrables if k in args]
16041633
grouped_attrs = []
@@ -1801,6 +1830,9 @@ def make_figure(args, constructor, trace_patch=None, layout_patch=None):
18011830
args = build_dataframe(args, constructor)
18021831
if constructor in [go.Treemap, go.Sunburst] and args["path"] is not None:
18031832
args = process_dataframe_hierarchy(args)
1833+
if constructor == "timeline":
1834+
constructor = go.Bar
1835+
args = process_dataframe_timeline(args)
18041836

18051837
trace_specs, grouped_mappings, sizeref, show_colorbar = infer_config(
18061838
args, constructor, trace_patch, layout_patch

0 commit comments

Comments
 (0)