1 package AtomicDescriptors::EStateValuesDescriptors; 2 # 3 # File: EStateValuesDescriptors.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 Matrix; 31 use Constants; 32 use TextUtil (); 33 use MathUtil (); 34 use StatisticsUtil (); 35 use Atom; 36 use Molecule; 37 use AtomicDescriptors::AtomicDescriptors; 38 39 use vars qw(@ISA @EXPORT @EXPORT_OK %EXPORT_TAGS); 40 41 @ISA = qw(AtomicDescriptors::AtomicDescriptors Exporter); 42 @EXPORT = qw(); 43 @EXPORT_OK = qw(); 44 45 %EXPORT_TAGS = (all => [@EXPORT, @EXPORT_OK]); 46 47 # Setup class variables... 48 my($ClassName); 49 _InitializeClass(); 50 51 # Overload Perl functions... 52 use overload '""' => 'StringifyEStateValuesDescriptors'; 53 54 # Class constructor... 55 sub new { 56 my($Class, %NamesAndValues) = @_; 57 58 # Initialize object... 59 my $This = $Class->SUPER::new(); 60 bless $This, ref($Class) || $Class; 61 $This->_InitializeEStateValuesDescriptors(); 62 63 $This->_InitializeEStateValuesDescriptorsProperties(%NamesAndValues); 64 65 return $This; 66 } 67 68 # Initialize class ... 69 sub _InitializeClass { 70 #Class name... 71 $ClassName = __PACKAGE__; 72 } 73 74 75 # Initialize object data... 76 # 77 sub _InitializeEStateValuesDescriptors { 78 my($This) = @_; 79 80 # Type of AtomicDescriptor... 81 $This->{Type} = 'EStateValue'; 82 83 # Intrinsic state values calculated for non-hydrogen atom... 84 # 85 %{$This->{IStateValues}} = (); 86 87 # Calculatetion of E-state values for types for hydrogens is not supported... 88 $This->{IgnoreHydrogens} = 1; 89 90 # Perturbation to intrinsic state values of atoms corresponding to all non-hydrogen 91 # atom pairs... 92 # 93 %{$This->{DeltaIStateMatrix}} = (); 94 95 # E-state values calculated for non-hydrogen atoms... 96 # 97 %{$This->{EStateValues}} = (); 98 99 return $This; 100 } 101 102 # Initialize object properties... 103 # 104 sub _InitializeEStateValuesDescriptorsProperties { 105 my($This, %NamesAndValues) = @_; 106 107 my($Name, $Value, $MethodName); 108 while (($Name, $Value) = each %NamesAndValues) { 109 $MethodName = "Set${Name}"; 110 $This->$MethodName($Value); 111 } 112 113 # Make sure molecule object was specified... 114 if (!exists $NamesAndValues{Molecule}) { 115 croak "Error: ${ClassName}->New: Object can't be instantiated without specifying molecule..."; 116 } 117 118 # Intialize atomic descriptor values... 119 $This->_InitializeDescriptorValues(); 120 121 return $This; 122 } 123 124 # Disable change of ignore hydrogens... 125 # 126 sub SetIgnoreHydrogens { 127 my($This, $IgnoreHydrogens) = @_; 128 129 carp "Warning: ${ClassName}->SetIgnoreHydrogens: Ignore hydrogens value can't be changed: It's not supported..."; 130 131 return $This; 132 } 133 134 # Generate electrotopological state (E-state) values [ Ref 75-78 ] for all atoms 135 # in the molecule... 136 # 137 # Calculation of E-state values for non-hydrogen atoms: 138 # 139 # Let: 140 # 141 # N = Principal quantum number or period number corresponding to element symbol 142 # 143 # Sigma = Number of sigma electrons involves in bonds to hydrogen and non-hydrogen atoms 144 # attached to atom 145 # = Number of sigma bonds to hydrogen and non-hydrogen atoms attached to atom 146 # PI = Number of PI electrons involved in bonds to non-hydrogen atoms attached to atom 147 # = Number of PI bonds to non-hydrogen atoms attached to atom 148 # 149 # LP = Number of lone pair electrons on atom 150 # 151 # Zv = Number of electrons in valence shell of atom 152 # 153 # X = Number of non-hydrogen atom neighbors or heavy atoms attached to atom 154 # H = Number of implicit and explicit hydrogens for atom 155 # 156 # Delta = Number of sigma electrons involved to bonds to non-hydrogen atoms 157 # DeltaV = ValenceDelta = Number of valence shell electrons not involved in bonding to hydrogen atoms 158 # 159 # Ii = Intrinsic state value for atom i 160 # 161 # DeltaIi = Sum of perturbations to intrinsic state value Ii of atom i by all other atoms besides atom i 162 # 163 # DeltaIij = Perturbation to intrinsic state value Ii of atom i by atom j 164 # 165 # Dij = Graph/bond distance between atom i and j 166 # Rij = Dij + 1 167 # 168 # Si = E-state value for atom i 169 # 170 # 171 # Then: 172 # 173 # Delta = Sigma - H = X 174 # 175 # DeltaV = Zv - H 176 # = Sigma + PI + LP - H 177 # 178 # Ii = ( ( ( 2 / N ) ** 2 ) * DeltaV + 1 ) / Delta 179 # 180 # Si = Ii + DeltaIi 181 # 182 # DeltaIij = (Ii - Ij) / (Rij ** 2) for j not equal to i 183 # 184 # DeltaIji = - DeltaIij 185 # 186 # DeltaIi = SUM ( (Ii - Ij) / (Rij ** 2) ) for j = 1 to num of atoms skipping atom i 187 # 188 # Methodology: 189 # . Calculate intrinsic state values for atoms. 190 # . Generate a distance matrix. 191 # . Use distance matrix to calculate DeltaIij matrix with each row i containing 192 # DeltaIij values corresponding to perturbation to intrinsic state value of atom 193 # i by atom j. 194 # . Calculate E-state values using DeltaIij matrix. 195 # . Assign E-state values to atoms. 196 # 197 # Notes: 198 # . The current release of MayaChemTools doesn't support calculation of Hydrogen 199 # E-state values. 200 # 201 sub GenerateDescriptors { 202 my($This) = @_; 203 204 # Cache appropriate molecule data... 205 $This->_SetupMoleculeDataCache(); 206 207 # Generate distance matrix... 208 if (!$This->_SetupDistanceMatrix()) { 209 carp "Warning: ${ClassName}->GenerateDescriptors: E-state values description generation didn't succeed: Couldn't generate a distance matrix..."; 210 return $This; 211 } 212 213 # Calculate EState values.. 214 if (!$This->_CalculateEStateValuesDescriptors()) { 215 carp "Warning: ${ClassName}->GenerateDescriptors: E-state values description generation didn't succeed: Couldn't calculate IState values for all atoms..."; 216 return $This; 217 } 218 219 # Set final descriptor values... 220 $This->_SetFinalValues(); 221 222 # Clear cached molecule data... 223 $This->_ClearMoleculeDataCache(); 224 225 return $This; 226 } 227 228 # Calculate E-state values for non-hydrogen atoms... 229 # 230 sub _CalculateEStateValuesDescriptors { 231 my($This) = @_; 232 my($DeltaIStateMatrix, $NumOfRows, $NumOfCols, $RowIndex, $AtomID, $EStateValue, $IStateValue, $DeltaIStateValue, @DeltaIStateMatrixRowValues); 233 234 %{$This->{EStateValues}} = (); 235 236 # Calculate intrinsic state values for non-hydrogen atoms... 237 if (!$This->_CalculateIStateValues()) { 238 return undef; 239 } 240 241 # Calculate delta intrinsic state matrix for non-hydrogen atoms... 242 $This->_CalculateDeltaIStateMatrix(); 243 244 # Get DeltaIState matrix information... 245 $DeltaIStateMatrix = $This->{DeltaIStateMatrix}; 246 ($NumOfRows, $NumOfCols) = $DeltaIStateMatrix->GetSize(); 247 248 # Calculate EState values... 249 ROWINDEX: for $RowIndex (0 .. ($NumOfRows - 1) ) { 250 $AtomID = $This->{AtomIndexToID}{$RowIndex}; 251 if ( !(exists($This->{IStateValues}{$AtomID})) ) { 252 next ROWINDEX; 253 } 254 $IStateValue = $This->{IStateValues}{$AtomID}; 255 256 @DeltaIStateMatrixRowValues = $DeltaIStateMatrix->GetRowValues($RowIndex); 257 $DeltaIStateValue = StatisticsUtil::Sum(\@DeltaIStateMatrixRowValues); 258 259 $EStateValue = $IStateValue + $DeltaIStateValue; 260 261 $This->{EStateValues}{$AtomID} = $EStateValue; 262 } 263 return $This; 264 } 265 266 # Calculate intrinsic state values for non-hydrogen atoms... 267 # 268 sub _CalculateIStateValues { 269 my($This) = @_; 270 my($Atom, $AtomID); 271 272 %{$This->{IStateValues}} = (); 273 274 ATOM: for $Atom (@{$This->{Atoms}}) { 275 # Irrespective of IgoreHydrogens value, just ignore hydrogens... 276 if ($Atom->IsHydrogen()) { 277 next ATOM; 278 } 279 $AtomID = $Atom->GetID(); 280 $This->{IStateValues}{$AtomID} = $This->_CalculateIStateValue($Atom); 281 if (!defined($This->{IStateValues}{$AtomID})) { 282 return undef; 283 } 284 } 285 return $This; 286 } 287 288 # Calculation intrinsic state value for non-hydrogen... 289 # 290 sub _CalculateIStateValue { 291 my($This, $Atom) = @_; 292 my($IStateValue, $Delta, $DeltaV, $PeriodNumber); 293 294 $PeriodNumber = $Atom->GetPeriodNumber(); 295 ($Delta, $DeltaV) = $This->_GetDeltaValues($Atom); 296 297 if (!(defined($Delta) && defined($PeriodNumber) && defined($DeltaV))) { 298 return undef; 299 } 300 301 $IStateValue = ($PeriodNumber && $Delta) ? (((2/$PeriodNumber)**2)*$DeltaV + 1)/$Delta : 0; 302 303 return $IStateValue; 304 } 305 306 # Get Delta and DeltaV values for atom... 307 # 308 sub _GetDeltaValues { 309 my($This, $Atom) = @_; 310 my($Delta, $DeltaV, $ValenceElectrons, $NumOfHydrogens); 311 312 ($Delta, $DeltaV) = (undef, undef); 313 314 $ValenceElectrons = $Atom->GetValenceElectrons(); 315 $NumOfHydrogens = $Atom->GetAtomicInvariantValue('H'); 316 317 $Delta = $Atom->GetAtomicInvariantValue('X'); 318 if (defined($ValenceElectrons) && defined($NumOfHydrogens)) { 319 $DeltaV = $ValenceElectrons - $NumOfHydrogens; 320 } 321 322 return ($Delta, $DeltaV); 323 } 324 325 # Calculate DeltaIState matrix for atoms with each row i containing DeltaIij values 326 # corresponding atom atoms i and j. 327 # 328 # Notes: 329 # . Matrix elements corresponding to hydrogen atoms or unconnected 330 # are assigned zero value. 331 # 332 sub _CalculateDeltaIStateMatrix { 333 my($This) = @_; 334 my($DistanceMatrix, $NumOfRows, $NumOfCols, $RowIndex, $ColIndex, $AtomID1, $AtomID2, $DeltaIStateMatrix, $IStateValue1, $IStateValue2, $GraphDistance, $DeltaIState12, $DeltaIState21, $SkipIndexCheck); 335 336 # Get distance matrix information... 337 $DistanceMatrix = $This->{DistanceMatrix}; 338 ($NumOfRows, $NumOfCols) = $DistanceMatrix->GetSize(); 339 340 # Initialize DeltaIState matrix... 341 $This->{DeltaIStateMatrix} = new Matrix($NumOfRows, $NumOfCols); 342 $DeltaIStateMatrix = $This->{DeltaIStateMatrix}; 343 344 $SkipIndexCheck = 1; 345 346 # Calculate DeltaIState matrix values... 347 ROWINDEX: for $RowIndex (0 .. ($NumOfRows - 1) ) { 348 $AtomID1 = $This->{AtomIndexToID}{$RowIndex}; 349 if (!exists($This->{IStateValues}{$AtomID1})) { 350 next ROWINDEX; 351 } 352 $IStateValue1 = $This->{IStateValues}{$AtomID1}; 353 354 COLINDEX: for $ColIndex (($RowIndex + 1) .. ($NumOfCols - 1) ) { 355 $AtomID2 = $This->{AtomIndexToID}{$ColIndex}; 356 if (!exists($This->{IStateValues}{$AtomID2})) { 357 next COLINDEX; 358 } 359 $IStateValue2 = $This->{IStateValues}{$AtomID2}; 360 361 # Make sure it's a connected atom... 362 $GraphDistance = $DistanceMatrix->GetValue($RowIndex, $ColIndex, $SkipIndexCheck); 363 if ($GraphDistance >= BigNumber) { 364 next COLINDEX; 365 } 366 367 $DeltaIState12 = ($IStateValue1 - $IStateValue2)/(($GraphDistance + 1)**2); 368 $DeltaIState21 = -$DeltaIState12; 369 370 # Set DeltaIState values... 371 $DeltaIStateMatrix->SetValue($RowIndex, $ColIndex, $DeltaIState12, $SkipIndexCheck); 372 $DeltaIStateMatrix->SetValue($ColIndex, $RowIndex, $DeltaIState21, $SkipIndexCheck); 373 } 374 } 375 } 376 377 # Setup distance matrix... 378 # 379 sub _SetupDistanceMatrix { 380 my($This) = @_; 381 382 $This->{DistanceMatrix} = $This->GetMolecule()->GetDistanceMatrix(); 383 384 if (!defined($This->{DistanceMatrix})) { 385 return undef; 386 } 387 388 return $This; 389 } 390 391 # Setup final descriptor values... 392 # 393 sub _SetFinalValues { 394 my($This) = @_; 395 my($Atom, $AtomID, $EStateValue); 396 397 ATOM: for $Atom (@{$This->{Atoms}}) { 398 $AtomID = $Atom->GetID(); 399 if (!exists $This->{EStateValues}{$AtomID}) { 400 next ATOM; 401 } 402 $EStateValue = $This->{EStateValues}{$AtomID}; 403 $This->SetDescriptorValue($Atom, $EStateValue); 404 } 405 406 return $This; 407 } 408 409 # Cache appropriate molecule data... 410 # 411 sub _SetupMoleculeDataCache { 412 my($This) = @_; 413 414 # Get all atoms including hydrogens to correctly map atom indicies to atom IDs for 415 # usage of distance matrix. 416 # 417 @{$This->{Atoms}} = $This->GetMolecule()->GetAtoms(); 418 419 # Get all atom IDs... 420 my(@AtomIDs); 421 @AtomIDs = (); 422 @AtomIDs = map { $_->GetID() } @{$This->{Atoms}}; 423 424 # Set AtomIndex to AtomID hash... 425 %{$This->{AtomIndexToID}} = (); 426 @{$This->{AtomIndexToID}}{ (0 .. $#AtomIDs) } = @AtomIDs; 427 428 return $This; 429 } 430 431 # Clear cached molecule data... 432 # 433 sub _ClearMoleculeDataCache { 434 my($This) = @_; 435 436 @{$This->{Atoms}} = (); 437 438 return $This; 439 } 440 441 # Return a string containg data for EStateValuesDescriptors object... 442 # 443 sub StringifyEStateValuesDescriptors { 444 my($This) = @_; 445 my($EStateValuesDescriptorsString); 446 447 # Type of AtomicValues... 448 $EStateValuesDescriptorsString = "AtomicDescriptorType: $This->{Type}; IgnoreHydrogens: " . ($This->{IgnoreHydrogens} ? "Yes" : "No"); 449 450 # Setup atomic descriptor information... 451 my($AtomID, $DescriptorValue, @DescriptorValuesInfo, %DescriptorValues); 452 453 @DescriptorValuesInfo = (); 454 %DescriptorValues = $This->GetDescriptorValues(); 455 456 for $AtomID (sort { $a <=> $b } keys %DescriptorValues) { 457 $DescriptorValue = $DescriptorValues{$AtomID}; 458 $DescriptorValue = (TextUtil::IsEmpty($DescriptorValue) || $DescriptorValue =~ /^None$/i) ? 'None' : MathUtil::round($DescriptorValue, 3) + 0; 459 push @DescriptorValuesInfo, "$AtomID:$DescriptorValue"; 460 } 461 $EStateValuesDescriptorsString .= "; AtomIDs:EStateValuesDescriptors: <" . TextUtil::JoinWords(\@DescriptorValuesInfo, ", ", 0) . ">"; 462 463 return $EStateValuesDescriptorsString; 464 } 465 466 # Is it a EStateValuesDescriptors object? 467 sub _IsEStateValuesDescriptors { 468 my($Object) = @_; 469 470 return (Scalar::Util::blessed($Object) && $Object->isa($ClassName)) ? 1 : 0; 471 } 472