modules/pmg_qt/builder.py (1,159 lines of code) (raw):
from __future__ import print_function
import itertools, os
from glob import glob
import pymol
import pymol._gui
from .pymol_gl_widget import PyMOLGLWidget
from pymol.Qt import QtGui, QtCore
from pymol.Qt import QtWidgets
Qt = QtCore.Qt
from pymol.wizard import Wizard
from pymol.parsing import QuietException
from pymol import editor
from pymol import computing
from chempy import cpv
active_sele = "_builder_active" # object we're working on...
newest_sele = "_builder_added" # last atom added?
display_sele = "_build_display"
undocontext = editor.undocontext
def undoablemethod(sele):
def decorator(func):
def wrapper(self, *args, **kwargs):
with undocontext(self.cmd, sele):
return func(self, *args, **kwargs)
return wrapper
return decorator
#############################################################
# Action-directing Wizards
class ActionWizard(Wizard):
def __init__(self,_self):
Wizard.__init__(self,_self)
self.actionHash = str(self.__class__)
def setActionHash(self, action_hash):
self.actionHash = action_hash
def activateOrDismiss(self):
activate_flag = 1
cur_wiz = self.cmd.get_wizard()
if cur_wiz != None:
if cur_wiz.__class__ == self.__class__:
if cur_wiz.actionHash == self.actionHash:
activate_flag = 0
if activate_flag:
self.cmd.set_wizard(self,replace=1)
self.cmd.refresh_wizard()
else:
self.actionWizardDone()
return activate_flag
def actionWizardDone(self):
self.cmd.delete(active_sele)
self.cmd.unpick()
self.cmd.set_wizard()
self.cmd.refresh_wizard()
def activeSeleValid(self):
if active_sele in self.cmd.get_names("selections"):
if self.cmd.select(active_sele, "byobj "+active_sele)<1:
self.cmd.delete(active_sele)
else:
enabled_list = self.cmd.get_names("objects",enabled_only=1)
active_obj_list = self.cmd.get_object_list(active_sele)
if len(active_obj_list) != 1:
self.cmd.delete(active_sele)
elif active_obj_list[0] not in enabled_list:
self.cmd.delete(active_sele)
if "pk1" in self.cmd.get_names("selections"):
self.cmd.select(active_sele,"byobj pk1")
else:
enabled_list = self.cmd.get_names("objects",enabled_only=1)
if len(enabled_list)==1:
if self.cmd.select(active_sele, enabled_list[0])<1:
self.cmd.delete(active_sele)
return active_sele in self.cmd.get_names("selections")
class CleanWizard(ActionWizard):
def __init__(self,_self):
self.clean_obj = None
ActionWizard.__init__(self,_self)
def run_job(self):
if active_sele in self.cmd.get_names("selections"):
obj_list = self.cmd.get_object_list(active_sele)
if len(obj_list)==1:
self.cmd.unpick()
self.cmd.set_wizard()
self.cmd.refresh_wizard()
self.cmd.do("_ cmd.clean('%s',message='''Cleaning %s...''',async_=1)"%(active_sele,obj_list[0]))
def do_pick(self, bondFlag):
if active_sele in self.cmd.get_names("selections"):
obj_list = self.cmd.get_object_list(active_sele)
if len(obj_list)!=1:
self.cmd.delete(active_sele)
else:
self.cmd.select(active_sele, "byobj pk1")
self.cmd.unpick()
self.cmd.deselect()
obj_list = self.cmd.get_object_list(active_sele)
if isinstance(obj_list,list) and (len(obj_list)==1):
self.run_job()
else:
print("Error: can only clean one object at a time")
def toggle(self):
if self.activateOrDismiss():
if self.activeSeleValid():
self.run_job()
def get_prompt(self):
return ["Pick object to clean..."]
def get_panel(self):
return [
[ 1, 'Clean', ''],
[ 2, 'Done','cmd.set_wizard()'],
]
class SculptWizard(ActionWizard):
def __init__(self,_self):
ActionWizard.__init__(self,_self)
self.sculpt_object = None
def sculpt_activate(self):
if active_sele in self.cmd.get_names("selections"):
obj_list = self.cmd.get_object_list(active_sele)
if len(obj_list)==1:
obj_name = obj_list[0]
self.cmd.push_undo(obj_name)
self.cmd.sculpt_activate(obj_name)
self.cmd.set("sculpting",1)
self.sculpt_object = obj_name
self.cmd.sculpt_activate(obj_name)
if int(self.cmd.get("sculpt_vdw_vis_mode")):
self.cmd.show("cgo",obj_name)
self.cmd.unpick()
self.cmd.refresh_wizard()
else:
print("Error: cannot sculpt more than one object at a time")
def sculpt_deactivate(self):
if ((self.sculpt_object != None) and
self.sculpt_object in self.cmd.get_names()):
self.cmd.set("sculpt_vdw_vis_mode","0",self.sculpt_object)
self.cmd.sculpt_iterate(self.sculpt_object,self.cmd.get_state(),0)
self.cmd.unset("sculpt_vdw_vis_mode",self.sculpt_object)
self.cmd.sculpt_deactivate(self.sculpt_object)
self.sculpt_object = None
self.cmd.refresh_wizard()
def do_pick(self, bondFlag):
if self.sculpt_object == None:
self.cmd.select(active_sele, "byobj pk1")
self.sculpt_activate()
else:
return 0 # already sculpting, so handle like a normal edit
def toggle(self):
if self.activateOrDismiss():
if self.activeSeleValid():
self.sculpt_activate()
def get_prompt(self):
if self.sculpt_object == None:
return ["Pick object to sculpt..."]
else:
return ["Sculpting %s..."%self.sculpt_object]
def finish_sculpting(self):
if self.sculpt_object:
self.sculpt_deactivate()
self.cmd.set("sculpting",0)
self.cmd.delete(active_sele)
self.cmd.set_wizard()
self.cmd.refresh_wizard()
def scramble(self,mode):
if self.cmd.count_atoms(self.sculpt_object):
sc_tmp = "_scramble_tmp"
if mode == 0:
self.cmd.select(sc_tmp,self.sculpt_object+
" and not (fixed or restrained)")
if mode == 1:
self.cmd.select(sc_tmp,self.sculpt_object+
" and not (fixed)")
extent = self.cmd.get_extent(sc_tmp)
center = self.cmd.get_position(sc_tmp)
radius = 1.25*cpv.length(cpv.sub(extent[0],extent[1]))
self.cmd.alter_state(self.cmd.get_state(), sc_tmp,
"(x,y,z)=rsp(pos,rds)",
space= { 'rsp' : cpv.random_displacement,
'pos' : center,
'rds' : radius })
self.cmd.delete(sc_tmp)
def get_panel(self):
return [
[ 1, 'Sculpt', ''],
[ 2, 'Undo', 'cmd.undo()'],
[ 2, 'Switch Object', 'cmd.get_wizard().sculpt_deactivate()'],
[ 2, 'Scramble Unrestrained Coords.', 'cmd.get_wizard().scramble(0)'],
[ 2, 'Scramble Unfixed Coords.', 'cmd.get_wizard().scramble(1)'],
[ 2, 'Done','cmd.get_wizard().finish_sculpting()'],
]
def cleanup(self):
self.sculpt_deactivate()
ActionWizard.cleanup(self)
class RepeatableActionWizard(ActionWizard):
def __init__(self,_self):
ActionWizard.__init__(self,_self)
self.repeating = 0
def repeat(self):
self.repeating = 1
self.cmd.refresh_wizard()
def getRepeating(self):
return self.repeating
def activateRepeatOrDismiss(self):
activate_flag = 1
cur_wiz = self.cmd.get_wizard()
if cur_wiz != None:
if cur_wiz.__class__ == self.__class__:
if cur_wiz.actionHash == self.actionHash:
if cur_wiz.getRepeating():
activate_flag = 0
else:
self.repeat()
elif cur_wiz.getRepeating():
self.repeat()
if activate_flag:
self.cmd.set_wizard(self,replace=1)
self.repeat() # always repeating for now...
self.cmd.refresh_wizard()
else:
self.actionWizardDone()
return activate_flag
def cleanup(self):
self.cmd.unpick()
ActionWizard.cleanup(self)
class ReplaceWizard(RepeatableActionWizard):
def do_pick(self, bondFlag):
self.cmd.select(active_sele, "bymol pk1")
self.cmd.replace(self.symbol, self.geometry, self.valence)
if not self.getRepeating():
self.actionWizardDone()
def toggle(self, symbol, geometry, valence, text):
self.symbol = symbol
self.geometry = geometry
self.valence = valence
self.text = text
self.setActionHash( (symbol,geometry,valence,text) )
self.activateRepeatOrDismiss()
def get_prompt(self):
if self.getRepeating():
return ["Pick atoms to replace with %s..."%self.text]
else:
return ["Pick atom to replace with %s..."%self.text]
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Replacing Multiple Atoms',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Replacing an Atom',''],
[ 2, 'Replace Multiple Atoms','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class AttachWizard(RepeatableActionWizard):
def __init__(self,_self):
RepeatableActionWizard.__init__(self,_self)
self.mode = 0
def do_pick(self, bondFlag):
if self.mode == 0:
self.cmd.select(active_sele, "bymol pk1")
editor.attach_fragment("pk1", self.fragment, self.position, self.geometry, _self=self.cmd)
elif self.mode == 1:
self.cmd.select(active_sele, "bymol pk1")
editor.combine_fragment("pk1", self.fragment, self.position,
self.geometry, _self=self.cmd)
self.mode = 0
self.cmd.refresh_wizard()
self.cmd.unpick()
if not self.getRepeating():
self.actionWizardDone()
def toggle(self, fragment, position, geometry, text):
self.fragment = fragment
self.position = position
self.geometry = geometry
self.text = text
self.setActionHash( (fragment, position, geometry, text) )
self.activateRepeatOrDismiss()
def create_new(self):
names = self.cmd.get_names("objects")
num = 1
while 1:
name = "obj%02d"%num
if name not in names:
break
num = num + 1
self.cmd.fragment(self.fragment, name)
if not self.getRepeating():
self.actionWizardDone()
def get_prompt(self):
if self.mode == 0:
if self.getRepeating():
return ["Pick locations to attach %s..."%self.text]
else:
return ["Pick location to attach %s..."%self.text]
else:
return ["Pick object to combine %s into..."%self.text]
def combine(self):
self.mode = 1
self.cmd.refresh_wizard()
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Attaching Multiple Fragmnts',''],
[ 2, 'Create As New Object','cmd.get_wizard().create_new()'],
[ 2, 'Combine w/ Existing Object','cmd.get_wizard().combine()'],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Attaching One Fragment',''],
[ 2, 'Create As New Object','cmd.get_wizard().create_new()'],
[ 2, 'Combine w/ Existing Object','cmd.get_wizard().combine()'],
[ 2, 'Attach Multiple Fragments','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class AminoAcidWizard(RepeatableActionWizard):
def __init__(self,_self):
RepeatableActionWizard.__init__(self,_self)
self.mode = 0
def do_pick(self, bondFlag):
# since this function can change any position of atoms in a related
# molecule, bymol is used
if self.mode == 0:
self.cmd.select(active_sele, "bymol pk1")
try:
with undocontext(self.cmd, "bymol ?pk1"):
editor.attach_amino_acid("pk1", self.aminoAcid, _self=self.cmd)
except QuietException:
fin = -1
elif self.mode == 1:
self.cmd.select(active_sele, "bymol pk1")
editor.combine_fragment("pk1", self.aminoAcid, 0, 1, _self=self.cmd)
self.mode = 0
self.cmd.refresh_wizard()
self.cmd.unpick()
if not self.getRepeating():
self.actionWizardDone()
def toggle(self, amino_acid):
self.aminoAcid = amino_acid
self.setActionHash( (amino_acid,) )
self.activateRepeatOrDismiss()
def create_new(self):
names = self.cmd.get_names("objects")
num = 1
while 1:
name = "obj%02d"%num
if name not in names:
break
num = num + 1
editor.attach_amino_acid("pk1", self.aminoAcid, object=name, _self=self.cmd)
if not self.getRepeating():
self.actionWizardDone()
def combine(self):
self.mode = 1
self.cmd.refresh_wizard()
def get_prompt(self):
if self.mode == 0:
if self.getRepeating():
return ["Pick locations to attach %s..."%self.aminoAcid]
else:
return ["Pick location to attach %s..."%self.aminoAcid]
else:
return ["Pick object to combine %s into..."%self.aminoAcid]
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Attaching Multiple Residues',''],
[ 2, 'Create As New Object','cmd.get_wizard().create_new()'],
[ 2, 'Combine w/ Existing Object','cmd.get_wizard().combine()'],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Attaching Amino Acid',''],
[ 2, 'Create As New Object','cmd.get_wizard().create_new()'],
[ 2, 'Combine w/ Existing Object','cmd.get_wizard().combine()'],
[ 2, 'Attach Multiple...','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class ValenceWizard(RepeatableActionWizard):
def cleanup(self):
self.cmd.button('single_left','none','PkAt')
self.cmd.button('double_left','none','MovA')
@undoablemethod("(?pk1 ?pk2) extend 1")
def do_pick(self, bondFlag):
self.cmd.select(active_sele, "bymol pk1")
if bondFlag:
if int(self.order)>=0:
self.cmd.valence(self.order, "pk1", "pk2")
self.cmd.h_fill()
else:
self.cmd.cycle_valence()
else:
self.cmd.button('double_left','none','PkBd')
self.cmd.button('single_left','none','PkBd')
self.cmd.unpick()
if not self.getRepeating():
self.actionWizardDone()
def toggle(self, order, text):
self.order = order
self.text = text
self.setActionHash( (order,text) )
self.activateRepeatOrDismiss()
if self.cmd.get_wizard() == self:
# get us into bond picking mode...
self.cmd.button('double_left','none','PkBd')
self.cmd.button('single_left','none','PkBd')
def get_prompt(self):
if self.getRepeating():
return ["Pick bonds to set as %s..."%self.text]
else:
return ["Pick bond to set as %s..."%self.text]
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Setting Multiple Valences',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Set a Bond Valence',''],
[ 2, 'Set Multiple Valences','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class ChargeWizard(RepeatableActionWizard):
@undoablemethod("bymol ?pk1")
def do_pick(self, bondFlag):
self.cmd.select(active_sele, "bymol pk1")
self.cmd.alter("pk1","formal_charge=%s" % self.charge)
self.cmd.h_fill()
if abs(float(self.charge))>0.0001:
self.cmd.label("pk1","'''"+self.text+"'''")
else:
self.cmd.label("pk1")
self.cmd.unpick()
if not self.getRepeating():
self.actionWizardDone()
def toggle(self, charge, text):
self.charge = charge
self.text = text
self.setActionHash( (charge,text) )
self.activateRepeatOrDismiss()
def get_prompt(self):
if self.getRepeating():
return ["Pick atoms to set charge = %s..."%self.text]
else:
return ["Pick atom to set charge = %s..."%self.text]
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Setting Multiple Charges',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Setting Atom Charge',''],
[ 2, 'Modify Multiple Atoms','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class InvertWizard(RepeatableActionWizard):
def do_pick(self, bondFlag):
self.cmd.select(active_sele, "bymol pk1")
picked = collectPicked(self.cmd)
if picked == ["pk1","pk2","pk3"]:
self.cmd.invert()
self.cmd.unpick()
if not self.getRepeating():
self.actionWizardDone()
self.cmd.refresh_wizard()
def toggle(self):
self.activateRepeatOrDismiss()
def get_prompt(self):
if "pk1" in self.cmd.get_names("selections"):
if "pk2" in self.cmd.get_names("selections"):
return ["Pick the second stationary atom..."]
else:
return ["Pick the first stationary atom..."]
else:
return ["Pick origin atom for inversion..."]
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Inverting Multiple',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Inverting Stereocenter',''],
[ 2, 'Invert Multiple','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class BondWizard(RepeatableActionWizard):
@staticmethod
def staticaction(cmd):
picked = collectPicked(cmd)
if picked != ["pk1","pk2"]:
return False
cmd.select(active_sele, "bymol ?pk1")
if ( cmd.count_atoms("?pk1&hydro") and
cmd.count_atoms("?pk2&hydro") and
cmd.count_atoms("(?pk1 extend 1)&!hydro") and
cmd.count_atoms("(?pk2 extend 1)&!hydro")):
# two hydrogens picked -> pick their heavy neighbors instead
cmd.select("pk1","(pk1 extend 1) and not hydro")
cmd.select("pk2","(pk2 extend 1) and not hydro")
with undocontext(cmd, "(?pk1 ?pk2) extend 1"):
cmd.bond("pk1", "pk2")
cmd.h_fill()
cmd.unpick()
return True
def do_pick(self, bondFlag):
if self.staticaction(self.cmd):
if not self.getRepeating():
self.actionWizardDone()
self.cmd.refresh_wizard()
def toggle(self):
self.activateRepeatOrDismiss()
def get_prompt(self):
if "pk1" in self.cmd.get_names("selections"):
return ["Pick second atom for bond..."]
else:
return ["Pick first atom for bond..."]
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Creating Multiple Bonds',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Creating Bond',''],
[ 2, 'Create Multiple Bonds','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class UnbondWizard(RepeatableActionWizard):
def cleanup(self):
self.cmd.button('single_left','none','PkAt')
@undoablemethod("(?pk1 ?pk2) extend 1")
def do_pick(self, bondFlag):
self.cmd.select(active_sele, "bymol pk1")
if bondFlag:
self.cmd.unbond("pk1", "pk2")
self.cmd.h_fill()
self.cmd.unpick()
else:
self.cmd.button('single_left','none','PkBd')
self.cmd.unpick()
if not self.getRepeating():
self.actionWizardDone()
def toggle(self):
self.activateRepeatOrDismiss()
if self.cmd.get_wizard() == self:
self.cmd.button('single_left','none','PkBd') # get us into bond picking mode...
def get_prompt(self):
if self.getRepeating():
return ["Pick bonds to delete..."]
else:
return ["Pick bond to delete..."]
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Deleting Multiple Bonds',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Deleting a Bond',''],
[ 2, 'Delete Multiple Bonds','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class HydrogenWizard(RepeatableActionWizard):
def run_add(self):
if self.mode == 'add':
self.cmd.h_add(active_sele)
self.cmd.delete(active_sele)
def do_pick(self, bondFlag):
self.cmd.select(active_sele, "bymol pk1")
if self.mode == 'fix':
self.cmd.h_fill()
self.cmd.unpick()
elif self.mode == 'add':
self.cmd.unpick()
self.run_add()
if not self.getRepeating():
self.actionWizardDone()
def toggle(self,mode):
self.mode = mode
self.setActionHash( (mode,) )
if self.mode == 'add':
if self.activateOrDismiss():
if self.activeSeleValid():
self.run_add()
else:
self.activateRepeatOrDismiss()
def get_prompt(self):
if self.mode == 'fix':
if self.getRepeating():
return ["Pick atom upon which to fix hydrogens..."]
else:
return ["Pick atoms upon which to fix hydrogens..."]
else:
return ["Pick molecule upon which to add hydrogens..."]
def get_panel(self):
if self.getRepeating():
if self.mode == 'fix':
return [
[ 1, 'Fixing Hydrogens',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Adding Hydrogens',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
if self.mode == 'fix':
return [
[ 1, 'Fixing Hydrogens',''],
[ 2, 'Fix Multiple Atoms','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Adding Hydrogens',''],
[ 2, 'Add To Multiple...','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class RemoveWizard(RepeatableActionWizard):
@undoablemethod("?pk1 extend 1")
def do_pick(self, bondFlag):
cnt = self.cmd.select(active_sele,
"((pk1 and not hydro) extend 1) and not hydro")
self.cmd.remove_picked()
if cnt:
self.cmd.fix_chemistry(active_sele)
self.cmd.h_add(active_sele)
self.cmd.delete(active_sele)
if not self.getRepeating():
self.actionWizardDone()
def toggle(self):
self.activateRepeatOrDismiss()
def get_prompt(self):
if self.getRepeating():
return ["Pick atoms to delete..."]
else:
return ["Pick atom to delete..."]
def get_panel(self):
if self.getRepeating():
return [
[ 1, 'Deleting Atoms',''],
[ 2, 'Done','cmd.set_wizard()'],
]
else:
return [
[ 1, 'Deleting an Atom',''],
[ 2, 'Delete Multiple Atoms','cmd.get_wizard().repeat()'],
[ 2, 'Done','cmd.set_wizard()'],
]
class AtomFlagWizard(ActionWizard):
def update_display(self):
if active_sele in self.cmd.get_names("selections"):
self.cmd.select(display_sele,active_sele+
" and flag %d"%self.flag)
self.cmd.enable(display_sele)
else:
self.cmd.delete(display_sele)
self.cmd.refresh_wizard()
def do_pick(self, bondFlag):
if not(active_sele not in self.cmd.get_names("selections")):
if self.cmd.count_atoms("pk1 and flag %d"%self.flag):
self.cmd.flag(self.flag,"pk1","clear")
else:
self.cmd.flag(self.flag,"pk1","set")
self.cmd.select(active_sele, "byobj pk1")
self.cmd.unpick()
self.update_display()
def do_select(self,selection):
if selection == display_sele:
self.cmd.flag(self.flag,active_sele+" and "+display_sele,"set")
self.cmd.flag(self.flag,active_sele+" and not "+display_sele,"clear")
self.cmd.refresh_wizard()
self.update_display()
def get_prompt(self):
if active_sele not in self.cmd.get_names("selections"):
return ["Pick object to operate on..."]
else:
self.cmd.reference("validate",active_sele) # overbroad
if self.flag == 2:
return ["Toggle restrained atoms..."]
elif self.flag ==3:
return ["Toggle fixed atoms..."]
else:
return ["Toggle unknown atom flag..."]
def toggle(self,flag=0):
self.flag = flag
if self.activateOrDismiss():
if self.activeSeleValid():
self.update_display()
else:
self.cmd.deselect()
self.cmd.unpick()
def do_all(self):
if active_sele in self.cmd.get_names("selections"):
self.cmd.flag(self.flag,active_sele,"set")
self.update_display()
def do_less(self,mode):
if active_sele in self.cmd.get_names("selections"):
if mode == 0:
self.cmd.flag(self.flag,"(( byobj " + active_sele +
" ) and not flag %d) extend 1"%self.flag,"clear")
elif mode == 1:
self.cmd.flag(self.flag,"byres ((( byobj " + active_sele +
" ) and not flag %d) extend 1)"%self.flag,"clear")
self.update_display()
def do_cas(self,mode):
if active_sele in self.cmd.get_names("selections"):
if mode == 1:
self.cmd.flag(self.flag,active_sele,"clear")
self.cmd.flag(self.flag,active_sele+" and polymer and name ca","set")
elif mode == 0:
self.cmd.flag(self.flag,active_sele+
" and flag %d and polymer and name ca"%self.flag,"set")
self.cmd.flag(self.flag,active_sele+
" and not (polymer and name ca)","clear")
self.update_display()
def do_more(self,mode):
if active_sele in self.cmd.get_names("selections"):
if mode == 0:
self.cmd.flag(self.flag,active_sele +
" and (flag %d extend 1)"%self.flag,"set")
elif mode == 1:
self.cmd.flag(self.flag,"byres ("+ active_sele +
" and (byres flag %d) extend 1)"%self.flag,"set")
elif mode == 2:
self.cmd.flag(self.flag,"byres ("+ active_sele +
" and flag %d )"%self.flag,"set")
self.update_display()
def do_none(self):
if active_sele in self.cmd.get_names("selections"):
self.cmd.flag(self.flag,active_sele,"clear")
self.update_display()
def do_store(self):
if active_sele in self.cmd.get_names("selections"):
self.cmd.reference("store",active_sele)
def do_recall(self):
if active_sele in self.cmd.get_names("selections"):
self.cmd.reference("recall",active_sele)
def do_swap(self):
if active_sele in self.cmd.get_names("selections"):
self.cmd.reference("swap",active_sele)
def get_panel(self):
title = {2:"Restrained Atoms",
3:"Fixed Atoms"}.get(self.flag)
verb = {2:"Restrain", 3:"Fix"}.get(self.flag)
result = [
[ 1, title, ''],
[ 2, "All",'cmd.get_wizard().do_all()'],
[ 2, "All C-alphas",'cmd.get_wizard().do_cas(1)'],
[ 2, "More (byres)",'cmd.get_wizard().do_more(1)'],
[ 2, "More",'cmd.get_wizard().do_more(0)'],
[ 2, "Byresidue",'cmd.get_wizard().do_more(2)'],
[ 2, "Less", 'cmd.get_wizard().do_less(0)'],
[ 2, "Less (by residue)", 'cmd.get_wizard().do_less(1)'],
[ 2, "Only C-alphas",'cmd.get_wizard().do_cas(0)'],
[ 2, "None", 'cmd.get_wizard().do_none()'],
[ 2, 'Done','cmd.set_wizard()'],
]
if self.flag == 2:
result[-1:-1] = [
[ 2, "Store Reference Coords.", 'cmd.get_wizard().do_store()'],
[ 2, "Recall Reference Coords.", 'cmd.get_wizard().do_recall()'],
[ 2, "Swap Reference Coords.", 'cmd.get_wizard().do_swap()']]
return result
def cleanup(self):
self.cmd.delete(display_sele)
Wizard.cleanup(self)
class FixAtomWizard(AtomFlagWizard):
pass
class RestAtomWizard(AtomFlagWizard):
pass
#############################################################
### Helper functions
def getSeleDict(self_cmd):
result = {}
for sele in self_cmd.get_names("selections"):
result[sele] = 1
return result
def collectPicked(self_cmd):
result = []
sele_dict = getSeleDict(self_cmd)
for sele in ["pk1","pk2","pk3","pk4"]:
if sele in sele_dict:
result.append(sele)
return result
#############################################################
### Actual GUI
def makeFragmentButton():
btn = QtWidgets.QPushButton()
btn.setAttribute(Qt.WA_LayoutUsesWidgetRect) # OS X workaround
btn.setSizePolicy(
QtWidgets.QSizePolicy.Minimum,
QtWidgets.QSizePolicy.MinimumExpanding)
btn.setAutoDefault(False)
return btn
class _BuilderPanel(QtWidgets.QWidget):
def __init__(self, parent=None, app=None):
super(_BuilderPanel, self).__init__(parent)
self.setWindowTitle("Builder")
self.setObjectName("builder")
self.cmd = app.pymol.cmd
self.layout = QtWidgets.QVBoxLayout()
self.setLayout(self.layout)
self.buttons_layout = QtWidgets.QVBoxLayout()
self.tabs = QtWidgets.QTabWidget(self)
self.layout.setContentsMargins(5, 5, 5, 5);
self.layout.setSpacing(5);
self.layout.addWidget(self.tabs)
self.layout.addLayout(self.buttons_layout)
self.layout.addStretch()
self.fragments_layout = QtWidgets.QGridLayout()
self.fragments_layout.setContentsMargins(5, 5, 5, 5);
self.fragments_layout.setSpacing(5);
self.fragments_tab = QtWidgets.QWidget()
self.fragments_tab.setLayout(self.fragments_layout)
self.protein_layout = QtWidgets.QGridLayout()
self.protein_layout.setContentsMargins(5, 5, 5, 5);
self.protein_layout.setSpacing(5);
self.protein_tab = QtWidgets.QWidget()
self.protein_tab.setLayout(self.protein_layout)
self.tabs.addTab(self.fragments_tab, "Chemical")
self.tabs.addTab(self.protein_tab, "Protein")
self.getIcons()
buttons = [
[ ("H", "Hydrogen", lambda: self.replace("H", 1, 1, "Hydrogen")),
("C", "Carbon", lambda: self.replace("C", 4, 4, "Carbon")),
("N", "Nitrogen", lambda: self.replace("N", 4, 3, "Nitrogen")),
("O", "Oxygen", lambda: self.replace("O", 4, 2, "Oxygen")),
("P", "Phosphorus", lambda: self.replace("P",4,3, "Phosphorous")),
("S", "Sulfur", lambda: self.replace("S",2,2, "Sulfur")),
("F", "Fluorine", lambda: self.replace("F",1,1, "Fluorine")),
("Cl", "Chlorrine", lambda: self.replace("Cl",1,1, "Chlorine")),
("Br", "Bromine", lambda: self.replace("Br",1,1, "Bromine")),
("I", "Iodine", lambda: self.replace("I",1,1, "Iodine")),
("-CF3", "Trifluoromethane", lambda: self.replace("trifluoromethane",4,0, "trifluoro")),
("-OMe", "Methanol", lambda: self.replace("methanol",5,0, "methoxy")),
],
[ ("CH4", "Methyl", lambda: self.grow("methane",1,0,"methyl")),
("C=C", "Ethylene", lambda: self.grow("ethylene",4,0,"vinyl")),
("C#C", "Acetylene", lambda: self.grow("acetylene",2,0,"alkynl")),
("C#N", "Cyanide", lambda: self.grow("cyanide",2,0,"cyano")),
("C=O", "Aldehyde", lambda: self.grow("formaldehyde",2,0,"carbonyl",)),
("C=OO", "Formic Acid", lambda: self.grow("formic",4,0,"carboxyl")),
("C=ON", "C->N amide", lambda: self.grow("formamide",5,0,"C->N amide")),
("NC=O", "N->C amide", lambda: self.grow("formamide",3,1,"N->C amide")),
("S=O2", "Sulfone", lambda: self.grow("sulfone",3,1,"sulfonyl")),
("P=O3", "Phosphite", lambda: self.grow("phosphite",4,0,"phosphoryl")),
("N=O2", "Nitro", lambda: self.grow("nitro",3,0,"nitro")),
],
[
("#cyc3", "Cyclopropane", lambda: self.grow("cyclopropane",4,0,"cyclopropyl")),
("#cyc4", "Cyclobutane", lambda: self.grow("cyclobutane",4,0,"cyclobutyl")),
("#cyc5", "Cyclopentane", lambda: self.grow("cyclopentane",5,0,"cyclopentyl")),
("#cyc6", "Cyclohexane", lambda: self.grow("cyclohexane",7,0,"cyclohexyl")),
("#cyc7", "Cycloheptane", lambda: self.grow("cycloheptane",8,0,"cycloheptyl")),
("#aro5", "Cyclopentadiene", lambda: self.grow("cyclopentadiene",5,0,"cyclopentadienyl")),
("#aro6", "Benzene", lambda: self.grow("benzene",6,0,"phenyl")),
("#aro65", "Indane", lambda: self.grow("indane",12,0,"indanyl")),
("#aro66", "Napthylene", lambda: self.grow("napthylene",13,0,"napthyl")),
("#aro67", "Benzocycloheptane", lambda: self.grow("benzocycloheptane",13,0, "benzocycloheptyl")),
]
]
self.btn_icons = {}
requestsize = QtCore.QSize(48, 48)
for row, btn_row in enumerate(buttons):
for col, bb in enumerate(btn_row):
btn_label, btn_tooltip, btn_command = bb
btn = makeFragmentButton()
if btn_label.startswith('#'):
icons = self.icons[btn_label[1:]]
btn.setIcon(icons[0])
btn.setIconSize(icons[1].actualSize(requestsize))
self.btn_icons[btn] = icons
else:
btn.setText(btn_label)
btn.setToolTip(btn_tooltip)
btn.clicked.connect(btn_command)
self.fragments_layout.addWidget(btn, row, col)
buttons = [
[ 'Ace', 'Ala', 'Arg', 'Asn', 'Asp', 'Cys', 'Gln', 'Glu', 'Gly', 'His', 'Ile', 'Leu' ],
[ 'Lys', 'Met', 'Phe', 'Pro', 'Ser', 'Thr', 'Trp', 'Tyr', 'Val', 'NMe', 'NHH' ]
]
for row, btn_row in enumerate(buttons):
for col, btn_label in enumerate(btn_row):
btn = makeFragmentButton()
btn.setText(btn_label)
btn.setToolTip("Build %s residue" % btn_label)
res = btn_label.lower()
slot = lambda val=None, s=self,r=res: s.attach(r)
btn.clicked.connect(slot)
self.protein_layout.addWidget(btn, row, col)
lab = QtWidgets.QLabel('Secondary Structure:')
lab_cols = 3
self.ss_cbox = QtWidgets.QComboBox()
self.ss_cbox.addItem("Alpha Helix")
self.ss_cbox.addItem("Beta Sheet (Anti-Parallel)")
self.ss_cbox.addItem("Beta Sheet (Parallel)")
self.protein_layout.addWidget(lab, 2, 0, 1, lab_cols)
self.protein_layout.addWidget(self.ss_cbox, 2, lab_cols, 1, 4)
buttons = [
[
( "@Atoms:", None, None),
( "Fix H", "Fix hydrogens on picked atoms", self.fixH),
( "Add H", "Add hydrogens to entire molecule", self.addH),
( "Invert", "Invert stereochemistry around pk1 (pk2 and pk3 will remain fixed)", self.invert),
( "Delete", "Remove atoms", self.removeAtom),
( "Clear", "Delete everything", self.clear),
( "@ Charge:", None, None),
( " +1 ", "Positive Charge", lambda: self.setCharge(1,"+1")),
( " 0 ", "Neutral Charge", lambda: self.setCharge(0,"neutral")),
( " -1 ", "Negative Charge", lambda: self.setCharge(-1,"-1")),
],
[
( "@Bonds:", None, None),
( "Create", "Create bond between pk1 and pk2", self.createBond),
( "Delete", "Delete bond between pk1 and pk2", self.deleteBond),
( "Cycle", "Cycle bond valence", self.cycleBond),
( " | ", "Create single bond", lambda: self.setOrder("1", "single")),
( " || ", "Create double bond", lambda: self.setOrder("2", "double")),
( " ||| ", "Create triple bond", lambda: self.setOrder("3", "triple")),
( "Arom", "Create aromatic bond", lambda: self.setOrder("4", "aromatic")),
( "@ Model:", None, None),
( "Clean", "Cleanup structure", self.clean),
( "Sculpt", "Molecular sculpting", self.sculpt),
( "Fix", "Fix atom positions", self.fix),
( "Rest", "Restrain atom positions", self.rest),
],
[
( "$El-stat", "Electrostatics term for 'Clean' action", "clean_electro_mode"),
( "@ ", None, None),
( "$Bumps", "Show VDW contacts during sculpting", "sculpt_vdw_vis_mode"),
( "@ ", None, None),
( "#Undo Enabled", "", "suspend_undo"),
( "Undo", "Undo last change", self.undo),
( "Redo", "Redo last change", self.redo),
]
]
for row, btn_row in enumerate(buttons):
btn_row_layout = QtWidgets.QHBoxLayout()
self.buttons_layout.addLayout(btn_row_layout)
for col, bb in enumerate(btn_row):
btn_label, btn_tooltip, btn_command = bb
if btn_label[0] == '@':
btn = QtWidgets.QLabel(btn_label[1:])
elif btn_label[0] in ('#', '$'):
btn = QtWidgets.QCheckBox(btn_label[1:])
setting = btn_command
value = self.cmd.get_setting_int(setting)
if btn_label[0] == '$':
btn.setChecked(bool(value))
@btn.toggled.connect
def _(checked, n=setting):
self.cmd.set(n, checked, quiet=0)
else:
btn.setChecked(not value)
@btn.toggled.connect
def _(checked, n=setting):
self.cmd.set(n, not checked, quiet=0)
else:
btn = makeFragmentButton()
btn.setText(btn_label)
btn.clicked.connect(btn_command)
if btn_tooltip:
btn.setToolTip(btn_tooltip)
btn_row_layout.addWidget(btn)
btn_row_layout.addStretch()
def getIcons(self):
self.icons = {}
# use old Tk icons
imgDir = os.path.join(os.environ['PYMOL_DATA'], "pmg_tk/bitmaps/builder")
imgList = glob("%s/aro*.gif" % imgDir) + glob("%s/cyc*.gif" % imgDir)
for imgFile in imgList:
imgName = os.path.splitext(os.path.split(imgFile)[1])[0]
if imgName not in list(self.icons.keys()):
image = QtGui.QImage(imgFile)
pixmap = QtGui.QPixmap.fromImage(image)
image.invertPixels()
inv_pixmap = QtGui.QPixmap.fromImage(image)
self.icons[imgName] = (QtGui.QIcon(pixmap), QtGui.QIcon(inv_pixmap))
def showEvent(self, event):
self.cmd.set("editor_auto_measure", 0)
self.cmd.set("auto_overlay")
self.cmd.set("valence")
self.cmd.edit_mode(1)
def grow(self, name, pos, geom, text):
if "pk1" in self.cmd.get_names("selections"):
self.cmd.select(active_sele,"byobj pk1")
editor.attach_fragment("pk1", name, pos, geom, _self=self.cmd)
self.doAutoPick()
else:
self.cmd.unpick()
AttachWizard(self.cmd).toggle(name, pos, geom, text)
def replace(self, atom, geometry, valence, text):
picked = collectPicked(self.cmd)
if len(picked):
self.cmd.select(active_sele,"byobj "+picked[0])
self.cmd.replace(atom, geometry, valence)
self.doAutoPick()
else:
ReplaceWizard(_self=self.cmd).toggle(atom,geometry,valence,text)
def attach(self, aa):
picked = collectPicked(self.cmd)
if len(picked)==1:
try:
with undocontext(self.cmd, "bymol %s" % picked[0]):
editor.attach_amino_acid(picked[0], aa,
ss=self.ss_cbox.currentIndex() + 1, _self=self.cmd)
except:
fin = -1
self.doZoom()
else:
self.cmd.unpick()
AminoAcidWizard(_self=self.cmd).toggle(aa)
def doAutoPick(self, old_atoms=None):
self.cmd.unpick()
if self.cmd.select(newest_sele,"(byobj "+active_sele+") and not "+active_sele)==0:
self.cmd.select(newest_sele, active_sele)
new_list = self.cmd.index(newest_sele+" and hydro")
if len(new_list)==0:
new_list = self.cmd.index(newest_sele)
if new_list:
index = new_list.pop()
try:
self.cmd.edit("%s`%d" % index)
if self.cmd.get_wizard()!=None:
self.cmd.do("_ cmd.get_wizard().do_pick(0)")
except pymol.CmdException:
print(" doAutoPick-Error: exception")
self.doZoom()
def doZoom(self, *ignore):
if "pk1" in self.cmd.get_names("selections"):
self.cmd.zoom("((neighbor pk1) extend 4)", 4.0, animate=-1)
def setCharge(self, charge, text):
picked = collectPicked(self.cmd)
if len(picked)>0:
sele = "?pk1 ?pk2 ?pk3 ?pk4"
with undocontext(self.cmd, sele):
self.cmd.alter(sele,"formal_charge=%s" % charge)
self.cmd.h_fill()
self.cmd.label(sele,'"%+d" % formal_charge if formal_charge else ""')
self.cmd.unpick()
else:
ChargeWizard(self.cmd).toggle(charge, text)
def createBond(self):
if not BondWizard.staticaction(self.cmd):
BondWizard(self.cmd).toggle()
def deleteBond(self):
picked = collectPicked(self.cmd)
if picked == ["pk1","pk2"]:
with undocontext(self.cmd, "(?pk1 ?pk2) extend 1"):
self.cmd.unbond("pk1", "pk2")
self.cmd.h_fill()
self.cmd.unpick()
else:
self.cmd.unpick()
UnbondWizard(self.cmd).toggle()
def cycleBond(self):
picked = collectPicked(self.cmd)
if picked == ["pk1","pk2"]:
with undocontext(self.cmd, "(?pk1 ?pk2) extend 1"):
self.cmd.cycle_valence()
self.cmd.unpick()
else:
ValenceWizard(_self=self.cmd).toggle(-1,"Cycle bond")
@undoablemethod("(?pk1 ?pk2) extend 1")
def setOrder(self, order, text):
picked = collectPicked(self.cmd)
if picked == ["pk1","pk2"]:
self.cmd.unbond("pk1", "pk2")
self.cmd.bond("pk1", "pk2", order)
self.cmd.h_fill()
self.cmd.unpick()
else:
self.cmd.unpick()
ValenceWizard(_self=self.cmd).toggle(order,text)
def fixH(self):
picked = collectPicked(self.cmd)
if len(picked):
self.cmd.h_fill()
self.cmd.unpick()
else:
HydrogenWizard(_self=self.cmd).toggle('fix')
def addH(self):
picked = collectPicked(self.cmd)
if len(picked):
self.cmd.h_add("pkmol")
self.cmd.unpick()
else:
HydrogenWizard(_self=self.cmd).toggle('add')
def invert(self):
picked = collectPicked(self.cmd)
if picked == ["pk1","pk2","pk3"]:
self.cmd.invert()
self.cmd.unpick()
else:
self.cmd.unpick()
InvertWizard(self.cmd).toggle()
def center(self):
if "pk1" in self.cmd.get_names("selections"):
self.cmd.zoom("pk1", 5.0, animate=-1)
else:
self.cmd.zoom("all", 3.0, animate=-1)
def removeAtom(self):
picked = collectPicked(self.cmd)
if len(picked):
if self.cmd.count_atoms("?pkbond"):
self.cmd.edit("(pk1)","(pk2)",pkbond=0)
cnt = self.cmd.select(active_sele,
"(((?pkset or ?pk1) and not hydro) extend 1) and not hydro")
with undocontext(self.cmd, "(?pkset ?pk1) extend 1"):
self.cmd.remove_picked()
if cnt:
self.cmd.fix_chemistry(active_sele)
self.cmd.h_add(active_sele)
self.cmd.delete(active_sele)
self.cmd.unpick()
else:
RemoveWizard(self.cmd).toggle()
def reset(self):
self.cmd.unpick()
def clear(self):
QMB = QtWidgets.QMessageBox
check = QMB.question(None, "Confirm",
"Really delete everything?", QMB.Yes | QMB.No)
if check == QMB.Yes:
self.cmd.delete("all")
self.cmd.refresh_wizard()
def sculpt(self):
picked = collectPicked(self.cmd)
if len(picked):
self.cmd.select(active_sele, ' or '.join(picked))
SculptWizard(_self=self.cmd).toggle()
def clean(self):
picked = collectPicked(self.cmd)
if len(picked):
self.cmd.select(active_sele, "pkmol")
self.cmd.unpick()
CleanWizard(_self=self.cmd).toggle()
def undo(self):
self.cmd.undo()
def redo(self):
self.cmd.redo()
def fix(self):
picked = collectPicked(self.cmd)
if len(picked):
self.cmd.select(active_sele,"pk1")
self.cmd.deselect()
else:
self.cmd.delete(active_sele)
FixAtomWizard(_self=self.cmd).toggle(3)
def rest(self):
picked = collectPicked(self.cmd)
if len(picked):
self.cmd.select(active_sele,"byobj ("+' or '.join(picked)+")")
self.cmd.deselect()
else:
self.cmd.delete(active_sele)
RestAtomWizard(_self=self.cmd).toggle(2)
def BuilderPanelDocked(parent, *args, **kwargs):
widget = _BuilderPanel(parent, *args, **kwargs)
window = QtWidgets.QDockWidget(parent)
window.setWindowTitle("Builder")
window.setWidget(widget)
window.setFloating(True)
return window