I have a upload filed which should trigger the callback to populate the options of a dropdown with values from a row of an uploaded xlsx/csv.
Unfortunately the callback does not get called. When I change the output of the callback from options of the dropdown to the children of some div the callback gets called.
I have seen an example where the uploaded file first gets rendered into a table and then a callback takes the values from the table to populate the dropdown, but this will not work for me as the browser freezes trying to render a table because my files are so large.
import base64
import datetime
import io
from dash.dependencies import Input, Output, State
import dash
import dash_core_components as dcc
import dash_html_components as html
import dash_table_experiments as dt
import pandas as pd
from builtins import str
external_stylesheets = ['https://codepen.io/chriddyp/pen/bWLwgP.css']
app = dash.Dash(__name__, external_stylesheets=external_stylesheets)
df = None
cols = None
app.layout = html.Div(children=[
html.H1(children='AK2Viz'),
html.Div(children='''
Large Data Visualizer
'''),
dcc.RadioItems(
id='separator',
options=[
{'label': 'Semicolon', 'value': ';'},
{'label': 'Comma or Excel', 'value': ','},
],
value=',',
labelStyle={'display': 'inline-block'}),
dcc.Upload(
id='upload-data',
children=html.Div([
'Drag and Drop or ',
html.A('Select Files')
]),
style={
'width': '50%',
'height': '60px',
'lineHeight': '60px',
'borderWidth': '1px',
'borderStyle': 'dashed',
'borderRadius': '5px',
'textAlign': 'center',
'margin': '10px'
},
multiple=True),
dcc.Dropdown(
id="select_column",
options=[{}
#({}) if ( df is None ) else
#( {'label': df.iloc[[0][i]], 'value': i} for i in df.iloc[[0]] )
],
searchable=True),
html.Div(id='output-container',
children="helper div"),
html.Div(id='col_helper', children='col_helper')
])
@app.callback(
dash.dependencies.Output('output-container', 'children'),
[dash.dependencies.Input('select_column', 'value')])
def update_output(value):
return 'You have selected "{}"'.format(value)
def parse_contents(contents, filename, date, separator):
content_type, content_string = contents.split(separator)
decoded = base64.b64decode(content_string)
print('decoded len: '+str(len(decoded)))
print('decoded raw: '+str(decoded[0:100]))
print('io.Bytes len: '+str(io.BytesIO(decoded).getbuffer().nbytes))
try:
if 'csv' in filename:
# Assume that the user uploaded a CSV file
df = pd.read_csv(
io.StringIO(decoded.decode('utf-8')))
elif 'xls' in filename:
# Assume that the user uploaded an excel file
df = pd.read_excel(io.BytesIO(decoded), sheet_name=1)
print('df.empty: '+str(df.empty))
print('row for drodown: '+str(df.iloc[[0]]))
print('len of row for drodown: '+str(len(df.iloc[[0]])))
except Exception as e:
print(e)
return html.Div([
'There was an error processing this file.'
])
return [{'label': i, 'value': i } for i in df.iloc[[0]]]
@app.callback(Output('select_column', 'options'),
[Input('upload-data', 'contents')],
[State('upload-data', 'filename'),
State('upload-data', 'last_modified'),
State('separator', 'value')])
def update_col_helper(list_of_contents, list_of_names, list_of_dates, separator):
if list_of_contents is not None:
#print('list_of_contents: '+str(list_of_contents))
print('size of contents: '+str(len(list_of_contents[0])))
print('list_of_names: '+str(list_of_names))
print('list_of_dates: '+str(list_of_dates))
print('seperator: '+separator)
options = [
parse_contents(c, n, d, s) for c, n, d, s in
zip(list_of_contents, list_of_names, list_of_dates, separator)]
print("options variable: "+str(options))
return options
if __name__ == '__main__':
app.run_server(debug=True)