-
Notifications
You must be signed in to change notification settings - Fork 18
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #31 from dwhswenson/docs
Colorbars in plot
- Loading branch information
Showing
14 changed files
with
643 additions
and
462 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,4 @@ | ||
dask-worker-space | ||
# netcdf outputs | ||
*nc | ||
|
||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,211 @@ | ||
import scipy | ||
import numpy as np | ||
import pandas as pd | ||
|
||
from .plot_utils import ranged_colorbar | ||
|
||
# matplotlib is technically optional, but required for plotting | ||
try: | ||
import matplotlib | ||
import matplotlib.pyplot as plt | ||
except ImportError: | ||
HAS_MATPLOTLIB = False | ||
else: | ||
HAS_MATPLOTLIB = True | ||
|
||
def _colorbar(with_colorbar, cmap_f, norm, min_val): | ||
if with_colorbar is False: | ||
return None | ||
elif with_colorbar is True: | ||
cbmin = np.floor(min_val) # [-1.0..0.0] => -1; [0.0..1.0] => 0 | ||
cbmax = 1.0 | ||
cb = ranged_colorbar(cmap_f, norm, cbmin, cbmax) | ||
# leave open other inputs to be parsed later (like tuples) | ||
return cb | ||
|
||
class ContactCount(object): | ||
"""Return object when dealing with contacts (residue or atom). | ||
This contains all the information about the contacts of a given type. | ||
This information can be represented several ways. One is as a list of | ||
contact pairs, each associated with the fraction of time the contact | ||
occurs. Another is as a matrix, where the rows and columns label the | ||
pair number, and the value is the fraction of time. This class provides | ||
several methods to get different representations of this data for | ||
further analysis. | ||
In general, instances of this class shouldn't be created by a user using | ||
``__init__``; instead, they will be returned by other methods. So users | ||
will often need to use this object for analysis. | ||
Parameters | ||
---------- | ||
counter : :class:`collections.Counter` | ||
the counter describing the count of how often the contact occurred; | ||
key is a frozenset of a pair of numbers (identifying the | ||
atoms/residues); value is the raw count of the number of times it | ||
occurred | ||
object_f : callable | ||
method to obtain the object associated with the number used in | ||
``counter``; typically :meth:`mdtraj.Topology.residue` or | ||
:meth:`mdtraj.Topology.atom`. | ||
n_x : int | ||
number of objects in the x direction (used in plotting) | ||
n_y : int | ||
number of objects in the y direction (used in plotting) | ||
""" | ||
def __init__(self, counter, object_f, n_x, n_y): | ||
self._counter = counter | ||
self._object_f = object_f | ||
self.n_x = n_x | ||
self.n_y = n_y | ||
|
||
@property | ||
def counter(self): | ||
""" | ||
:class:`collections.Counter` : | ||
keys use index number; count is contact occurrences | ||
""" | ||
return self._counter | ||
|
||
@property | ||
def sparse_matrix(self): | ||
""" | ||
:class:`scipy.sparse.dok.dok_matrix` : | ||
sparse matrix representation of contacts | ||
Rows/columns correspond to indices and the values correspond to | ||
the count | ||
""" | ||
mtx = scipy.sparse.dok_matrix((self.n_x, self.n_y)) | ||
for (k, v) in self._counter.items(): | ||
key = list(k) | ||
mtx[key[0], key[1]] = v | ||
mtx[key[1], key[0]] = v | ||
return mtx | ||
|
||
@property | ||
def df(self): | ||
""" | ||
:class:`pandas.SparseDataFrame` : | ||
DataFrame representation of the contact matrix | ||
Rows/columns correspond to indices and the values correspond to | ||
the count | ||
""" | ||
mtx = self.sparse_matrix.tocoo() | ||
index = list(range(self.n_x)) | ||
columns = list(range(self.n_y)) | ||
return pd.SparseDataFrame(mtx, index=index, columns=columns) | ||
|
||
def plot(self, cmap='seismic', vmin=-1.0, vmax=1.0, with_colorbar=True): | ||
""" | ||
Plot contact matrix (requires matplotlib) | ||
Parameters | ||
---------- | ||
cmap : str | ||
color map name, default 'seismic' | ||
vmin : float | ||
minimum value for color map interpolation; default -1.0 | ||
vmax : float | ||
maximum value for color map interpolation; default 1.0 | ||
Returns | ||
------- | ||
fig : :class:`matplotlib.Figure` | ||
matplotlib figure object for this plot | ||
ax : :class:`matplotlib.Axes` | ||
matplotlib axes object for this plot | ||
""" | ||
if not HAS_MATPLOTLIB: # pragma: no cover | ||
raise RuntimeError("Error importing matplotlib") | ||
norm = matplotlib.colors.Normalize(vmin=vmin, vmax=vmax) | ||
cmap_f = plt.get_cmap(cmap) | ||
|
||
fig, ax = plt.subplots() | ||
ax.axis([0, self.n_x, 0, self.n_y]) | ||
ax.set_facecolor(cmap_f(norm(0.0))) | ||
|
||
min_val = 0.0 | ||
|
||
for (pair, value) in self.counter.items(): | ||
if value < min_val: | ||
min_val = value | ||
pair_list = list(pair) | ||
patch_0 = matplotlib.patches.Rectangle( | ||
pair_list, 1, 1, | ||
facecolor=cmap_f(norm(value)), | ||
linewidth=0 | ||
) | ||
patch_1 = matplotlib.patches.Rectangle( | ||
(pair_list[1], pair_list[0]), 1, 1, | ||
facecolor=cmap_f(norm(value)), | ||
linewidth=0 | ||
) | ||
ax.add_patch(patch_0) | ||
ax.add_patch(patch_1) | ||
|
||
_colorbar(with_colorbar, cmap_f, norm, min_val) | ||
|
||
return (fig, ax) | ||
|
||
def most_common(self, obj=None): | ||
""" | ||
Most common values (ordered) with object as keys. | ||
This uses the objects for the contact pair (typically MDTraj | ||
``Atom`` or ``Residue`` objects), instead of numeric indices. This | ||
is more readable and can be easily used for further manipulation. | ||
Parameters | ||
---------- | ||
obj : MDTraj Atom or Residue | ||
if given, the return value only has entries including this | ||
object (allowing one to, for example, get the most common | ||
contacts with a specific residue) | ||
Returns | ||
------- | ||
list : | ||
the most common contacts in order. If the list is ``l``, then | ||
each element ``l[e]`` is a tuple with two parts: ``l[e][0]`` is | ||
the key, which is a pair of Atom or Residue objects, and | ||
``l[e][1]`` is the count of how often that contact occurred. | ||
See also | ||
-------- | ||
most_common_idx : same thing, using index numbers as key | ||
""" | ||
if obj is None: | ||
result = [ | ||
([self._object_f(idx) for idx in common[0]], common[1]) | ||
for common in self.most_common_idx() | ||
] | ||
else: | ||
obj_idx = obj.index | ||
result = [ | ||
([self._object_f(idx) for idx in common[0]], common[1]) | ||
for common in self.most_common_idx() | ||
if obj_idx in common[0] | ||
] | ||
return result | ||
|
||
def most_common_idx(self): | ||
""" | ||
Most common values (ordered) with indices as keys. | ||
Returns | ||
------- | ||
list : | ||
the most common contacts in order. The if the list is ``l``, | ||
then each element ``l[e]`` consists of two parts: ``l[e][0]`` is | ||
a pair of integers, representing the indices of the objects | ||
associated with the contact, and ``l[e][1]`` is the count of how | ||
often that contact occurred | ||
See also | ||
-------- | ||
most_common : same thing, using objects as key | ||
""" | ||
return self._counter.most_common() |
Oops, something went wrong.