I have a dataframe with amounts grouped by categories and I want to generate a pie chart containing the percentages per category. I am using the Bokeh library in Python.
My problem is that some percetange labels are not displayed properly in the pie chart as shown in the following image.
Here is the code that I use to produce the pie chart:
JavaScript
x
31
31
1
df['angle'] = df['paymentAmount'] / df['paymentAmount'].sum() * 2 * pi
2
if len(df.index) > 2:
3
df["color"] = GnBu[len(df.index)]
4
elif len(df.index) == 2:
5
df["color"] = ["steelblue", "skyblue"]
6
else:
7
df["color"] = ["steelblue"]
8
df["percentage"] = df["paymentAmount"] / df["paymentAmount"].sum()
9
df['percentage'] = df['percentage'].astype(float).map(lambda n: '{:.2%}'.format(n))
10
df["percentage"] = df['percentage'].astype(str)
11
df["percentage"] = df["percentage"].str.pad(35, side="left")
12
source = ColumnDataSource(df)
13
p = figure(width=figsize[0], height=figsize[1], toolbar_location=None,
14
tooltips=[("percentage", "@percentage")],
15
tools="hover", x_range=(-0.5, 1.0))
16
pie_chart = p.wedge(x=0, y=1, radius=0.4,
17
start_angle=cumsum('angle', include_zero=True), end_angle=cumsum('angle'),
18
line_color="white", fill_color='color', source=source)
19
labels = LabelSet(x=0, y=1, text='percentage', angle=cumsum('angle', include_zero=True), source=source,
20
render_mode='canvas')
21
p.add_layout(labels)
22
23
legend = Legend(items=[LegendItem(label=dict(field='category'), renderers=[pie_chart])],
24
location=(0, figsize[1] - 100))
25
p.add_layout(legend, 'right')
26
27
p.axis.axis_label = None
28
p.axis.visible = False
29
p.grid.grid_line_color = None
30
save(p)
31
Some data to reproduce the the pie chart above (tab separated)
JavaScript
1
5
1
category paymentAmount
2
Chemicals 52800.95001766206
3
Research 110878.70999269483
4
Construction 266189.2100121978
5
Any ideas how to solve this?
Thank you in advance!
Advertisement
Answer
I have found two useful StackOverflow posts to solve this:
- In LabelSet you have to set the angle as zero so as not to rotate the labels (Adding labels to bokeh pie chart wedge)
- You have to calculate the coordinates (x and y) of LabelSet for each item. For this I used this post: adding percentage label to Bokeh Pie chart
Therefore, I have modified your code as follows:
JavaScript
1
34
34
1
df['angle'] = df['paymentAmount'] / df['paymentAmount'].sum() * 2 * pi
2
value = list(df["paymentAmount"])
3
df["cumulative_angle"] = [(sum(value[0:i + 1]) - (item / 2)) / sum(value) * 2 * pi for i, item in enumerate(value)]
4
df['cos'] = np.cos(df['cumulative_angle']) * 0.3
5
df['sin'] = np.sin(df['cumulative_angle']) * 0.3
6
if len(df.index) > 2:
7
df["color"] = GnBu[len(df.index)]
8
elif len(df.index) == 2:
9
df["color"] = ["steelblue", "skyblue"]
10
else:
11
df["color"] = ["steelblue"]
12
df["percentage"] = df["paymentAmount"] / df["paymentAmount"].sum()
13
df['percentage'] = df['percentage'].astype(float).map(lambda n: '{:.2%}'.format(n))
14
df["percentage"] = df['percentage'].astype(str)
15
df["percentage"] = df["percentage"].str.pad(35, side="left")
16
source = ColumnDataSource(df)
17
p = figure(width=figsize[0], height=figsize[1], toolbar_location=None, x_range=(-1.0, 1.0), y_range=(-1.0, 1.0),
18
tooltips=[("percentage", "@percentage")], tools="hover")
19
pie_chart = p.wedge(x=0, y=0, radius=0.5,
20
start_angle=cumsum('angle', include_zero=True), end_angle=cumsum('angle'),
21
line_color="white", fill_color='color', source=source)
22
labels = LabelSet(x="cos", y="sin", y_offset=0, text='percentage', text_align="center", angle=0, source=source,
23
render_mode='canvas')
24
p.add_layout(labels)
25
26
legend = Legend(items=[LegendItem(label=dict(field="category"), renderers=[pie_chart])],
27
location=(0, figsize[1] - 100))
28
p.add_layout(legend, 'right')
29
30
p.axis.axis_label = None
31
p.axis.visible = False
32
p.grid.grid_line_color = None
33
save(p)
34
The produced pie chart is: