Skip to content
Snippets Groups Projects
1-samples_selection.py 16.7 KiB
Newer Older
from Packages import *
st.set_page_config(page_title="NIRS Utils", page_icon=":goat:", layout="wide")
from Modules import *

# HTML pour le bandeau "CEFE - CNRS"
# bandeau_html = """
# <div style="width: 100%; background-color: #4682B4; padding: 10px; margin-bottom: 10px;">
#   <h1 style="text-align: center; color: white;">CEFE - CNRS</h1>
# </div>
# """
# # Injecter le code HTML du bandeau
# st.markdown(bandeau_html, unsafe_allow_html=True)
add_header()

st.session_state["interface"] = st.session_state.get('interface')
if st.session_state["interface"] == 'simple':
    hide_pages("Predictions")

DIANE's avatar
DIANE committed
################################### I - Data Loading and Visualization ########################################
col2, col1 = st.columns([3, 1])
DIANE's avatar
DIANE committed
col1.header("Data Loading", divider='blue')
col2.header("Spectral Data Visualization", divider='blue')
## Preallocation of data structure
DIANE's avatar
DIANE committed
spectra = pd.DataFrame
meta_data = pd.DataFrame
selected_samples = pd.DataFrame
DIANE's avatar
DIANE committed
non_clustered = None
DIANE's avatar
DIANE committed
colnames = []
rownames = []
DIANE's avatar
DIANE committed
# loader for datafile
data_file = col1.file_uploader("Load NIRS Data", type=["csv","dx"], help=" :mushroom: select a csv matrix with samples as rows and lambdas as columns", key=5)
DIANE's avatar
DIANE committed
if data_file:
    # Retrieve the extension of the file
    test = data_file.name[data_file.name.find('.'):]
DIANE's avatar
DIANE committed
    ## Load .csv file
    if test== '.csv':
        with col1:
            # Select list for CSV delimiter
DIANE's avatar
DIANE committed
            psep = st.radio("Select csv separator - _detected_: " + str(find_delimiter('data/'+data_file.name)), options=[";", ","], index=[";", ","].index(str(find_delimiter('data/'+data_file.name))), key=9)
DIANE's avatar
DIANE committed
                # Select list for CSV header True / False
DIANE's avatar
DIANE committed
            phdr = st.radio("indexes column in csv? - _detected_: " + str(find_col_index('data/'+data_file.name)), options=["no", "yes"], index=["no", "yes"].index(str(find_col_index('data/'+data_file.name))), key=31)
            if phdr == 'yes':
                col = 0
            else:
                col = False
DIANE's avatar
DIANE committed
            imp = pd.read_csv(data_file, sep=psep, index_col=col)
            # spectra = col_cat(imp)[0]
            # meta_data = col_cat(imp)[1]
DIANE's avatar
DIANE committed
            spectra, md_df_st_ = col_cat(imp)
            st.success("The data have been loaded successfully", icon="")

DIANE's avatar
DIANE committed
    ## Load .dx file
    elif test == '.dx':
        # Create a temporary file to save the uploaded file
        with NamedTemporaryFile(delete=False, suffix=".dx") as tmp:
DIANE's avatar
DIANE committed
            tmp.write(data_file.read())
            tmp_path = tmp.name
            with col1:
DIANE's avatar
DIANE committed
                _, spectra, meta_data, md_df_st_ = read_dx(file = tmp_path)
DIANE's avatar
DIANE committed
                st.success("The data have been loaded successfully", icon="")
        os.unlink(tmp_path)
DIANE's avatar
DIANE committed
## Visualize spectra
if not spectra.empty:
DIANE's avatar
DIANE committed
    # retrieve columns name and rows name of spectra
    colnames = list(spectra.columns)
    rownames = [str(i) for i in list(spectra.index)]
    spectra.index = rownames

    with col2:
DIANE's avatar
DIANE committed
        if test =='.dx':
            if meta_data.loc[:,'xunits'][0] == '1/cm':
                lab = 'Wavenumber (1/cm)'
            else:
                lab = 'Wavelength (nm)'
            fig = plot_spectra(spectra, xunits = lab, yunits = meta_data.loc[:,'yunits'][0])
        else:
            fig = plot_spectra(spectra, xunits = 'Wavelength/Wavenumber', yunits = 'Signal intensity')

        st.pyplot(fig)
        fig.savefig("./Report/figures/Spectra_Plot.png")
DIANE's avatar
DIANE committed

############################## Exploratory data analysis ###############################
container2 = st.container(border=True)
container2.header("Exploratory Data Analysis-Multivariable Data Analysis", divider='blue')
scores, loadings, pc = st.columns([2, 3, 0.5])
influence, hotelling, qexp = st.columns([2, 2, 1])
DIANE's avatar
DIANE committed
st.header('Selected samples for chemical analysis', divider='blue')
DIANE's avatar
DIANE committed
selected_s, selected_samples_metd = st.columns([3, 3])
DIANE's avatar
DIANE committed
dim_red_methods=['', 'PCA','UMAP', 'NMF']  # List of dimensionality reduction algos
cluster_methods = ['', 'Kmeans','HDBSCAN', 'AP'] # List of clustering algos

dr_model = None # dimensionality reduction model
cl_model = None # clustering model
DIANE's avatar
DIANE committed
###### 1- Dimensionality reduction ######
DIANE's avatar
DIANE committed
t = pd.DataFrame # scores
p = pd.DataFrame # loadings
labels = []
if not spectra.empty:
    dim_red_method = pc.selectbox("Dimensionality reduction techniques: ", options = dim_red_methods, key = 37)
    clus_method = pc.selectbox("Clustering techniques: ", options = cluster_methods, key = 38)
DIANE's avatar
DIANE committed
    xc = standardize(spectra, center=True, scale=False)
DIANE's avatar
DIANE committed

    if dim_red_method == dim_red_methods[1]:
        dr_model = LinearPCA(xc, Ncomp=8)
    elif dim_red_method == dim_red_methods[2]:
        if not meta_data.empty:
DIANE's avatar
DIANE committed
            filter = md_df_st_.columns
            col = pc.selectbox('Supervised UMAP by:', options= filter, key=108)
DIANE's avatar
DIANE committed
            supervised = md_df_st_[col]
        else:
            supervised = None
DIANE's avatar
DIANE committed
        dr_model = Umap(numerical_data = MinMaxScale(spectra), cat_data = supervised)
DIANE's avatar
DIANE committed
    elif dim_red_method == dim_red_methods[3]:
DIANE's avatar
DIANE committed
        dr_model = Nmf(spectra, Ncomp= 3)
    if dr_model:
        axis1 = pc.selectbox("x-axis", options = dr_model.scores_.columns, index=0)
        axis2 = pc.selectbox("y-axis", options = dr_model.scores_.columns, index=1)
        axis3 = pc.selectbox("z-axis", options = dr_model.scores_.columns, index=2)
        t = pd.concat([dr_model.scores_.loc[:,axis1], dr_model.scores_.loc[:,axis2], dr_model.scores_.loc[:,axis3]], axis = 1)


DIANE's avatar
DIANE committed
###### II - clustering #######
if not t.empty:
DIANE's avatar
DIANE committed
    tcr = standardize(t)
        # Clustering
DIANE's avatar
DIANE committed
    # 1- K-MEANS Clustering
DIANE's avatar
DIANE committed
    if clus_method == cluster_methods[1]:
DIANE's avatar
DIANE committed
        cl_model = Sk_Kmeans(tcr, max_clusters = 25)
        ncluster = scores.number_input(min_value=2, max_value=25, value=cl_model.suggested_n_clusters_, label = 'Select the desired number of clusters')
DIANE's avatar
DIANE committed
        fig2 = px.scatter(cl_model.inertia_.T, y = 'inertia')
DIANE's avatar
DIANE committed
        scores.write(f"Suggested n_clusters : {cl_model.suggested_n_clusters_}")
        scores.plotly_chart(fig2,use_container_width=True)
        img = pio.to_image(fig2, format="png")
        with open("./Report/figures/Elbow.png", "wb") as f:
                f.write(img)    
DIANE's avatar
DIANE committed
        data, labels, clu_centers = cl_model.fit_optimal(nclusters = ncluster)
DIANE's avatar
DIANE committed

DIANE's avatar
DIANE committed
    # 2- HDBSCAN clustering
DIANE's avatar
DIANE committed
    elif clus_method == cluster_methods[2]:
DIANE's avatar
DIANE committed
        optimized_hdbscan = Hdbscan(np.array(t))
DIANE's avatar
DIANE committed
        all_labels, hdbscan_score, clu_centers = optimized_hdbscan.HDBSCAN_scores_
        labels = [f'cluster#{i+1}' if i !=-1 else 'Non clustered' for i in all_labels]
        non_clustered = np.where(np.array(labels) == 'Non clustered')
        clustered = np.where(np.array(labels) != 'Non clustered')[0]
        
        #st.write(optimized_hdbscan.non_clustered)
DIANE's avatar
DIANE committed
    # 3- Affinity propagation
DIANE's avatar
DIANE committed
    elif clus_method == cluster_methods[3]:
DIANE's avatar
DIANE committed
        cl_model = AP(X = tcr)
DIANE's avatar
DIANE committed
        data, labels, clu_centers = cl_model.fit_optimal_

DIANE's avatar
DIANE committed
###### III - Samples selection using the reduced data preentation ######
DIANE's avatar
DIANE committed
selec_strategy = ['center','random']
samples_df_chem = pd.DataFrame
selected_samples = []
selected_samples_idx = []

DIANE's avatar
DIANE committed

DIANE's avatar
DIANE committed
if labels:
DIANE's avatar
DIANE committed
    if clus_method:
            selection = scores.radio('Select samples selection strategy:',
                                    options = selec_strategy)
DIANE's avatar
DIANE committed
    if clus_method == cluster_methods[2]:
        tcr = tcr.iloc[clustered,:]

DIANE's avatar
DIANE committed
    if selection == selec_strategy[0]:
        # list samples at clusters centers - Use sklearn.metrics.pairwise_distances_argmin if you want more than 1 sample per cluster
        closest, _ = pairwise_distances_argmin_min(clu_centers, tcr)
        selected_samples_idx = list(closest)
DIANE's avatar
DIANE committed
    elif selection == selec_strategy[1]:
DIANE's avatar
DIANE committed
        selection_number = scores.number_input('How many samples per cluster?',
                                                min_value = 1, step=1, value = 3)
DIANE's avatar
DIANE committed
        for i in np.unique(labels):
DIANE's avatar
DIANE committed
            C = np.where(np.array(labels) == i)[0]
DIANE's avatar
DIANE committed
            if C.shape[0] >= selection_number:
DIANE's avatar
DIANE committed
                # scores.write(list(tcr.index)[labels== i])
DIANE's avatar
DIANE committed
                km2 = KMeans(n_clusters = selection_number)
                km2.fit(tcr.iloc[C,:])
                clos, _ = pairwise_distances_argmin_min(km2.cluster_centers_, tcr.iloc[C,:])
DIANE's avatar
DIANE committed
                selected_samples_idx.extend(tcr.iloc[C,:].iloc[list(clos)].index)
DIANE's avatar
DIANE committed
            else:
                selected_samples_idx.extend(tcr.iloc[C,:].index.to_list())
            # list indexes of selected samples for colored plot    

    if selected_samples_idx:
DIANE's avatar
DIANE committed
        if meta_data.empty:
            sam = pd.DataFrame({'name': spectra.index[selected_samples_idx],
                                'cluster':np.array(labels)[selected_samples_idx]},
                                index = selected_samples_idx)
        else:
            sam = meta_data.iloc[selected_samples_idx,:]
            sam.insert(loc=0, column='index', value=selected_samples_idx)
            sam.insert(loc=1, column='cluster', value=np.array(labels)[selected_samples_idx])
        
        sam.index = np.arange(len(selected_samples_idx))+1
        
        selected_s.write(f' The selected subset consists of {sam.shape[0]} samples')
DIANE's avatar
DIANE committed
        selected_s.write(sam)
DIANE's avatar
DIANE committed
        selected_s.checkbox("Include non clustered samples (for HDBSCAN clustering)")
        
DIANE's avatar
DIANE committed
################################      Plots visualization          ############################################
DIANE's avatar
DIANE committed
    ## Scores
if not t.empty:
    with scores:
DIANE's avatar
DIANE committed
        fig1, ((ax1, ax2),(ax3,ax4)) = plt.subplots(2,2)
        st.write('Scores plot')
        # scores plot with clustering
        if list(labels) and meta_data.empty:
DIANE's avatar
DIANE committed
            fig = px.scatter_3d(tcr, x=axis1, y=axis2, z = axis3, color = labels)
DIANE's avatar
DIANE committed
            sns.scatterplot(data = tcr, x = axis1, y =axis2 , hue = labels, ax = ax1)
            

        # scores plot with metadata
        elif len(list(labels)) == 0 and not meta_data.empty:
DIANE's avatar
DIANE committed
            filter = md_df_st_.columns
            col = st.selectbox('Color by:', options= filter)
DIANE's avatar
DIANE committed
                fig = px.scatter_3d(tcr, x=axis1, y=axis2, z = axis3)
DIANE's avatar
DIANE committed
                sns.scatterplot(data = tcr, x = axis1, y =axis2 , ax = ax1)
                sns.scatterplot(data = tcr, x = axis2, y =axis3 , ax = ax2)
DIANE's avatar
DIANE committed
                sns.scatterplot(data = tcr, x = axis1, y =axis3 , hue = list(map(str.lower,md_df_st_[col])), ax = ax3)
DIANE's avatar
DIANE committed
                fig = px.scatter_3d(tcr, x=axis1, y=axis2, z = axis3, color = list(map(str.lower,md_df_st_[col])) )
                sns.scatterplot(data = tcr, x = axis1, y =axis2 , hue = list(map(str.lower,md_df_st_[col])), ax = ax1)
                sns.scatterplot(data = tcr, x = axis2, y =axis3 , hue = list(map(str.lower,md_df_st_[col])), ax = ax2)
                sns.scatterplot(data = tcr, x = axis1, y =axis3 , hue = list(map(str.lower,md_df_st_[col])), ax = ax3)
        # color with scores and metadata
        elif len(list(labels)) > 0  and not meta_data.empty:
            if clus_method in cluster_methods[1:]:
                filter = ['None', clus_method]
DIANE's avatar
DIANE committed
                filter.extend(md_df_st_.columns)
DIANE's avatar
DIANE committed
                filter = md_df_st_.columns.insert(0,'None')
            col = st.selectbox('Color by:', options= filter)
                fig = px.scatter_3d(tcr, x=axis1, y=axis2, z = axis3)
DIANE's avatar
DIANE committed
                sns.scatterplot(data = tcr, x = axis1, y =axis2 , ax = ax1)
DIANE's avatar
DIANE committed
                fig = px.scatter_3d(tcr, x=axis1, y=axis2, z = axis3, color = labels)
DIANE's avatar
DIANE committed
                sns.scatterplot(data = tcr, x = axis1, y =axis2 , ax = ax1)
DIANE's avatar
DIANE committed
                fig = px.scatter_3d(tcr, x=axis1, y=axis2, z = axis3, color = list(map(str.lower,md_df_st_[col])))
                sns.scatterplot(data = tcr, x = axis1, y =axis2 , hue = list(map(str.lower,md_df_st_[col])), ax = ax1)
                sns.scatterplot(data = tcr, x = axis1, y =axis2 , hue = list(map(str.lower,md_df_st_[col])), ax = ax2)
                sns.scatterplot(data = tcr, x = axis1, y =axis2 , hue = list(map(str.lower,md_df_st_[col])), ax = ax3)
            fig = px.scatter_3d(tcr, x=axis1, y=axis2, z = axis3)
DIANE's avatar
DIANE committed
            sns.scatterplot(data = tcr, x = axis1, y =axis2 , ax = ax1)
        fig.update_traces(marker=dict(size=4))
DIANE's avatar
DIANE committed

        if selected_samples_idx:
            tt = tcr.iloc[selected_samples_idx,:]
            fig.add_scatter3d(x = tt.loc[:,axis1], y = tt.loc[:,axis2],
DIANE's avatar
DIANE committed
                              z = tt.loc[:,axis3], mode ='markers', marker = dict(size = 5, color = 'black'),
DIANE's avatar
DIANE committed
                              name = 'selected samples')
DIANE's avatar
DIANE committed
        
        plt.savefig("./Report/Figures/test.png")
        st.plotly_chart(fig, use_container_width=True)
DIANE's avatar
DIANE committed
        if labels:
            num_clusters = len(np.unique(labels))
DIANE's avatar
DIANE committed
            custom_color_palette = px.colors.qualitative.Plotly[:num_clusters]
            color_discrete_sequence=custom_color_palette
DIANE's avatar
DIANE committed
            # Créer et exporter le graphique Axe1-Axe2 en PNG
            fig_axe1_axe2 = px.scatter(tcr, x=axis1, y=axis2, color=labels if list(labels) else None, color_discrete_sequence=custom_color_palette)
            fig_axe1_axe2.update_layout(title='Axe1-Axe2')
            fig_axe1_axe2.update_traces(marker=dict(size=4))
Mouhcine's avatar
Mouhcine committed
            fig_axe1_axe2.write_image("./Report/Figures/plot_axe1_axe2.png")
DIANE's avatar
DIANE committed
            # Créer et exporter le graphique Axe1-Axe3 en PNG
            fig_axe1_axe3 = px.scatter(tcr, x=axis1, y=axis3, color=labels if list(labels) else None, color_discrete_sequence=custom_color_palette)
            fig_axe1_axe3.update_layout(title='Axe1-Axe3')
            fig_axe1_axe3.update_traces(marker=dict(size=4))
Mouhcine's avatar
Mouhcine committed
            fig_axe1_axe3.write_image("./Report/Figures/plot_axe1_axe3.png")
DIANE's avatar
DIANE committed
            # Créer et exporter le graphique Axe2-Axe3 en PNG
            fig_axe2_axe3 = px.scatter(tcr, x=axis2, y=axis3, color=labels if list(labels) else None, color_discrete_sequence=custom_color_palette)
            fig_axe2_axe3.update_layout(title='Axe2-Axe3')
            fig_axe2_axe3.update_traces(marker=dict(size=4))
Mouhcine's avatar
Mouhcine committed
            fig_axe2_axe3.write_image("./Report/Figures/plot_axe2_axe3.png")
DIANE's avatar
DIANE committed
if not spectra.empty:
DIANE's avatar
DIANE committed
    if dim_red_method == dim_red_methods[1] or dim_red_method == dim_red_methods[3]:
        with loadings:
            st.write('Loadings plot')
            p = dr_model.loadings_
DIANE's avatar
DIANE committed
            freq = pd.DataFrame(colnames, index=p.index)
            
            
DIANE's avatar
DIANE committed
            
            if test =='.dx':
                if meta_data.loc[:,'xunits'][0] == '1/cm':
                    freq.columns = ['Wavenumber (1/cm)']
DIANE's avatar
DIANE committed
                    xlab = "Wavenumber (1/cm)"
                    inv = 'reversed'
DIANE's avatar
DIANE committed
                else:
                    freq.columns = ['Wavelength (nm)']
DIANE's avatar
DIANE committed
                    xlab = 'Wavelength (nm)'
                    inv = None
                    
DIANE's avatar
DIANE committed
            else:
                freq.columns = ['Wavelength/Wavenumber']
DIANE's avatar
DIANE committed
                xlab = 'Wavelength/Wavenumber'
                inv = None
DIANE's avatar
DIANE committed
                
            pp = pd.concat([p, freq], axis=1)
            #########################################
            df1 = pp.melt(id_vars=freq.columns)
            fig = px.line(df1, x=freq.columns, y='value', color='variable', color_discrete_sequence=px.colors.qualitative.Plotly)
            fig.update_layout(legend=dict(x=1, y=0, font=dict(family="Courier", size=12, color="black"),
                                        bordercolor="black", borderwidth=2))
DIANE's avatar
DIANE committed
            fig.update_layout(xaxis_title = xlab,yaxis_title = "Intensity" ,xaxis = dict(autorange= inv))

            st.plotly_chart(fig, use_container_width=True)

            # Export du graphique
            img = pio.to_image(fig, format="png")
            with open("./Report/figures/graphe_loadings.png", "wb") as f:
                f.write(img)
DIANE's avatar
DIANE committed
    if dim_red_method == dim_red_methods[1]:
        with influence:
            st.write('Influence plot')
            ax1 = st.selectbox("Component", options=dr_model.scores_.columns, index=3)
            leverage = dr_model.leverage_
            residuals = dr_model.residuals_
Mouhcine's avatar
Mouhcine committed
            fig = px.scatter(x=leverage[ax1], y=residuals[ax1], color=leverage[ax1]*residuals[ax1], color_continuous_scale='Blues')
            fig.update_layout(xaxis_title="Leverage", yaxis_title="Residuals")
BARTHES Nicolas's avatar
BARTHES Nicolas committed
            st.plotly_chart(fig, use_container_width=True)
            img = pio.to_image(fig, format="png")
Mouhcine's avatar
Mouhcine committed
            with open("./Report/figures/graphe_influence.png", "wb") as f:
        with hotelling:
            st.write('T²-Hotelling vs Q residuals plot')
            hotelling = dr_model.hotelling_
            ax2 = st.selectbox("Component", options=dr_model.scores_.columns, index=4)
Mouhcine's avatar
Mouhcine committed
            hotelling = dr_model.hotelling_
            fig = px.scatter(t, x=hotelling[ax2], y=residuals[ax2]).update_layout(xaxis_title="",yaxis_title="Residuals")
            st.plotly_chart(fig, use_container_width=True)
DIANE's avatar
DIANE committed
            fig.write_image("./Report/figures/graphe_hotelling.png", format="png")