pythonmatplotlibseaborncolorbarkdeplot

Log scales with kdeplot


I am trying to make a nice free energy surface (heat map) using Seaborn's kdeplot. I am very close but can not figure out a way to change the color bar scale. The color bar scale is important since it is supposed to represent the difference in energy at different coordinates on the map. I need to know how to scale the values of the color bar by -(0.5961573)*log(x), where x is the values of the color bar. I may also then need to normalize the color bar from there so that the max value is 0.

Here is what I currently have:

import numpy as np
import matplotlib as mpl
import matplotlib.pyplot as plt
import matplotlib.pyplot as plt
import seaborn as sns


rs=[]
dihes=[]
with open(sys.argv[1], 'r') as f:
    for line in f:
        time,r,dihe = line.split()
        rs.append(float(r))
        dihes.append(float(dihe))
    sns.set_style("white")
    sns.kdeplot(rs, dihes, n_levels=25, cbar=True, cmap="Purples_d")
    plt.show()

This gets me:

sample output

The arrays rs and dihes are simple one dimensional arrays.

Any suggestions on how to scale the color bar (z-axis) would be very helpful!


Solution

  • One way to do it is to create the graph manually and then modify the labels directly. This involves a couple more lines of code. You may have to tweak the formatting a bit but something like this should get you on the right track.

    The following is adapted from this answer and this answer.

    import numpy as np
    import matplotlib as mpl
    import matplotlib.pyplot as plt
    import seaborn as sns
    from scipy import stats
    
    rs=[]
    dihes=[]
    with open(sys.argv[1], 'r') as f:
        for line in f:
            time,r,dihe = line.split()
            rs.append(float(r))
            dihes.append(float(dihe))
    
        x = rs
        y = dihes
        kde = stats.gaussian_kde([x, y])
    
        xx, yy = np.mgrid[min(x):max(x):(max(x)-min(x))/100, min(y):max(y):(max(y)-min(y))/100]
        density = kde(np.c_[xx.flat, yy.flat].T).reshape(xx.shape)
    
        sns.set_style("white")
        fig, ax = plt.subplots()
        cset = ax.contour(xx, yy, density, 25, cmap="Purples_r")
        cb = fig.colorbar(cset)
        cb.ax.set_yticklabels(map(lambda x: -0.5961573*np.log(float(x.get_text())), 
                                  cb.ax.get_yticklabels()))