Newer
Older
# ##### BEGIN GPL LICENSE BLOCK #####
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software Foundation,
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# ##### END GPL LICENSE BLOCK #####
bl_info = {
"name": "PDB Atomic Blender",
"description": "Loading and manipulating atoms from PDB files",
"author": "Clemens Barth",
"blender": (2,6),
"location": "File -> Import -> PDB (.pdb), Panel: View 3D - Tools",
"warning": "",
"wiki_url": "http://wiki.blender.org/index.php/Extensions:2.6/"
"Py/Scripts/Import-Export/PDB",
from bpy.types import Operator, Panel
Thomas Larsson
committed
from bpy_extras.io_utils import ImportHelper, ExportHelper
from bpy.props import (StringProperty,
BoolProperty,
EnumProperty,
IntProperty,
FloatProperty)
# TODO, allow reload
from . import import_pdb
Thomas Larsson
committed
from . import export_pdb
# -----------------------------------------------------------------------------
# GUI
# The panel, which is loaded after the file has been
# chosen via the menu 'File -> Import'
class CLASS_atom_pdb_panel(Panel):
bl_label = "PDB - Atomic Blender"
bl_space_type = "VIEW_3D"
bl_region_type = "TOOL_PROPS"
@classmethod
def poll(self, context):
global ATOM_PDB_PANEL
if ATOM_PDB_PANEL == "0" and import_pdb.ATOM_PDB_FILEPATH == "":
if ATOM_PDB_PANEL == "0" and import_pdb.ATOM_PDB_FILEPATH != "":
return True
if ATOM_PDB_PANEL == "1":
row.label(text="Outputs and custom data file")
box = layout.box()
row = box.row()
row.label(text="Reload structure")
box = layout.box()
row = box.row()
col = row.column()
col.label(text="Scaling factors")
row = box.row()
col = row.column(align=True)
col.active = scn.use_mesh
col.prop(scn, "mesh_azimuth")
col.prop(scn, "mesh_zenith")
col = row.column(align=True)
col.prop(scn, "sticks_sectors")
col.prop(scn, "sticks_radius")
col.prop(scn, "sticks_unit_length")
col.prop(scn, "use_sticks_color")
col.prop(scn, "use_sticks_smooth")
col.prop(scn, "use_sticks_bonds")
col.active = scn.use_sticks and scn.use_sticks_bonds
col.prop(scn, "sticks_dist")
row.label(text="Modify atom radii")
box = layout.box()
row = box.row()
row.label(text="1. Change type of radii")
row.label(text="2. Change atom radii in pm")
row.label(text="3. Change atom radii by scale")
col = row.column(align=True)
col.operator( "atom_pdb.radius_all_bigger" )
col.operator( "atom_pdb.radius_all_smaller" )
row = box.row()
row.label(text="4. Show sticks only")
row = box.row()
col = row.column()
col.operator( "atom_pdb.radius_sticks" )
row.label(text="Separate atom")
box = layout.box()
row = box.row()
# The properties (gadgets) in the panel. They all go to scene
# during initialization (see end)
class CLASS_atom_pdb_Properties(bpy.types.PropertyGroup):
name="Camera", default=False,
description="Do you need a camera?")
name="Lamp", default=False,
description = "Do you need a lamp?")
name = "Mesh balls", default=False,
scale_ballradius = FloatProperty(
name = "Balls", default=1.0, min=0.0001,
scale_distances = FloatProperty (
name = "Distances", default=1.0, min=0.0001,
name = "Object to origin", default=True,
description="Number of sectors of a stick")
sticks_radius = FloatProperty(
name = "Radius", default=0.1, min=0.0001,
description ="Radius of a stick")
sticks_unit_length = FloatProperty(
name = "Unit", default=0.2, min=0.0001,
name="Smooth", default=False,
description="The sticks are round (sectors are not visible)")
Thomas Larsson
committed
description="Show double and tripple bonds.")
name="Distance", default = 1.1, min=1.0, max=3.0,
description="Distance between sticks measured in stick diameter")
name="Type of radius",
description="Choose type of atom radius",
items=(('0', "Pre-defined", "Use pre-defined radius"),
('1', "Atomic", "Use atomic radius"),
('2', "van der Waals", "Use van der Waals radius")),
name = "", description="Path to your custom data file",
description = "Path of the PDB file")
default="Number", description = "This output shows "
"the number of atoms which have been loaded")
name="", default="Distance (A)",
description="Distance of 2 objects in Angstrom")
name="",
description="Which objects shall be modified?",
items=(('ALL_ACTIVE',"all active objects", "in the current layer"),
('ALL_IN_LAYER',"all"," in active layer(s)")),
name="Type",
description="Which type of atom radii?",
items=(('0',"predefined", "Use pre-defined radii"),
('1',"atomic", "Use atomic radii"),
default='0',update=Callback_radius_type)
name="", default="Atom name",
description="Put in the radius of the atom (in pm)",
radius_all = FloatProperty(
name="Scale", default = 1.05, min=1.0, max=5.0,
class CLASS_atom_pdb_datafile_apply(Operator):
return {'FINISHED'}
# TODO, move this into 'import_pdb' and call the function
for obj in bpy.context.selected_objects:
if len(obj.children) != 0:
child = obj.children[0]
if child.type == "SURFACE" or child.type == "MESH":
for element in import_pdb.ATOM_PDB_ELEMENTS:
child.scale = (element.radii[0],) * 3
child.active_material.diffuse_color = element.color
else:
if obj.type == "SURFACE" or obj.type == "MESH":
for element in import_pdb.ATOM_PDB_ELEMENTS:
obj.scale = (element.radii[0],) * 3
return {'FINISHED'}
class CLASS_atom_pdb_separate_atom(Operator):
bl_idname = "atom_pdb.separate_atom"
bl_label = "Separate atom"
bl_description = "Separate the atom you have chosen"
def execute(self, context):
# Get first all important properties from the atom which the user
# has chosen: location, color, scale
obj = bpy.context.edit_object
name = obj.name
loc_obj_vec = obj.location
material = obj.children[0].active_material
# Separate the vertex from the main mesh and create a new mesh.
bpy.ops.mesh.separate()
new_object = bpy.context.scene.objects[0]
# Keep in mind the coordinates <= We only need this
loc_vec = new_object.data.vertices[0].co
bpy.ops.object.mode_set(mode='OBJECT', toggle=False)
bpy.ops.object.select_all(action='DESELECT')
bpy.ops.object.delete() # TODO, use scene.objects.unlink()
current_layers=bpy.context.scene.layers
view_align=False, enter_editmode=False,
location=loc_vec+loc_obj_vec,
rotation=(0.0, 0.0, 0.0),
layers=current_layers)
size=1, view_align=False, enter_editmode=False,
location=loc_vec+loc_obj_vec,
rotation=(0, 0, 0),
layers=current_layers)
else:
view_align=False, enter_editmode=False,
location=loc_vec+loc_obj_vec,
rotation=(0.0, 0.0, 0.0),
layers=current_layers)
new_atom = bpy.context.scene.objects.active
# Scale, material and name it.
new_atom.scale = scale
new_atom.active_material = material
new_atom.name = name + "_sep"
# Switch back into the 'Edit mode' because we would like to seprate
# other atoms may be (more convinient)
new_atom.select = False
obj.select = True
bpy.context.scene.objects.active = obj
bpy.ops.object.select_all(action='DESELECT')
bpy.ops.object.mode_set(mode='EDIT', toggle=False)
return {'FINISHED'}
# Button for measuring the distance of the active objects
class CLASS_atom_pdb_distance_button(Operator):
bl_description = "Measure the distance between two objects (only in Object Mode)"
# The string length is cut, 3 digits after the first 3 digits
# after the '.'. Append also "Angstrom".
# Remember: 1 Angstrom = 10^(-10) m
dist = dist + " A"
# Put the distance into the string of the output field.
return {'FINISHED'}
# Button for increasing the radii of all atoms
class CLASS_atom_pdb_radius_all_bigger_button(Operator):
bl_idname = "atom_pdb.radius_all_bigger"
bl_label = "Bigger ..."
bl_description = "Increase the radii of the atoms"
def execute(self, context):
)
return {'FINISHED'}
# Button for decreasing the radii of all atoms
class CLASS_atom_pdb_radius_all_smaller_button(Operator):
bl_idname = "atom_pdb.radius_all_smaller"
bl_label = "Smaller ..."
bl_description = "Decrease the radii of the atoms"
def execute(self, context):
# Button for showing the sticks only - the radii of the atoms downscaled onto
# 90% of the stick radius
class CLASS_atom_pdb_radius_sticks_button(Operator):
bl_idname = "atom_pdb.radius_sticks"
bl_label = "Show sticks"
bl_description = "Show only the sticks (atom radii = stick radii)"
def execute(self, context):
if result == False:
ATOM_PDB_ERROR = "No sticks => no changes"
bpy.ops.atom_pdb.error_dialog('INVOKE_DEFAULT')
return {'FINISHED'}
# The button for reloading the atoms and creating the scene
class CLASS_atom_pdb_load_button(Operator):
bl_idname = "atom_pdb.button_reload"
bl_label = "RELOAD"
bl_description = "Load the structure again"
scn.use_mesh,
scn.mesh_azimuth,
scn.mesh_zenith,
scn.scale_ballradius,
scn.atomradius,
scn.scale_distances,
scn.use_sticks,
scn.use_sticks_color,
scn.use_sticks_smooth,
scn.use_sticks_bonds,
scn.sticks_unit_length,
scn.sticks_dist,
scn.sticks_sectors,
scn.sticks_radius,
scn.use_center,
scn.use_camera,
scn.use_lamp,
scn.datafile)
scn.number_atoms = str(atom_number) + " atoms"
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
def DEF_panel_yes_no():
global ATOM_PDB_PANEL
datafile_path = bpy.utils.user_resource('SCRIPTS', path='', create=False)
if os.path.isdir(datafile_path) == False:
bpy.utils.user_resource('SCRIPTS', path='', create=True)
datafile_path = os.path.join(datafile_path, "presets")
if os.path.isdir(datafile_path) == False:
os.mkdir(datafile_path)
datafile = os.path.join(datafile_path, "io_mesh_pdb.pref")
if os.path.isfile(datafile):
datafile_fp = io.open(datafile, "r")
for line in datafile_fp:
if "Panel" in line:
ATOM_PDB_PANEL = line[-2:]
ATOM_PDB_PANEL = ATOM_PDB_PANEL[0:1]
bpy.context.scene.use_panel = ATOM_PDB_PANEL
break
datafile_fp.close()
else:
DEF_panel_write_pref("0")
def DEF_panel_write_pref(value):
datafile_path = bpy.utils.user_resource('SCRIPTS', path='', create=False)
datafile_path = os.path.join(datafile_path, "presets")
datafile = os.path.join(datafile_path, "io_mesh_pdb.pref")
datafile_fp = io.open(datafile, "w")
datafile_fp.write("Atomic Blender PDB - Import/Export - Preferences\n")
datafile_fp.write("================================================\n")
datafile_fp.write("\n")
datafile_fp.write("Panel: "+value+"\n\n\n")
datafile_fp.close()
class CLASS_atom_pdb_error_dialog(bpy.types.Operator):
bl_idname = "atom_pdb.error_dialog"
bl_label = "Attention !"
def draw(self, context):
layout = self.layout
row = layout.row()
row.label(text=" "+ATOM_PDB_ERROR)
def execute(self, context):
print("Atomic Blender - Error: "+ATOM_PDB_ERROR+"\n")
return {'FINISHED'}
def invoke(self, context, event):
return context.window_manager.invoke_props_dialog(self)
bl_idname = "import_mesh.pdb"
filename_ext = ".pdb"
filter_glob = StringProperty(default="*.pdb", options={'HIDDEN'},)
bpy.types.Scene.use_panel = EnumProperty(
name="Panel",
description="Choose whether the panel shall appear or not in the View 3D.",
items=(('0', "Once", "The panel appears only in this session"),
('1', "Always", "The panel always appears when Blender is started"),
('2', "Never", "The panel never appears")),
default='0')
use_camera = BoolProperty(
name="Camera", default=False,
description="Do you need a camera?")
use_lamp = BoolProperty(
name="Lamp", default=False,
description = "Do you need a lamp?")
use_mesh = BoolProperty(
name = "Mesh balls", default=False,
description = "Use mesh balls instead of NURBS")
mesh_azimuth = IntProperty(
name = "Azimuth", default=32, min=1,
description = "Number of sectors (azimuth)")
mesh_zenith = IntProperty(
name = "Zenith", default=32, min=1,
description = "Number of sectors (zenith)")
scale_ballradius = FloatProperty(
name = "Balls", default=1.0, min=0.0001,
description = "Scale factor for all atom radii")
scale_distances = FloatProperty (
name = "Distances", default=1.0, min=0.0001,
description = "Scale factor for all distances")
atomradius = EnumProperty(
name="Type of radius",
description="Choose type of atom radius",
items=(('0', "Pre-defined", "Use pre-defined radius"),
('1', "Atomic", "Use atomic radius"),
('2', "van der Waals", "Use van der Waals radius")),
default='0',)
use_sticks = BoolProperty(
name="Use sticks", default=True,
description="Do you want to display the sticks?")
sticks_sectors = IntProperty(
name = "Sector", default=20, min=1,
description="Number of sectors of a stick")
sticks_radius = FloatProperty(
name = "Radius", default=0.1, min=0.0001,
description ="Radius of a stick")
sticks_unit_length = FloatProperty(
name = "Unit", default=0.2, min=0.0001,
description = "Length of the unit of a stick in Angstrom")
use_sticks_color = BoolProperty(
name="Color", default=True,
description="The sticks appear in the color of the atoms")
use_sticks_smooth = BoolProperty(
name="Smooth", default=False,
description="The sticks are round (sectors are not visible)")
use_sticks_bonds = BoolProperty(
name="Bonds", default=False,
description="Show double and tripple bonds.")
sticks_dist = FloatProperty(
name="Distance", default = 1.1, min=1.0, max=3.0,
description="Distance between sticks measured in stick diameter")
use_center = BoolProperty(
name = "Object to origin", default=True,
description = "Put the object into the global origin")
datafile = StringProperty(
name = "", description="Path to your custom data file",
maxlen = 256, default = "", subtype='FILE_PATH')
row = layout.row()
col = row.column()
col = row.column(align=True)
col.active = self.use_mesh
col.prop(self, "mesh_azimuth")
col.prop(self, "mesh_zenith")
row = layout.row()
col = row.column()
col.prop(self, "scale_ballradius")
col.prop(self, "scale_distances")
col.prop(self, "sticks_sectors")
col.prop(self, "sticks_radius")
col.prop(self, "sticks_unit_length")
col.prop(self, "use_sticks_color")
col.prop(self, "use_sticks_smooth")
col.prop(self, "use_sticks_bonds")
col.active = self.use_sticks and self.use_sticks_bonds
col.prop(self, "sticks_dist")
def execute(self, context):
# This is in order to solve this strange 'relative path' thing.
import_pdb.ATOM_PDB_FILEPATH = bpy.path.abspath(self.filepath)
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
self.use_mesh,
self.mesh_azimuth,
self.mesh_zenith,
self.scale_ballradius,
self.atomradius,
self.scale_distances,
self.use_sticks,
self.use_sticks_color,
self.use_sticks_smooth,
self.use_sticks_bonds,
self.sticks_unit_length,
self.sticks_dist,
self.sticks_sectors,
self.sticks_radius,
self.use_center,
self.use_camera,
self.use_lamp,
self.datafile)
# Copy the whole bunch of values into the property collection.
scn = context.scene.atom_pdb[0]
scn.use_mesh = self.use_mesh
scn.mesh_azimuth = self.mesh_azimuth
scn.mesh_zenith = self.mesh_zenith
scn.scale_ballradius = self.scale_ballradius
scn.atomradius = self.atomradius
scn.scale_distances = self.scale_distances
scn.use_sticks = self.use_sticks
scn.use_sticks_color = self.use_sticks_color
scn.use_sticks_smooth = self.use_sticks_smooth
scn.use_sticks_bonds = self.use_sticks_bonds
scn.sticks_unit_length = self.sticks_unit_length
scn.sticks_dist = self.sticks_dist
scn.sticks_sectors = self.sticks_sectors
scn.sticks_radius = self.sticks_radius
scn.use_center = self.use_center
scn.use_camera = self.use_camera
scn.use_lamp = self.use_lamp
scn.datafile = self.datafile
scn.number_atoms = str(atom_number) + " atoms"
scn.PDB_file = import_pdb.ATOM_PDB_FILEPATH
global ATOM_PDB_PANEL
ATOM_PDB_PANEL = bpy.context.scene.use_panel
DEF_panel_write_pref(bpy.context.scene.use_panel)
Thomas Larsson
committed
Thomas Larsson
committed
bl_idname = "export_mesh.pdb"
bl_label = "Export Protein Data Bank(*.pdb)"
filename_ext = ".pdb"
filter_glob = StringProperty(
default="*.pdb", options={'HIDDEN'},)
atom_pdb_export_type = EnumProperty(
name="Type of Objects",
description="Choose type of objects",
items=(('0', "All", "Export all active objects"),
('1', "Elements", "Export only those active objects which have"
" a proper element name")),
Thomas Larsson
committed
def draw(self, context):
layout = self.layout
Thomas Larsson
committed
def execute(self, context):
# This is in order to solve this strange 'relative path' thing.
export_pdb.ATOM_PDB_FILEPATH = bpy.path.abspath(self.filepath)
Thomas Larsson
committed
return {'FINISHED'}
def DEF_menu_func_import(self, context):
self.layout.operator(CLASS_ImportPDB.bl_idname, text="Protein Data Bank (.pdb)")
Thomas Larsson
committed
# The entry into the menu 'file -> export'
def DEF_menu_func_export(self, context):
self.layout.operator(CLASS_ExportPDB.bl_idname, text="Protein Data Bank (.pdb)")
Thomas Larsson
committed
bpy.types.INFO_MT_file_import.append(DEF_menu_func_import)
bpy.types.INFO_MT_file_export.append(DEF_menu_func_export)
bpy.types.Scene.atom_pdb = bpy.props.CollectionProperty(type=CLASS_atom_pdb_Properties)
bpy.context.scene.atom_pdb.add()
bpy.utils.unregister_module(__name__)
bpy.types.INFO_MT_file_import.remove(DEF_menu_func_import)
bpy.types.INFO_MT_file_export.remove(DEF_menu_func_export)