Py.Cafe

jhsmit/

ipymolstar-rainbow-residues

Visualize Rainbow Residues with Molstar

DocsPricing
  • 1qyn.pdb
  • app.py
  • requirements.txt
app.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
from pathlib import Path

import solara
from Bio.PDB import PDBParser, Structure
from ipymolstar import PDBeMolstar
from matplotlib import colormaps
from matplotlib.colors import Normalize

AMINO_ACIDS = [
    "ALA",
    "ARG",
    "ASN",
    "ASP",
    "CYS",
    "GLN",
    "GLU",
    "GLY",
    "HIS",
    "ILE",
    "LEU",
    "LYS",
    "MET",
    "PHE",
    "PRO",
    "PYL",
    "SEC",
    "SER",
    "THR",
    "TRP",
    "TYR",
    "VAL",
]

# %%

pdb_path = Path("1qyn.pdb")
parser = PDBParser(QUIET=True)
structure = parser.get_structure("1qyn", pdb_path)
MAX_R = max(r.id[1] for r in structure.get_residues())

custom_data = {"data": pdb_path.read_bytes(), "format": "pdb", "binary": False}


# %%
def color_residues(
    structure: Structure.Structure, auth: bool = False, phase: int = 0
) -> dict:
    _, resn, _ = zip(
        *[r.id for r in structure.get_residues() if r.get_resname() in AMINO_ACIDS]
    )

    rmin, rmax = min(resn), max(resn)
    norm = Normalize(vmin=rmin, vmax=rmax)
    auth_str = "_auth" if auth else ""

    cmap = colormaps["hsv"]
    data = []
    for i in range(rmin, rmax + 1):
        range_size = rmax + 1 - rmin
        j = rmin + ((i - rmin + phase) % range_size)
        r, g, b, a = cmap(norm(i), bytes=True)
        color = {"r": int(r), "g": int(g), "b": int(b)}
        elem = {
            f"start{auth_str}_residue_number": j,
            f"end{auth_str}_residue_number": j,
            "color": color,
            "focus": False,
        }
        data.append(elem)

    color_data = {"data": data, "nonSelectedColor": None}
    return color_data


@solara.component
def Page():
    phase = solara.use_reactive(0)
    dark_effective = solara.lab.use_dark_effective()
    
    solara.Title("Rainbow Residues")
    color_data = color_residues(structure, auth=True, phase=phase.value)
    theme = 'dark' if dark_effective else 'light'
    with solara.Card():
        PDBeMolstar.element(
            molecule_id="",
            custom_data=custom_data,
            hide_water=True,
            color_data=color_data,
            theme=theme,
        )
        solara.FloatSlider(label="Phase", min=0, max=MAX_R, value=phase, step=1)