Spaces:
Running
Running
# NOTE: https://stackoverflow.com/questions/77062368/streamlit-bokeh-event-callback-to-get-clicked-values | |
# Taptool: https://docs.bokeh.org/en/2.4.2/docs/reference/models/tools.html#taptool | |
import streamlit as st | |
from bokeh.plotting import figure | |
from bokeh.plotting import figure, show | |
from bokeh.sampledata.periodic_table import elements | |
from bokeh.transform import dodge, factor_cmap | |
import streamlit as st | |
from bokeh.plotting import figure | |
from bokeh.models import ColumnDataSource, CustomJS, TapTool | |
from bokeh.sampledata.periodic_table import elements | |
from bokeh.transform import dodge, factor_cmap | |
periods = ["I", "II", "III", "IV", "V", "VI", "VII"] | |
groups = [str(x) for x in range(1, 19)] | |
df = elements.copy() | |
df["atomic mass"] = df["atomic mass"].astype(str) | |
df["group"] = df["group"].astype(str) | |
df["period"] = [periods[x-1] for x in df.period] | |
df = df[df.group != "-"] | |
df = df[df.symbol != "Lr"] | |
df = df[df.symbol != "Lu"] | |
cmap = { | |
"alkali metal" : "#a6cee3", | |
"alkaline earth metal" : "#1f78b4", | |
"metal" : "#d93b43", | |
"halogen" : "#999d9a", | |
"metalloid" : "#e08d49", | |
"noble gas" : "#eaeaea", | |
"nonmetal" : "#f1d4Af", | |
"transition metal" : "#599d7A", | |
} | |
TOOLTIPS = [ | |
("Name", "@name"), | |
("Atomic number", "@{atomic number}"), | |
("Atomic mass", "@{atomic mass}"), | |
("Type", "@metal"), | |
("CPK color", "$color[hex, swatch]:CPK"), | |
("Electronic configuration", "@{electronic configuration}"), | |
] | |
p = figure(title="Periodic Table (omitting LA and AC Series)", width=1000, height=450, | |
x_range=groups, y_range=list(reversed(periods)), | |
tools="hover,tap", toolbar_location=None, tooltips=TOOLTIPS) | |
# Convert DataFrame to ColumnDataSource | |
df["selected"] = False | |
source = ColumnDataSource(df) | |
r = p.rect("group", "period", 0.95, 0.95, source=source, fill_alpha=0.6, | |
legend_field="metal", | |
color=factor_cmap('metal', palette=list(cmap.values()), factors=list(cmap.keys())), | |
selection_color="firebrick", selection_alpha=0.9) | |
# r = p.rect("group", "period", 0.95, 0.95, source=df, fill_alpha=0.6, legend_field="metal", | |
# color=factor_cmap('metal', palette=list(cmap.values()), factors=list(cmap.keys()))) | |
text_props = dict(source=df, text_align="left", text_baseline="middle") | |
x = dodge("group", -0.4, range=p.x_range) | |
p.text(x=x, y="period", text="symbol", text_font_style="bold", **text_props) | |
p.text(x=x, y=dodge("period", 0.3, range=p.y_range), text="atomic number", | |
text_font_size="11px", **text_props) | |
p.text(x=x, y=dodge("period", -0.35, range=p.y_range), text="name", | |
text_font_size="7px", **text_props) | |
p.text(x=x, y=dodge("period", -0.2, range=p.y_range), text="atomic mass", | |
text_font_size="7px", **text_props) | |
p.text(x=["3", "3"], y=["VI", "VII"], text=["LA", "AC"], text_align="center", text_baseline="middle") | |
p.outline_line_color = None | |
p.grid.grid_line_color = None | |
p.axis.axis_line_color = None | |
p.axis.major_tick_line_color = None | |
p.axis.major_label_standoff = 0 | |
p.legend.orientation = "horizontal" | |
p.legend.location ="top_center" | |
p.hover.renderers = [r] # only hover element boxes | |
print(source.dataspecs()) | |
# Create a CustomJS callback | |
callback = CustomJS(args=dict(source=source), code=""" | |
var data = source.data; | |
var selected_elements = []; | |
for (var i = 0; i < data.symbol.length; i++) { | |
if (data.selected[i]) { // Corrected if statement with braces | |
selected_elements.push(data.symbol[i]); | |
} | |
} | |
console.log('Selected elements:', selected_elements); | |
document.dispatchEvent(new CustomEvent("selection_event", {detail: JSON.stringify(selected_elements)})); | |
""") | |
# yield j | |
# st.rerun() | |
# Add TapTool with the callback | |
tap_tool = TapTool() | |
p.add_tools(tap_tool) | |
p.js_on_event('tap', callback) | |
st.bokeh_chart(p, use_container_width=True) | |
# show(p) | |
selected_info = st.empty() | |
# Use session state to store selected elements | |
if 'selected_elements' not in st.session_state: | |
st.session_state.selected_elements = [] | |
st.markdown(""" | |
<script> | |
document.addEventListener('selection_event', function(e) { | |
var selected_elements = JSON.parse(e.detail); | |
window.parent.postMessage({ | |
type: 'streamlit:set_session_state', | |
data: { | |
selected_elements: selected_elements | |
} | |
}, '*'); | |
}); | |
</script> | |
""", unsafe_allow_html=True) | |
# Display selected elements | |
if st.session_state.selected_elements: | |
st.write("Selected Elements:") | |
for element in st.session_state.selected_elements: | |
st.write(f"{element['symbol']} ({element['name']}):") | |
st.write(f" Atomic Number: {element['atomic_number']}") | |
st.write(f" Atomic Mass: {element['atomic_mass']}") | |
st.write(f" Type: {element['metal']}") | |
st.write("---") | |
else: | |
st.write("No elements selected. Click on elements in the periodic table to select them.") | |
# st.rerun() | |
# Add a button to clear selection | |
if st.button("Clear Selection"): | |
st.session_state.selected_elements = [] | |
st.rerun() | |