12

I'm trying to create a labeled graph using networkx but am having trouble getting the nodes and labels to turn out correctly. In short, the labels don't line up over the right nodes and there are some nodes which have no edges when displayed.

First I created a graph, added nodes and edges, then added labels.

The graph data comes from a pandas DataFrame object with two columns, employee and manager names:

                emp_name             mgr_name
0        Marianne Becker                 None
1            Evan Abbott      Marianne Becker
2               Jay Page      Marianne Becker
3             Seth Reese      Marianne Becker
4         Maxine Collier      Marianne Becker

...

Each node is a name and the edges are the mgr_name to emp_name relationship.

My graph code:

import networkx as nx
G=nx.DiGraph()

#set layout
pos=nx.spring_layout(G)

#add nodes
G.add_nodes_from(df.emp_name)
G.nodes()
G.add_node('None')

#create tuples for edges
subset = df[['mgr_name','emp_name']]
tuples = [tuple(x) for x in subset.values]

#add edges
G.add_edges_from(tuples)
G.number_of_edges()

#draw graph
import matplotlib.pyplot as plt
nx.draw(G, labels = True)
plt.show()

Ideally I would have a tree-like structure with employee names as the labels for each of the nodes.

Output image is enter image description here

Michele d'Amico
  • 22,111
  • 8
  • 69
  • 76
Marie
  • 121
  • 1
  • 1
  • 3
  • What is the response to `G.number_of_edges()`? It would be great if you could add an image --- I don't think you have enough 'reputation' to do that, but can you put it online somewhere and post a link? I can't see any obvious error. – Joel Apr 20 '15 at 04:44
  • Was this sorted after all? – A_A Apr 27 '15 at 10:06

1 Answers1

17

Networkx has a number of functions to draw graphs but also allow the user fine control over the whole process.

draw is basic and its docstring specifically mentions:

Draw the graph as a simple representation with no nodeabels or edge labels and using the full Matplotlib figure areas labels by default. See draw_networkx() for more fatured drawing that allows title, axis labels

The functions prefixed by draw_networkx followed by edges, nodes, edge_labels and edge_nodes allow finer control over the whole drawing process.

Your example worked fine when using draw_networkx.

In addition, if you are looking for an output that resembles an organogram, I would suggest the use of graphviz through networkx. Graphviz's dot is ideal for this kind of diagrams (please also see this for dot).

In what follows, I have tried to modify your code slightly to demonstrate the use of both functions:

import networkx as nx
import matplotlib.pyplot as plt
import pandas

#Build the dataset
df = pandas.DataFrame({'emp_name':pandas.Series(['Marianne Becker', 'Evan Abbott', 'Jay Page', 'Seth Reese', 'Maxine Collier'], index=[0,1,2,3,4]), 'mgr_name':pandas.Series(['None', 'Marianne Becker', 'Marianne Becker', 'Marianne Becker', 'Marianne Becker'], index = [0,1,2,3,4])})

#Build the graph
G=nx.DiGraph()   
G.add_nodes_from(df.emp_name)
G.nodes()
G.add_node('None')
#
#Over here, you are manually adding 'None' but in reality
#your nodes are the unique entries of the concatenated
#columns, i.e. emp_name, mgr_name. You could achieve this by
#doing something like
#
#G.add_nodes_from(list(set(list(D.emp_name.values) + list(D.mgr_name.values))))
#
# Which does exactly that, retrieves the contents of the two columns
#concatenates them and then selects the unique names by turning the
#combined list into a set.

#Add edges
subset = df[['mgr_name','emp_name']]
tuples = [tuple(x) for x in subset.values] 
G.add_edges_from(tuples)
G.number_of_edges()

#Perform Graph Drawing
#A star network  (sort of)
nx.draw_networkx(G)
plt.show()
t = raw_input()
#A tree network (sort of)
nx.draw_graphviz(G, prog = 'dot')
plt.show()

You could also try using graphviz's dot from the command line directly, by saving your networkx network via nx.write_dot. To do this:

From within your python script:

nx.write_dot(G, 'test.dot')

After this, from your (linux) command line and assuming that you have graphviz installed:

dot test.dot -Tpng>test_output.png
feh test_output.png #Feh is just an image viewer.
firefox test_output.png & #In case you don't have feh installed.

For a more typical organogram format, you can force orthogonal edge routing by

dot test.dot -Tpng -Gsplines=ortho>test_output.png

Finally, here are the outputs

Output of draw_networkx Output of <code>draw_networkx</code>

Output of draw_graphviz Output of <code>draw_graphviz</code>

Output of dot without orthogonal edges Output of <code>dot</code> without orthogonal edges

Output of dot with orthogonal edges Output of <code>dot</code> with orthogonal edges

Hope this helps.

kasparg
  • 428
  • 3
  • 14
A_A
  • 2,326
  • 18
  • 27
  • This is great. Do you know how to plot labels on the sides? I have this problem: https://stackoverflow.com/q/49368341/4421575 and the labels cannot be read because they are just over the nodes. – Lucas Aimaretto Mar 19 '18 at 17:35