Skip to content

Commit 8f32b7d

Browse files
Merge pull request #2626 from plotly/timeline
bar base and timeline
2 parents efc6650 + 19a0544 commit 8f32b7d

File tree

9 files changed

+229
-144
lines changed

9 files changed

+229
-144
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/figure-factories.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -35,13 +35,12 @@ jupyter:
3535

3636
#### `plotly.figure_factory`
3737

38-
The `plotly.figure_factory` module contains dedicated functions for creating very specific types of plots that were at the time of their creation difficult to create with `plotly.graph_objects` and prior to the existence of [Plotly Express](/python/plotly-express/). As new functionality gets added to [Plotly.js](https://plotly.com/javascript/) and to Plotly Express, certain Figure Factories become unecessary and are therefore deprecated as "legacy", but remain in the module for backwards-compatibility reasons.
38+
The `plotly.figure_factory` module contains dedicated functions for creating very specific types of plots that were at the time of their creation difficult to create with [graph objects](/python/graph-objects/) and prior to the existence of [Plotly Express](/python/plotly-express/). As new functionality gets added to [Plotly.js](https://plotly.com/javascript/) and to Plotly Express, certain Figure Factories become unecessary and are therefore deprecated as "legacy", but remain in the module for backwards-compatibility reasons.
3939

4040
The following types of plots are still difficult to create with Graph Objects or Plotly Express and therefore the corresponding Figure Factories are *not* deprecated:
4141

4242
* [Annotated Heatmaps](/python/annotated-heatmap/)
4343
* [Dendrograms](/python/dendrogram/)
44-
* [Gantt Charts](/python/gantt/)
4544
* [Quiver Plots](/python/quiver-plots/)
4645
* [Streamline Plots](/python/streamline-plots/)
4746
* [Tables](/python/figure-factory-table/)
@@ -52,7 +51,8 @@ Deprecated "legacy" Figure Factories include:
5251

5352
* [County Choropleth Maps](/python/county-choropleth/), deprecated by regular [Choropleth maps with GeoJSON input](/python/choropleth-maps/)
5453
* [Distplots](/python/distplot/), mostly deprecated by [`px.histogram`](/python/histograms/)
54+
* [Gantt Charts](/python/gantt/), deprecated by [`px.timeline`](/python/gantt/)
5555

5656
#### Reference
5757

58-
For more information about the contents of `plotly.figure_factory`, including deprecated methods, please refer to our [API Reference documentation](https://plotly.com/python-api-reference/plotly.figure_factory.html).
58+
For more information about the contents of `plotly.figure_factory`, including deprecated methods, please refer to our [API Reference documentation](https://plotly.com/python-api-reference/plotly.figure_factory.html).

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={},

0 commit comments

Comments
 (0)