1 package AtomicDescriptors::AtomicDescriptors; 2 # 3 # File: AtomicDescriptors.pm 4 # Author: Manish Sud <msud@san.rr.com> 5 # 6 # Copyright (C) 2025 Manish Sud. All rights reserved. 7 # 8 # This file is part of MayaChemTools. 9 # 10 # MayaChemTools is free software; you can redistribute it and/or modify it under 11 # the terms of the GNU Lesser General Public License as published by the Free 12 # Software Foundation; either version 3 of the License, or (at your option) any 13 # later version. 14 # 15 # MayaChemTools is distributed in the hope that it will be useful, but without 16 # any warranty; without even the implied warranty of merchantability of fitness 17 # for a particular purpose. See the GNU Lesser General Public License for more 18 # details. 19 # 20 # You should have received a copy of the GNU Lesser General Public License 21 # along with MayaChemTools; if not, see <http://www.gnu.org/licenses/> or 22 # write to the Free Software Foundation Inc., 59 Temple Place, Suite 330, 23 # Boston, MA, 02111-1307, USA. 24 # 25 26 use strict; 27 use Carp; 28 use Exporter; 29 use Scalar::Util (); 30 use ObjectProperty; 31 use TextUtil (); 32 33 use vars qw(@ISA @EXPORT @EXPORT_OK %EXPORT_TAGS); 34 35 @ISA = qw(ObjectProperty Exporter); 36 @EXPORT = qw(); 37 @EXPORT_OK = qw(); 38 39 %EXPORT_TAGS = (all => [@EXPORT, @EXPORT_OK]); 40 41 # Setup class variables... 42 my($ClassName); 43 _InitializeClass(); 44 45 # Class constructor... 46 sub new { 47 my($Class, %PropertyNamesAndValues) = @_; 48 49 # Initialize object... 50 my $This = {}; 51 bless $This, ref($Class) || $Class; 52 $This->_InitializeAtomicDescriptors(); 53 54 $This->_InitializeAtomicDescriptorsProperties(%PropertyNamesAndValues); 55 56 return $This; 57 } 58 59 # Initialize object data... 60 # 61 sub _InitializeAtomicDescriptors { 62 my($This) = @_; 63 64 # Molecule object... 65 $This->{Molecule} = ''; 66 67 # Type of atomic descriptors... 68 $This->{Type} = ''; 69 70 # By default, atomic decriptor values are also calculated for hydrogens... 71 $This->{IgnoreHydrogens} = 0; 72 73 # Calculated atomic descriptor values hash. Instead of assigning the calculated values to Atom 74 # objects, these values are stored in the current object in a hash with atom ID and atomic descriptor 75 # values as key/value pairs. 76 # 77 # Unlike molecular descriptors, no descriptor names are assigned to individual atomic descriptor 78 # values. 79 # 80 %{$This->{DescriptorValues}} = (); 81 } 82 83 # Initialize class ... 84 sub _InitializeClass { 85 #Class name... 86 $ClassName = __PACKAGE__; 87 } 88 89 # Initialize object properties.... 90 sub _InitializeAtomicDescriptorsProperties { 91 my($This, %PropertiesNamesAndValues) = @_; 92 93 my($Name, $Value, $MethodName); 94 while (($Name, $Value) = each %PropertiesNamesAndValues) { 95 $MethodName = "Set${Name}"; 96 $This->$MethodName($Value); 97 } 98 99 return $This; 100 } 101 102 # Initialize descriptor values for all atoms in a molecule... 103 # 104 sub _InitializeDescriptorValues { 105 my($This) = @_; 106 107 if (!$This->{Molecule}) { 108 return $This; 109 } 110 111 # Assign 'None' to all atomic descriptor values... 112 # 113 my($Atom, $AtomID); 114 115 ATOM: for $Atom ($This->{Molecule}->GetAtoms()) { 116 $AtomID = $Atom->GetID(); 117 $This->{DescriptorValues}{$AtomID} = 'None'; 118 } 119 120 return $This; 121 } 122 123 # Set molecule object and make sure it's not already set... 124 # 125 sub SetMolecule { 126 my($This, $Molecule) = @_; 127 128 if ($This->{Molecule}) { 129 croak "Error: ${ClassName}->SetMolecule: Can't change molecule object: It's already set..."; 130 } 131 $This->{Molecule} = $Molecule; 132 133 # Weaken the reference to disable increment of reference count... 134 Scalar::Util::weaken($This->{Molecule}); 135 136 return $This; 137 } 138 139 # Set type and make sure it's not already set... 140 # 141 sub SetType { 142 my($This, $Type) = @_; 143 144 if ($This->{Type}) { 145 croak "Error: ${ClassName}->SetType: Can't change AtomicDescriptors type: It's already set..."; 146 } 147 $This->{Type} = $Type; 148 149 return $This; 150 } 151 152 # Set specific atomic descriptor value... 153 # 154 sub SetDescriptorValue { 155 my($This, $Atom, $AtomicDescriptor) = @_; 156 my($AtomID); 157 158 $AtomID = $Atom->GetID(); 159 $This->{DescriptorValues}{$AtomID} = $AtomicDescriptor; 160 161 return $This; 162 } 163 164 # Get specific atomic descriptor value... 165 # 166 sub GetDescriptorValue { 167 my($This, $Atom) = @_; 168 my($AtomID); 169 170 $AtomID = $Atom->GetID(); 171 172 return exists $This->{DescriptorValues}{$AtomID} ? $This->{DescriptorValues}{$AtomID} : 'None'; 173 } 174 175 # Get calculated atomic descriptor values as a hash with atom ID and atomic descriptor 176 # values as key/value pairs... 177 # 178 sub GetDescriptorValues { 179 my($This) = @_; 180 181 return %{$This->{DescriptorValues}}; 182 } 183 184 # Are all atomic descriptor values successfully calculated? 185 # 186 # Notes: 187 # . Dynamic checking of calculated descriptor values for atoms eliminates the need 188 # to check and synchronize valid descriptor values during SetDescriptorValue. 189 # 190 sub IsDescriptorsGenerationSuccessful { 191 my($This) = @_; 192 my($Atom, $DescriptorValue, @Atoms); 193 194 ATOM: for $Atom ($This->{Molecule}->GetAtoms()) { 195 if ($This->{IgnoreHydrogens} && $Atom->IsHydrogen()) { 196 next ATOM; 197 } 198 $DescriptorValue = $This->GetDescriptorValue($Atom); 199 if ($DescriptorValue =~ /^None$/i) { 200 return 0; 201 } 202 } 203 204 return 1; 205 } 206