Python Plotly - Multiple dropdown plots, each of which have subplots

10,856

Solution 1

Naren is right, only one subplots figure needs to be made. And to create the drop down menu effect, you just need to add two traces to the same position like this..

fig.append_trace(trace1, 1, 1)
fig.append_trace(trace2, 1, 1)

Working code

import plotly.offline as py
import plotly.graph_objs as go
from plotly.offline import init_notebook_mode, iplot, plot
from plotly import tools
init_notebook_mode(connected=True)

x = [i for i in range(100)]
df_1 = pd.DataFrame([(i, 1+i) for i in range(100)], columns=["X", "Y"])
df_2 = pd.DataFrame([(i, i*i) for i in range(100)], columns=["X", "Y"])
labels = ["Plus one", "Square"]

### Create individual figures
# START
fig = tools.make_subplots(rows=1, cols=2)

trace1 = go.Bar(
                x=df_1.head(10).X,
                y=df_1.head(10).Y, 
                showlegend=False
            )
trace2 = go.Bar(
                x=df_2.head(10).X,
                y=df_2.head(10).Y, 
                showlegend=False
            )

fig.append_trace(trace1, 1, 1)
fig.append_trace(trace2, 1, 1)

trace1 = go.Bar(
                x=df_1.tail(10).X,
                y=df_1.tail(10).Y, 
                showlegend=False
            )
trace2 = go.Bar(
                x=df_2.tail(10).X,
                y=df_2.tail(10).Y, 
                showlegend=False
            )   
fig.append_trace(trace1, 1, 2)
fig.append_trace(trace2, 1, 2)
# END

### Create buttons for drop down menu
buttons = []
for i, label in enumerate(labels):
    visibility = [i==j for j in range(len(labels))]
    button = dict(
                 label =  label,
                 method = 'update',
                 args = [{'visible': visibility},
                     {'title': label}])
    buttons.append(button)

updatemenus = list([
    dict(active=-1,
         x=-0.15,
         buttons=buttons
    )
])

fig['layout']['title'] = 'Title'
fig['layout']['showlegend'] = False
fig['layout']['updatemenus'] = updatemenus

iplot(fig, filename='dropdown')

Thank you

So much to Naren Murali for your help! Editing your code a bit gave me the answer.

I was looking to create this.. enter image description here enter image description here

However your code was giving me these plots... enter image description here enter image description here

Solution 2

I think you have misunderstood the concept of subplots, the reason you are getting the error is because the make_subplots() function will create a layout object of its own, hence you are getting the error.

'layout' is not allowed in 'scatter'

Path To Error: ['data'][0]['layout']

The proper way to modify the layout of a subplot is to create the subplot and access the individual object properties and set them, like shown below.

updatemenus = list([
    dict(active=-1,
         x=-0.15,
         buttons=buttons
    )
])

fig['layout']['title'] = 'Title'
fig['layout']['showlegend'] = False
fig['layout']['updatemenus'] = updatemenus

Also you are creating a new subplot object each time the for loop is run, which is wrong. I am talking about the below line.

traces = []
for data in datas:
    fig = tools.make_subplots(rows=1, cols=2)

You need to assign it only once, please refer the below working example and try to implement the same method to your usecase.

import plotly.offline as py
import plotly.graph_objs as go
from plotly.offline import init_notebook_mode, iplot, plot
from plotly import tools
init_notebook_mode(connected=True)
df = pd.DataFrame([[1, 2], [3, 4], [5, 6], [7, 8]], columns=["A", "B"])
labels = ["A", "B"]
datas = [df, df]
### Create individual figures
# START
fig = tools.make_subplots(rows=2, cols=2)

for i, data in enumerate(datas):

    trace1 = go.Bar(
                    x=data.head(10).A,
                    y=data.head(10).B, 
                    showlegend=False
                )
    trace2 = go.Bar(
                    x=data.tail(10).A,
                    y=data.tail(10).B, 
                    showlegend=False
                )   
    fig.append_trace(trace1, i + 1, 1)
    fig.append_trace(trace2, i + 1, 2)

### Create buttons for drop down menu
buttons = []
for i, label in enumerate(labels):
    visibility = [i==j for j in range(len(labels))]
    button = dict(
                 label =  label,
                 method = 'update',
                 args = [{'visible': visibility},
                     {'title': label}])
    buttons.append(button)

updatemenus = list([
    dict(active=-1,
         x=-0.15,
         buttons=buttons
    )
])

fig['layout']['title'] = 'Title'
fig['layout']['showlegend'] = False
fig['layout']['updatemenus'] = updatemenus

iplot(fig, filename='dropdown')

Please do let me know if this solves your issue!

Share:
10,856
Patrick Stetz
Author by

Patrick Stetz

Updated on June 10, 2022

Comments

  • Patrick Stetz
    Patrick Stetz almost 2 years

    Problem

    I'm trying to combine two Python Plotly features. One of them is a drop down menu where a user can switch between plots (link to examples). The other feature is subplots.

    My attempt

    I have working code that uses the dropdown menu, but it doesn't have subplots. So I looked up how to create subplots here and I thought I could treat a tools.make_subplots figure the same way I treated a go.Box figure. I'm sorry if this seems naive, I'm actually fairly new to Plotly.

    However, this attempt is not working at all, I'm seeing two exceptions rise which I've put at the very bottom.

    My Working Code (no subplots)

    # NOTE: This code goes in between 'START' and 'END' below
    traces = []
    for data in datas:
        traces.append(go.Box(
                                x=data.index,
                                y=data.values, 
                                showlegend=False
                            ))
    

    My Subplot Code

    import plotly.offline as py
    import plotly.graph_objs as go
    from plotly.offline import init_notebook_mode, iplot, plot
    from plotly import tools
    init_notebook_mode(connected=True)
    
    ### Create individual figures
    # START
    traces = []
    for data in datas:
        fig = tools.make_subplots(rows=1, cols=2)
    
        trace1 = go.Box(
                        x=data.head(10).index,
                        y=data.head(10).values, 
                        showlegend=False
                    )
        trace2 = go.Box(
                        x=data.tail(10).index,
                        y=data.tail(10).values, 
                        showlegend=False
                    )   
        fig.append_trace(trace1, 1, 1)
        fig.append_trace(trace2, 1, 2)
        traces.append(fig)
    # END
    
    ### Create buttons for drop down menu
    buttons = []
    for i, label in enumerate(labels):
        visibility = [i==j for j in range(len(labels))]
        button = dict(
                     label =  label,
                     method = 'update',
                     args = [{'visible': visibility},
                         {'title': label}])
        buttons.append(button)
    
    updatemenus = list([
        dict(active=-1,
             x=-0.15,
             buttons=buttons
        )
    ])
    
    layout = dict(title='Title', 
                  showlegend=False,
                  updatemenus=updatemenus)
    
    fig = dict(data=traces, layout=layout)
    
    iplot(fig, filename='dropdown')
    

    Error 1

    'layout' is not allowed in 'scatter'

    Path To Error: ['data'][0]['layout']

    Error 2

    PlotlyError: Invalid 'figure_or_data' argument. Plotly will not be
    able to properly parse the resulting JSON. If you want to send this 'figure_or_data' to Plotly anyway (not recommended), you can set 'validate=False' as a plot option.

    Here's why you're seeing this error:

    'layout' is not allowed in 'scatter'

    ...

    NOTE: When I pass validate=False as a plot option, all I see is an empty plot with drop down menu functionality

  • Naren Murali
    Naren Murali almost 6 years
    Glad to be of assistance :)
  • Ritambhara Chauhan
    Ritambhara Chauhan about 4 years
    The code print both dataset on load. To show only one of the dataset, visibility of one trace can be set to false using visible=False
  • Guido
    Guido almost 4 years
    Not sure what happened, but the code snippet does not generate the same result as the screenshot in plotly 4.8
  • Patrick Stetz
    Patrick Stetz almost 4 years
    Hi Guido, can you please describe what you're seeing? The code still works, but there is a deprecation warning to use plotly.subplots instead of plotly.tools. The code works with this change however