# This Source Code Form is subject to the terms of the Mozilla Public# License, v. 2.0. If a copy of the MPL was not distributed with this# file, You can obtain one at https://mozilla.org/MPL/2.0/.from__future__importannotationsfromcollectionsimportCounterfromcollections.abcimportIterablefromnumbersimportIntegral,RealfromtypingimportLiteralimportnumpyasnpimportsisl._arrayas_afromsisl._dispatch_classimport_Dispatchsfromsisl._dispatcherimportAbstractDispatch,ClassDispatcherfromsisl._helpimportarray_fill_repeatfromsisl._indicesimportlist_index_lefromsisl._internalimportset_modulefromsisl.messagesimportdeprecate_argument,deprecation,infofromsisl.shapeimportSpherefrom.orbitalimportOrbital__all__=["PeriodicTable","Atom","AtomUnknown","AtomGhost","Atoms"]@set_module("sisl")classPeriodicTable:r"""Periodic table for creating an `Atom`, or retrieval of atomic information via atomic numbers Enables *lookup* of atomic numbers/names/labels to get the atomic number. Several quantities available to the atomic species are available from <https://en.wikipedia.org/wiki/Atomic_radii_of_the_elements_(data_page)>. The following values are accessible: * atomic mass (in atomic units) * empirical atomic radius (in Ang) * calculated atomic radius (in Ang) * van der Waals atomic radius (in Ang) For certain species the above quantities are not available and a negative number is returned. Examples -------- >>> 79 == PeriodicTable().Z('Au') True >>> 79 == PeriodicTable().Z_int('Au') True >>> 'Au' == PeriodicTable().Z_short(79) True >>> 'Au' == PeriodicTable().Z_label(79) True >>> 'Au' == PeriodicTable().Z_label('Gold') True >>> 12.0107 == PeriodicTable().atomic_mass('C') True >>> 12.0107 == PeriodicTable().atomic_mass(6) True >>> 12.0107 == PeriodicTable().atomic_mass('Carbon') True >>> .67 == PeriodicTable().radius('Carbon') True >>> .67 == PeriodicTable().radius(6,'calc') True >>> .7 == PeriodicTable().radius(6,'empirical') True >>> 1.7 == PeriodicTable().radius(6,'vdw') True """# fmt: off_Z_int={'Actinium':89,'Ac':89,'89':89,89:89,'Aluminum':13,'Al':13,'13':13,13:13,'Americium':95,'Am':95,'95':95,95:95,'Antimony':51,'Sb':51,'51':51,51:51,'Argon':18,'Ar':18,'18':18,18:18,'Arsenic':33,'As':33,'33':33,33:33,'Astatine':85,'At':85,'85':85,85:85,'Barium':56,'Ba':56,'56':56,56:56,'Berkelium':97,'Bk':97,'97':97,97:97,'Beryllium':4,'Be':4,'4':4,4:4,'Bismuth':83,'Bi':83,'83':83,83:83,'Bohrium':107,'Bh':107,'107':107,107:107,'Boron':5,'B':5,'5':5,5:5,'Bromine':35,'Br':35,'35':35,35:35,'Cadmium':48,'Cd':48,'48':48,48:48,'Calcium':20,'Ca':20,'20':20,20:20,'Californium':98,'Cf':98,'98':98,98:98,'Carbon':6,'C':6,'6':6,6:6,'Cerium':58,'Ce':58,'58':58,58:58,'Cesium':55,'Cs':55,'55':55,55:55,'Chlorine':17,'Cl':17,'17':17,17:17,'Chromium':24,'Cr':24,'24':24,24:24,'Cobalt':27,'Co':27,'27':27,27:27,'Copper':29,'Cu':29,'29':29,29:29,'Curium':96,'Cm':96,'96':96,96:96,'Darmstadtium':110,'Ds':110,'110':110,110:110,'Dubnium':105,'Db':105,'105':105,105:105,'Dysprosium':66,'Dy':66,'66':66,66:66,'Einsteinium':99,'Es':99,'99':99,99:99,'Erbium':68,'Er':68,'68':68,68:68,'Europium':63,'Eu':63,'63':63,63:63,'Fermium':100,'Fm':100,'100':100,100:100,'Fluorine':9,'F':9,'9':9,9:9,'Francium':87,'Fr':87,'87':87,87:87,'Gadolinium':64,'Gd':64,'64':64,64:64,'Gallium':31,'Ga':31,'31':31,31:31,'Germanium':32,'Ge':32,'32':32,32:32,'Gold':79,'Au':79,'79':79,79:79,'Hafnium':72,'Hf':72,'72':72,72:72,'Hassium':108,'Hs':108,'108':108,108:108,'Helium':2,'He':2,'2':2,2:2,'Holmium':67,'Ho':67,'67':67,67:67,'Hydrogen':1,'H':1,'1':1,1:1,'Indium':49,'In':49,'49':49,49:49,'Iodine':53,'I':53,'53':53,53:53,'Iridium':77,'Ir':77,'77':77,77:77,'Iron':26,'Fe':26,'26':26,26:26,'Krypton':36,'Kr':36,'36':36,36:36,'Lanthanum':57,'La':57,'57':57,57:57,'Lawrencium':103,'Lr':103,'103':103,103:103,'Lead':82,'Pb':82,'82':82,82:82,'Lithium':3,'Li':3,'3':3,3:3,'Lutetium':71,'Lu':71,'71':71,71:71,'Magnesium':12,'Mg':12,'12':12,12:12,'Manganese':25,'Mn':25,'25':25,25:25,'Meitnerium':109,'Mt':109,'109':109,109:109,'Mendelevium':101,'Md':101,'101':101,101:101,'Mercury':80,'Hg':80,'80':80,80:80,'Molybdenum':42,'Mo':42,'42':42,42:42,'Neodymium':60,'Nd':60,'60':60,60:60,'Neon':10,'Ne':10,'10':10,10:10,'Neptunium':93,'Np':93,'93':93,93:93,'Nickel':28,'Ni':28,'28':28,28:28,'Niobium':41,'Nb':41,'41':41,41:41,'Nitrogen':7,'N':7,'7':7,7:7,'Nobelium':102,'No':102,'102':102,102:102,'Osmium':76,'Os':76,'76':76,76:76,'Oxygen':8,'O':8,'8':8,8:8,'Palladium':46,'Pd':46,'46':46,46:46,'Phosphorus':15,'P':15,'15':15,15:15,'Platinum':78,'Pt':78,'78':78,78:78,'Plutonium':94,'Pu':94,'94':94,94:94,'Polonium':84,'Po':84,'84':84,84:84,'Potassium':19,'K':19,'19':19,19:19,'Praseodymium':59,'Pr':59,'59':59,59:59,'Promethium':61,'Pm':61,'61':61,61:61,'Protactinium':91,'Pa':91,'91':91,91:91,'Radium':88,'Ra':88,'88':88,88:88,'Radon':86,'Rn':86,'86':86,86:86,'Rhenium':75,'Re':75,'75':75,75:75,'Rhodium':45,'Rh':45,'45':45,45:45,'Rubidium':37,'Rb':37,'37':37,37:37,'Ruthenium':44,'Ru':44,'44':44,44:44,'Rutherfordium':104,'Rf':104,'104':104,104:104,'Samarium':62,'Sm':62,'62':62,62:62,'Scandium':21,'Sc':21,'21':21,21:21,'Seaborgium':106,'Sg':106,'106':106,106:106,'Selenium':34,'Se':34,'34':34,34:34,'Silicon':14,'Si':14,'14':14,14:14,'Silver':47,'Ag':47,'47':47,47:47,'Sodium':11,'Na':11,'11':11,11:11,'Strontium':38,'Sr':38,'38':38,38:38,'Sulfur':16,'S':16,'16':16,16:16,'Tantalum':73,'Ta':73,'73':73,73:73,'Technetium':43,'Tc':43,'43':43,43:43,'Tellurium':52,'Te':52,'52':52,52:52,'Terbium':65,'Tb':65,'65':65,65:65,'Thallium':81,'Tl':81,'81':81,81:81,'Thorium':90,'Th':90,'90':90,90:90,'Thulium':69,'Tm':69,'69':69,69:69,'Tin':50,'Sn':50,'50':50,50:50,'Titanium':22,'Ti':22,'22':22,22:22,'Tungsten':74,'W':74,'74':74,74:74,'Ununbium':112,'Uub':112,'112':112,112:112,'Ununhexium':116,'Uuh':116,'116':116,116:116,'Ununoctium':118,'Uuo':118,'118':118,118:118,'Ununpentium':115,'Uup':115,'115':115,115:115,'Ununquadium':114,'Uuq':114,'114':114,114:114,'Ununseptium':117,'Uus':117,'117':117,117:117,'Ununtrium':113,'Uut':113,'113':113,113:113,'Ununium':111,'Uuu':111,'111':111,111:111,'Uranium':92,'U':92,'92':92,92:92,'Vanadium':23,'V':23,'23':23,23:23,'Xenon':54,'Xe':54,'54':54,54:54,'Ytterbium':70,'Yb':70,'70':70,70:70,'Yttrium':39,'Y':39,'39':39,39:39,'Zinc':30,'Zn':30,'30':30,30:30,'Zirconium':40,'Zr':40,'40':40,40:40,}_Z_short={'Actinium':'Ac','Ac':'Ac','89':'Ac',89:'Ac','Aluminum':'Al','Al':'Al','13':'Al',13:'Al','Americium':'Am','Am':'Am','95':'Am',95:'Am','Antimony':'Sb','Sb':'Sb','51':'Sb',51:'Sb','Argon':'Ar','Ar':'Ar','18':'Ar',18:'Ar','Arsenic':'As','As':'As','33':'As',33:'As','Astatine':'At','At':'At','85':'At',85:'At','Barium':'Ba','Ba':'Ba','56':'Ba',56:'Ba','Berkelium':'Bk','Bk':'Bk','97':'Bk',97:'Bk','Beryllium':'Be','Be':'Be','4':'Be',4:'Be','Bismuth':'Bi','Bi':'Bi','83':'Bi',83:'Bi','Bohrium':'Bh','Bh':'Bh','107':'Bh',107:'Bh','Boron':'B','B':'B','5':'B',5:'B','Bromine':'Br','Br':'Br','35':'Br',35:'Br','Cadmium':'Cd','Cd':'Cd','48':'Cd',48:'Cd','Calcium':'Ca','Ca':'Ca','20':'Ca',20:'Ca','Californium':'Cf','Cf':'Cf','98':'Cf',98:'Cf','Carbon':'C','C':'C','6':'C',6:'C','Cerium':'Ce','Ce':'Ce','58':'Ce',58:'Ce','Cesium':'Cs','Cs':'Cs','55':'Cs',55:'Cs','Chlorine':'Cl','Cl':'Cl','17':'Cl',17:'Cl','Chromium':'Cr','Cr':'Cr','24':'Cr',24:'Cr','Cobalt':'Co','Co':'Co','27':'Co',27:'Co','Copper':'Cu','Cu':'Cu','29':'Cu',29:'Cu','Curium':'Cm','Cm':'Cm','96':'Cm',96:'Cm','Darmstadtium':'Ds','Ds':'Ds','110':'Ds',110:'Ds','Dubnium':'Db','Db':'Db','105':'Db',105:'Db','Dysprosium':'Dy','Dy':'Dy','66':'Dy',66:'Dy','Einsteinium':'Es','Es':'Es','99':'Es',99:'Es','Erbium':'Er','Er':'Er','68':'Er',68:'Er','Europium':'Eu','Eu':'Eu','63':'Eu',63:'Eu','Fermium':'Fm','Fm':'Fm','100':'Fm',100:'Fm','Fluorine':'F','F':'F','9':'F',9:'F','Francium':'Fr','Fr':'Fr','87':'Fr',87:'Fr','Gadolinium':'Gd','Gd':'Gd','64':'Gd',64:'Gd','Gallium':'Ga','Ga':'Ga','31':'Ga',31:'Ga','Germanium':'Ge','Ge':'Ge','32':'Ge',32:'Ge','Gold':'Au','Au':'Au','79':'Au',79:'Au','Hafnium':'Hf','Hf':'Hf','72':'Hf',72:'Hf','Hassium':'Hs','Hs':'Hs','108':'Hs',108:'Hs','Helium':'He','He':'He','2':'He',2:'He','Holmium':'Ho','Ho':'Ho','67':'Ho',67:'Ho','Hydrogen':'H','H':'H','1':'H',1:'H','Indium':'In','In':'In','49':'In',49:'In','Iodine':'I','I':'I','53':'I',53:'I','Iridium':'Ir','Ir':'Ir','77':'Ir',77:'Ir','Iron':'Fe','Fe':'Fe','26':'Fe',26:'Fe','Krypton':'Kr','Kr':'Kr','36':'Kr',36:'Kr','Lanthanum':'La','La':'La','57':'La',57:'La','Lawrencium':'Lr','Lr':'Lr','103':'Lr',103:'Lr','Lead':'Pb','Pb':'Pb','82':'Pb',82:'Pb','Lithium':'Li','Li':'Li','3':'Li',3:'Li','Lutetium':'Lu','Lu':'Lu','71':'Lu',71:'Lu','Magnesium':'Mg','Mg':'Mg','12':'Mg',12:'Mg','Manganese':'Mn','Mn':'Mn','25':'Mn',25:'Mn','Meitnerium':'Mt','Mt':'Mt','109':'Mt',109:'Mt','Mendelevium':'Md','Md':'Md','101':'Md',101:'Md','Mercury':'Hg','Hg':'Hg','80':'Hg',80:'Hg','Molybdenum':'Mo','Mo':'Mo','42':'Mo',42:'Mo','Neodymium':'Nd','Nd':'Nd','60':'Nd',60:'Nd','Neon':'Ne','Ne':'Ne','10':'Ne',10:'Ne','Neptunium':'Np','Np':'Np','93':'Np',93:'Np','Nickel':'Ni','Ni':'Ni','28':'Ni',28:'Ni','Niobium':'Nb','Nb':'Nb','41':'Nb',41:'Nb','Nitrogen':'N','N':'N','7':'N',7:'N','Nobelium':'No','No':'No','102':'No',102:'No','Osmium':'Os','Os':'Os','76':'Os',76:'Os','Oxygen':'O','O':'O','8':'O',8:'O','Palladium':'Pd','Pd':'Pd','46':'Pd',46:'Pd','Phosphorus':'P','P':'P','15':'P',15:'P','Platinum':'Pt','Pt':'Pt','78':'Pt',78:'Pt','Plutonium':'Pu','Pu':'Pu','94':'Pu',94:'Pu','Polonium':'Po','Po':'Po','84':'Po',84:'Po','Potassium':'K','K':'K','19':'K',19:'K','Praseodymium':'Pr','Pr':'Pr','59':'Pr',59:'Pr','Promethium':'Pm','Pm':'Pm','61':'Pm',61:'Pm','Protactinium':'Pa','Pa':'Pa','91':'Pa',91:'Pa','Radium':'Ra','Ra':'Ra','88':'Ra',88:'Ra','Radon':'Rn','Rn':'Rn','86':'Rn',86:'Rn','Rhenium':'Re','Re':'Re','75':'Re',75:'Re','Rhodium':'Rh','Rh':'Rh','45':'Rh',45:'Rh','Rubidium':'Rb','Rb':'Rb','37':'Rb',37:'Rb','Ruthenium':'Ru','Ru':'Ru','44':'Ru',44:'Ru','Rutherfordium':'Rf','Rf':'Rf','104':'Rf',104:'Rf','Samarium':'Sm','Sm':'Sm','62':'Sm',62:'Sm','Scandium':'Sc','Sc':'Sc','21':'Sc',21:'Sc','Seaborgium':'Sg','Sg':'Sg','106':'Sg',106:'Sg','Selenium':'Se','Se':'Se','34':'Se',34:'Se','Silicon':'Si','Si':'Si','14':'Si',14:'Si','Silver':'Ag','Ag':'Ag','47':'Ag',47:'Ag','Sodium':'Na','Na':'Na','11':'Na',11:'Na','Strontium':'Sr','Sr':'Sr','38':'Sr',38:'Sr','Sulfur':'S','S':'S','16':'S',16:'S','Tantalum':'Ta','Ta':'Ta','73':'Ta',73:'Ta','Technetium':'Tc','Tc':'Tc','43':'Tc',43:'Tc','Tellurium':'Te','Te':'Te','52':'Te',52:'Te','Terbium':'Tb','Tb':'Tb','65':'Tb',65:'Tb','Thallium':'Tl','Tl':'Tl','81':'Tl',81:'Tl','Thorium':'Th','Th':'Th','90':'Th',90:'Th','Thulium':'Tm','Tm':'Tm','69':'Tm',69:'Tm','Tin':'Sn','Sn':'Sn','50':'Sn',50:'Sn','Titanium':'Ti','Ti':'Ti','22':'Ti',22:'Ti','Tungsten':'W','W':'W','74':'W',74:'W','Ununbium':'Uub','Uub':'Uub','112':'Uub',112:'Uub','Ununhexium':'Uuh','Uuh':'Uuh','116':'Uuh',116:'Uuh','Ununoctium':'Uuo','Uuo':'Uuo','118':'Uuo',118:'Uuo','Ununpentium':'Uup','Uup':'Uup','115':'Uup',115:'Uup','Ununquadium':'Uuq','Uuq':'Uuq','114':'Uuq',114:'Uuq','Ununseptium':'Uus','Uus':'Uus','117':'Uus',117:'Uus','Ununtrium':'Uut','Uut':'Uut','113':'Uut',113:'Uut','Ununium':'Uuu','Uuu':'Uuu','111':'Uuu',111:'Uuu','Uranium':'U','U':'U','92':'U',92:'U','Vanadium':'V','V':'V','23':'V',23:'V','Xenon':'Xe','Xe':'Xe','54':'Xe',54:'Xe','Ytterbium':'Yb','Yb':'Yb','70':'Yb',70:'Yb','Yttrium':'Y','Y':'Y','39':'Y',39:'Y','Zinc':'Zn','Zn':'Zn','30':'Zn',30:'Zn','Zirconium':'Zr','Zr':'Zr','40':'Zr',40:'Zr',}_atomic_mass={1:1.00794,2:4.002602,3:6.941,4:9.012182,5:10.811,6:12.0107,7:14.0067,8:15.9994,9:18.9984032,10:20.1797,11:22.98976928,12:24.3050,13:26.9815386,14:28.0855,15:30.973762,16:32.065,17:35.453,18:39.948,19:39.0983,20:40.078,21:44.955912,22:47.867,23:50.9415,24:51.9961,25:54.938045,26:55.845,27:58.933195,28:58.6934,29:63.546,30:65.409,31:69.723,32:72.64,33:74.92160,34:78.96,35:79.904,36:83.798,37:85.4678,38:87.62,39:88.90585,40:91.224,41:92.906,42:95.94,43:98.,44:101.07,45:102.905,46:106.42,47:107.8682,48:112.411,49:114.818,50:118.710,51:121.760,52:127.60,53:126.904,54:131.293,55:132.9054519,56:137.327,57:138.90547,58:140.116,59:140.90765,60:144.242,61:145.,62:150.36,63:151.964,64:157.25,65:158.92535,66:162.500,67:164.930,68:167.259,69:168.93421,70:173.04,71:174.967,72:178.49,73:180.94788,74:183.84,75:186.207,76:190.23,77:192.217,78:195.084,79:196.966569,80:200.59,81:204.3833,82:207.2,83:208.98040,84:210.,85:210.,86:220.,87:223.,88:226.,89:227.,91:231.03588,90:232.03806,93:237.,92:238.02891,95:243.,94:244.,96:247.,97:247.,98:251.,99:252.,100:257.,101:258.,102:259.,103:262.,104:261.,105:262.,106:266.,107:264.,108:277.,109:268.,110:271.,111:272.,112:285.,113:284.,114:289.,115:288.,116:292.,118:293.,}_radius_empirical={1:25,2:-1,3:145,4:105,5:85,6:70,7:65,8:60,9:50,10:-1,11:180,12:150,13:125,14:110,15:100,16:100,17:100,18:71,19:220,20:180,21:160,22:140,23:135,24:140,25:140,26:140,27:135,28:135,29:135,30:135,31:130,32:125,33:115,34:115,35:115,36:-1,37:235,38:200,39:180,40:155,41:145,42:145,43:135,44:130,45:135,46:140,47:160,48:155,49:155,50:145,51:145,52:140,53:140,54:-1,55:260,56:215,57:195,58:185,59:185,60:185,61:185,62:185,63:185,64:180,65:175,66:175,67:175,68:175,69:175,70:175,71:175,72:155,73:145,74:135,75:135,76:130,77:135,78:135,79:135,80:150,81:190,82:180,83:160,84:190,85:-1,86:-1,87:-1,88:215,89:195,90:180,91:180,92:175,93:175,94:175,95:175,96:-1,97:-1,98:-1,99:-1,100:-1,101:-1,102:-1,103:-1,104:-1,105:-1,106:-1,107:-1,108:-1,109:-1,110:-1,111:-1,112:-1,113:-1,114:-1,115:-1,116:-1,117:-1,118:-1,}_radius_calc={1:53,2:31,3:167,4:112,5:87,6:67,7:56,8:48,9:42,10:38,11:190,12:145,13:118,14:111,15:98,16:88,17:79,18:71,19:243,20:194,21:184,22:176,23:171,24:166,25:161,26:156,27:152,28:149,29:145,30:142,31:136,32:125,33:114,34:103,35:94,36:88,37:265,38:219,39:212,40:206,41:198,42:190,43:183,44:178,45:173,46:169,47:165,48:161,49:156,50:145,51:133,52:123,53:115,54:108,55:298,56:253,57:-1,58:-1,59:247,60:206,61:205,62:238,63:231,64:233,65:225,66:228,67:-1,68:226,69:222,70:222,71:217,72:208,73:200,74:193,75:188,76:185,77:180,78:177,79:174,80:171,81:156,82:154,83:143,84:135,85:-1,86:120,87:-1,88:-1,89:-1,90:-1,91:-1,92:-1,93:-1,94:-1,95:-1,96:-1,97:-1,98:-1,99:-1,100:-1,101:-1,102:-1,103:-1,104:-1,105:-1,106:-1,107:-1,108:-1,109:-1,110:-1,111:-1,112:-1,113:-1,114:-1,115:-1,116:-1,117:-1,118:-1,}_radius_vdw={1:120,2:140,3:182,4:153,5:192,6:170,7:155,8:152,9:147,10:154,11:227,12:173,13:184,14:210,15:180,16:180,17:175,18:188,19:275,20:231,21:211,22:-1,23:-1,24:-1,25:-1,26:-1,27:-1,28:163,29:140,30:139,31:187,32:211,33:185,34:190,35:185,36:202,37:303,38:249,39:-1,40:-1,41:-1,42:-1,43:-1,44:-1,45:-1,46:163,47:172,48:158,49:193,50:217,51:206,52:206,53:198,54:216,55:343,56:268,57:-1,58:-1,59:-1,60:-1,61:-1,62:-1,63:-1,64:-1,65:-1,66:-1,67:-1,68:-1,69:-1,70:-1,71:-1,72:-1,73:-1,74:-1,75:-1,76:-1,77:-1,78:175,79:166,80:155,81:196,82:202,83:207,84:197,85:202,86:220,87:348,88:283,89:-1,90:-1,91:-1,92:186,93:-1,94:-1,95:-1,96:-1,97:-1,98:-1,99:-1,100:-1,101:-1,102:-1,103:-1,104:-1,105:-1,106:-1,107:-1,108:-1,109:-1,110:-1,111:-1,112:-1,113:-1,114:-1,115:-1,116:-1,117:-1,118:-1,}# fmt: on
[docs]defZ(self,key):"""Atomic number based on general input Return the atomic number corresponding to the `key` lookup. Parameters ---------- key : array_like or str or int Uses value to lookup the atomic number in the `PeriodicTable` object. Returns ------- numpy.ndarray or int atomic number corresponding to `key`, if `key` is array_like, so will the returned value be. Examples -------- >>> 79 == PeriodicTable().Z_int('Au') True >>> 79 == PeriodicTable().Z('Au') True >>> 6 == PeriodicTable().Z('Carbon') True """key=np.asarray(key).ravel()get=self._Z_int.getiflen(key)==1:returnget(key[0],key[0])return_a.fromiteri(map(get,key,key))
Z_int=Z
[docs]defZ_label(self,key):"""Atomic label of the corresponding atom Return the atomic short name corresponding to the `key` lookup. Parameters ---------- key : array_like or str or int Uses value to lookup the atomic short name in the `PeriodicTable` object. Returns ------- numpy.ndarray or str atomic short name corresponding to `key`, if `key` is array_like, so will the returned value be. """ak=np.asarray(key).ravel()get=self._Z_short.getiflen(ak)==1:returnget(ak[0],"fa")return[get(ia,"fa")foriainak]
Z_short=Z_label
[docs]defatomic_mass(self,key):"""Atomic mass of the corresponding atom Return the atomic mass corresponding to the `key` lookup. Parameters ---------- key : array_like or str or int Uses value to lookup the atomic mass in the `PeriodicTable` object. Returns ------- numpy.ndarray or float atomic mass in atomic units corresponding to `key`, if `key` is array_like, so will the returned value be. """Z=self.Z_int(key)get=self._atomic_mass.getifisinstance(Z,(Integral,Real)):returnget(Z,0.0)return_a.fromiterd(map(get,Z,iter(float,1)))
[docs]defradius(self,key,method:Literal["calc","empirical","vdw"]="calc"):"""Atomic radius using different methods Return the atomic radius. Parameters ---------- key : array_like or str or int Uses value to lookup the atomic mass in the `PeriodicTable` object. method : {'calc', 'empirical', 'vdw'} There are 3 different radii stored: 1. ``calc``, the calculated atomic radius 2. ``empirical``, the empirically found values 3. ``vdw``, the van-der-Waals found values Returns ------- numpy.ndarray or float atomic radius in `Ang` """Z=self.Z_int(key)func=getattr(self,f"_radius_{method}").getifisinstance(Z,Integral):returnfunc(Z)/100return_a.fromiterd(map(func,Z))/100
# Create a local instance of the periodic table to# faster look up_ptbl=PeriodicTable()classAtomMeta(type):"""Meta class for key-lookup on the class."""def__getitem__(cls,key):"""Create a new atom object"""ifisinstance(key,Atom):# if the key already is an atomic object# return itreturnkeyelifisinstance(key,dict):# The key is a dictionary, hence# we can return the atom directlyreturncls(**key)elifisinstance(key,(list,tuple)):# The key is a list,# we need to create a list of atomsreturn[cls[k]forkinkey]# pylint: disable=E1136# Index Z basedreturncls(key)# Note the with_metaclass which is required for python3 support.# The designation of metaclass in python3 is actually:# class ...(..., metaclass=MetaClass)# This below construct handles both python2 and python3 cases@set_module("sisl")classAtom(_Dispatchs,dispatchs=[ClassDispatcher("to",type_dispatcher=None)],when_subclassing="keep",metaclass=AtomMeta,):"""Atomic information for a single atomic species An atomic object retaining information about a single atomic species. It describes the atomic number (integer), the mass of the atom, and holds a list of atomic centered orbitals. It also allows one to tag the atom to distinguish it from other atoms of the same species. Parameters ---------- Z : determine species for the atomic species. orbitals : list of Orbital or float, optional orbitals associated with this atom. See `Orbital` for details on how to define orbitals. Defaults to one orbital. mass : the atomic mass, defaults to the mass found in `PeriodicTable`. tag : arbitrary designation for user handling similar atoms with different settings (defaults to the label of the atom) Examples -------- >>> Carbon = Atom(6) >>> Carbon = Atom("C") >>> Carbon = Atom("Carbon") Add a tag to be able to distinguish it from other atoms >>> tagged_Carbon = Atom("Carbon", tag="siteA") Create deuterium >>> D = Atom("H", mass=2.014) Define an atom with 3 orbitals, each with a range of 2 Angstroem >>> C3 = Atom("C", orbitals=[2, 2, 2]) Define an atom outside of the periodic table (negative will yield an AtomGhost object) >>> ghost_C = Atom(-6) Define an unknown atom (basically anything can do) >>> unknown_atom = Atom(1000) Notes ----- One can define atoms outside of the periodic table. They will generally be handled in this order: * negative numbers will be converted into positive ones, and the returned object will be an `AtomGhost` * any other number (or name) not found in the periodic table will be returned in an AtomUnknown object The mass for atoms outside the periodic table will default to 1e40 amu. See Also -------- Orbital : define an orbital Atoms : an efficient collection of Atom objects """def__new__(cls,*args,**kwargs):"""Figure out which class to actually use"""# Handle the case where no arguments are passed (e.g. for serializing stuff)iflen(args)==0and"Z"notinkwargs:returnsuper().__new__(cls)# direct calliflen(args)>0:Z=args[0]if"Z"inkwargs:raiseValueError(f"{cls.__name__} got both Z as argument and keyword argument. Please only use one.")else:Z=NoneZ=kwargs.get("Z",Z)ifisinstance(Z,Atom):returnsuper().__new__(Z.__class__)try:# Try and convert to an integerZ=int(Z)exceptException:passifisinstance(Z,Integral)andnotissubclass(cls,AtomGhost)andZ<0:cls=AtomGhostelifZnotin_ptbl._Z_intandnotissubclass(cls,AtomUnknown):cls=AtomUnknownreturnsuper().__new__(cls)def__init__(self,Z:Union[str,int],orbitals=None,mass:Optional[float]=None,tag:Optional[str]=None,**kwargs,):# try and cast to integer, it might be cast differently later# but this is to try and see if we can easily get itmass_Z=Ztry:Z=int(Z)exceptException:passifisinstance(Z,Atom):self._Z=Z.Zmass_Z=self.Zelifisinstance(Z,Integral):self._Z=Zmass_Z=self.Zelse:self._Z=_ptbl.Z_int(Z)ifnotisinstance(self._Z,Integral):raiseValueError(f"{self.__class__.__name__} got an unparseable Z argument, needs to be an integer, got='{Z}'.")self._orbitals=Noneifisinstance(orbitals,(tuple,list,np.ndarray)):iflen(orbitals)==0:# This may be the same as only regarding `R` argumentpasselifisinstance(orbitals[0],Orbital):# all is goodself._orbitals=orbitalselifisinstance(orbitals[0],Real):# radius has been givenself._orbitals=[Orbital(R)forRinorbitals]elifisinstance(orbitals[0],str):# radius has been givenself._orbitals=[Orbital(-1,tag=tag)fortaginorbitals]elifisinstance(orbitals[0],tuple):# likely a radiuse + tagself._orbitals=[Orbital(R,tag=tag)forR,taginorbitals]elifall(orbisNonefororbinorbitals):orbitals=Noneelifisinstance(orbitals,Orbital):self._orbitals=[orbitals]elifisinstance(orbitals,Real):self._orbitals=[Orbital(orbitals)]ifself._orbitalsisNone:iforbitalsisnotNone:raiseValueError(f"{self.__class__.__name__}.__init__ got unparseable 'orbitals' argument: {orbitals}")if"R"inkwargs:# backwards compatibility (possibly remove this in the future)R=_a.asarrayd(kwargs["R"]).ravel()self._orbitals=[Orbital(r)forrinR]else:self._orbitals=[Orbital(-1.0)]ifmassisNone:self._mass=_ptbl.atomic_mass(mass_Z)else:self._mass=mass# self.tag will return self.symbol if not setself._tag=tagdef__hash__(self):returnhash((self.tag,self._mass,self._Z,*self._orbitals))@propertydefZ(self)->int:"""Atomic number"""returnself._Z@propertydeforbitals(self):"""List of orbitals"""returnself._orbitals@propertydefmass(self)->float:"""Atomic mass"""returnself._mass@propertydeftag(self)->str:"""Tag for atom"""ifself._tagisNone:returnself.symbolreturnself._tag@propertydefno(self)->int:"""Number of orbitals on this atom"""returnlen(self.orbitals)def__len__(self):"""Return number of orbitals in this atom"""returnself.no
[docs]defindex(self,orbital):"""Return the index of the orbital in the atom object"""ifnotisinstance(orbital,Orbital):orbital=self[orbital]fori,oinenumerate(self.orbitals):ifo==orbital:returniraiseKeyError("Could not find `orbital` in the list of orbitals.")
[docs]defradius(self,method:Literal["calc","empirical","vdw"]="calc"):"""Return the atomic radius of the atom (in Ang) See `PeriodicTable.radius` for details on the argument. """return_ptbl.radius(self.Z,method)
@propertydefsymbol(self):"""Return short atomic name (Au==79)."""return_ptbl.Z_short(self.Z)def__getitem__(self,key):"""The orbital corresponding to index `key`"""ifisinstance(key,slice):ol=key.indices(len(self))return[self.orbitals[o]foroinrange(*ol)]elifisinstance(key,Integral):returnself.orbitals[key]elifisinstance(key,str):orbs=[orbfororbinself.orbitalsifkeyinorb.name()]# In case none are found, None will be returnedifnotorbs:returnNonereturnorbsiflen(orbs)!=1elseorbs[0]return[self.orbitals[o]foroinkey]
[docs]defmaxR(self):"""Return the maximum range of orbitals."""mR=-1e10foroinself.orbitals:mR=max(mR,o.R)returnmR
def__iter__(self):"""Loop on all orbitals in this atom"""yield fromself.orbitals
[docs]defiter(self,group:bool=False):"""Loop on all orbitals in this atom Parameters ---------- group : bool, optional if two orbitals share the same radius one may be able to group two orbitals together Returns ------- Orbital current orbital, if `group` is ``True`` this is a list of orbitals, otherwise a single orbital is returned """ifgroup:i=0no=self.no-1whilei<=no:# Figure out how many share the same radial partj=i+1whilej<=no:ifnp.allclose(self.orbitals[i].R,self.orbitals[j].R):j+=1else:breakyieldself.orbitals[i:j]i=jreturnyield fromself.orbitals
def__str__(self):# Create orbitals outputorbs=",\n ".join([str(o)foroinself.orbitals])return(self.__class__.__name__+"{{{0}, Z: {1:d}, mass(au): {2:.5f}, maxR: {3:.5f},\n{4}\n}}".format(self.tag,self.Z,self.mass,self.maxR(),orbs))def__repr__(self):returnf"<{self.__module__}.{self.__class__.__name__}{self.tag}, Z={self.Z}, M={self.mass}, maxR={self.maxR()}, no={len(self.orbitals)}>"def__getattr__(self,attr):"""Pass attribute calls to the orbital classes and return lists/array Parameters ---------- attr : str """# First we create a list of values that the orbitals have# Some may have it, others may notvals=[None]*len(self.orbitals)found=Falseis_Integral=is_Real=is_callable=Trueforio,orbinenumerate(self.orbitals):try:vals[io]=getattr(orb,attr)found=Trueis_callable&=callable(vals[io])is_Integral&=isinstance(vals[io],Integral)is_Real&=isinstance(vals[io],Real)exceptAttributeError:passiffound==0:# we never got any values, reraise the AttributeErrorraiseAttributeError(f"'{self.__class__.__name__}.orbitals' objects has no attribute '{attr}'")# Now parse the data, currently we'll only allow Integral, Real, Complexifis_Integral:forioinrange(len(vals)):ifvals[io]isNone:vals[io]=0return_a.arrayi(vals)elifis_Real:forioinrange(len(vals)):ifvals[io]isNone:vals[io]=0.0return_a.arrayd(vals)elifis_callable:def_ret_none(*args,**kwargs):returnNoneforioinrange(len(vals)):ifvals[io]isNone:vals[io]=_ret_none# Now subclass the content and return values per methodclassArrayCall:def__init__(self,methods):self.methods=methodsdef__call__(self,*args,**kwargs):return[m(*args,**kwargs)forminself.methods]returnArrayCall(vals)# We don't know how to handle this, simply return...returnvals
[docs]@deprecation("toSphere is deprecated, use shape.to.Sphere(...) instead.","0.15","0.16")deftoSphere(self,center=None):"""Return a sphere with the maximum orbital radius equal Returns ------- ~sisl.shape.Sphere a sphere with radius equal to the maximum radius of the orbitals """returnself.to.Sphere(center=center)
[docs]defequal(self,other,R:bool=True,psi:bool=False):"""True if `other` is the same as this atomic species Parameters ---------- other : Atom the other object to check againts R : bool, optional if True the equality check also checks the orbital radius, else they are not compared psi : bool, optional if True, also check the wave-function component of the orbitals, see `Orbital.psi` """ifnotisinstance(other,Atom):returnFalsesame=self.Z==other.Zsame&=self.no==other.noifsameandR:same&=all([self.orbitals[i].equal(other.orbitals[i],psi=psi)foriinrange(self.no)])same&=np.isclose(self.mass,other.mass)same&=self.tag==other.tagreturnsame
# Check whether they are equaldef__eq__(self,b):"""Return true if the saved quantities are the same"""returnself.equal(b)def__ne__(self,b):returnnot(self==b)# Create pickling routinesdef__getstate__(self):"""Return the state of this object"""return{"Z":self.Z,"orbitals":self.orbitals,"mass":self.mass,"tag":self.tag,}def__setstate__(self,d):"""Re-create the state of this object"""self.__init__(d["Z"],d["orbitals"],d["mass"],d["tag"])@set_module("sisl")classAtomUnknown(Atom):def__init__(self,Z,*args,**kwargs):"""Instantiate with overridden tag"""iflen(args)<3and"tag"notinkwargs:kwargs["tag"]="unknown"iflen(args)<2and"mass"notinkwargs:kwargs["mass"]=1e40super().__init__(Z,*args,**kwargs)@set_module("sisl")classAtomGhost(AtomUnknown):def__init__(self,Z,*args,**kwargs):"""Instantiate with overridden tag and taking the absolute value of Z"""try:# here we also need to do the conversion as we want# to remove the negative signZ=abs(int(Z))exceptExecption:passiflen(args)<3and"tag"notinkwargs:kwargs["tag"]="ghost"super().__init__(Z,*args,**kwargs)classAtomToDispatch(AbstractDispatch):"""Base dispatcher from class passing from an Atom class"""to_dispatch=Atom.toclassToSphereDispatch(AtomToDispatch):defdispatch(self,*args,center=None,**kwargs):returnSphere(self._get_object().maxR(),center)to_dispatch.register("Sphere",ToSphereDispatch)@set_module("sisl")classAtoms:"""Efficient collection of `Atom` objects A container object for `Atom` objects in a specific order. No two `Atom` objects will be duplicated and indices will be used to determine which `Atom` any indexable atom corresponds to. This is convenient when having geometries with millions of atoms because it will not duplicate the `Atom` object, only a list index. Parameters ---------- atoms : atoms to be contained in this list of atoms If a str, or a single `Atom` it will be the only atom in the resulting class repeated `na` times. If a list, it will create all unique atoms and retain these, each item in the list may a single argument passed to the `Atom` or a dictionary that is passed to `Atom`, see examples. na : total number of atoms, if ``len(atoms)`` is smaller than `na` it will be repeated to match `na`. Examples -------- Creating an atoms object consisting of 5 atoms, all the same. >>> atoms = Atoms("H", na=5) Creating a set of 4 atoms, 2 Hydrogen, 2 Helium, in an alternate ordere >>> Z = [1, 2, 1, 2] >>> atoms = Atoms(Z) >>> atoms = Atoms([1, 2], na=4) # equivalent Creating individual atoms using dictionary entries, two Hydrogen atoms, one with a tag H_ghost. >>> Atoms([dict(Z=1, tag="H_ghost"), 1]) """# Using the slots should make this class slightly faster.__slots__=("_atom","_species","_firsto")def__init__(self,atoms:AtomsLike="H",na:Optional[int]=None):# Default value of the atom objectifatomsisNone:atoms=Atom("H")# Correct the atoms input to Atomifisinstance(atoms,Atom):uatoms=[atoms]species=[0]elifisinstance(atoms,Atoms):# Ensure we make a copy to not operate# on the same data.catoms=atoms.copy()uatoms=catoms.atom[:]species=catoms.species[:]elifisinstance(atoms,(str,Integral)):uatoms=[Atom(atoms)]species=[0]elifisinstance(atoms,dict):uatoms=[Atom(**atoms)]species=[0]elifisinstance(atoms,Iterable):# TODO this is very inefficient for large MD filesuatoms=[]species=[]forainatoms:ifisinstance(a,dict):a=Atom(**a)elifnotisinstance(a,Atom):a=Atom(a)try:s=uatoms.index(a)exceptException:s=len(uatoms)uatoms.append(a)species.append(s)else:raiseValueError(f"atoms keyword type is not acceptable {type(atoms)}")# Default for number of atomsifnaisNone:na=len(species)# Create atom and species objectsself._atom=list(uatoms)self._species=array_fill_repeat(species,na,cls=np.int16)self._update_orbitals()def_update_orbitals(self):"""Internal routine for updating the `firsto` attribute"""# Get number of orbitals per specieuorbs=_a.arrayi([a.noforainself.atom])self._firsto=np.insert(_a.cumsumi(uorbs[self.species]),0,0)@propertydefatom(self):"""List of unique atoms in this group of atoms"""returnself._atom@property@deprecation("nspecie is deprecated, use nspecies instead.","0.15","0.16")defnspecie(self):"""Number of different species"""returnlen(self._atom)@propertydefnspecies(self):"""Number of different species"""returnlen(self._atom)@propertydefspecies(self):"""List of atomic species"""returnself._species@property@deprecation("specie is deprecated, use species instead.","0.15","0.16")defspecie(self):"""List of atomic species"""returnself._species@propertydefno(self)->int:"""Total number of orbitals in this list of atoms"""uorbs=_a.arrayi([a.noforainself.atom])returnint(uorbs[self.species].sum())@propertydeforbitals(self):"""Array of orbitals of the contained objects"""returnnp.diff(self.firsto)@propertydeffirsto(self):"""First orbital of the corresponding atom in the consecutive list of orbitals"""returnself._firsto@propertydeflasto(self):"""Last orbital of the corresponding atom in the consecutive list of orbitals"""returnself._firsto[1:]-1@propertydefq0(self):"""Initial charge per atom"""q0=_a.arrayd([a.q0.sum()forainself.atom])returnq0[self.species]
[docs]deforbital(self,io):"""Return an array of orbital of the contained objects"""io=_a.asarrayi(io)ndim=io.ndimio=io.ravel()%self.noa=list_index_le(io,self.lasto)io=io-self.firsto[a]a=self.species[a]# Now extract the list of orbitalsifndim==0:returnself.atom[a[0]].orbitals[io[0]]return[self.atom[ia].orbitals[o]foria,oinzip(a,io)]
[docs]defmaxR(self,all:bool=False):"""The maximum radius of the atoms Parameters ---------- all : bool determine the returned maximum radii. If `True` is passed an array of all atoms maximum radii is returned (array). Else, if `False` the maximum of all atoms maximum radii is returned (scalar). """ifall:maxR=_a.arrayd([a.maxR()forainself.atom])returnmaxR[self.species]returnnp.amax([a.maxR()forainself.atom])
@propertydefmass(self):"""Array of masses of the contained objects"""umass=_a.arrayd([a.massforainself.atom])returnumass[self.species]@propertydefZ(self):"""Array of atomic numbers"""uZ=_a.arrayi([a.Zforainself.atom])returnuZ[self.species]
[docs]defindex(self,atom):"""Return the indices of the atom object"""return(self._species==self.species_index(atom)).nonzero()[0]
[docs]defspecies_index(self,atom):"""Return the species index of the atom object"""ifnotisinstance(atom,Atom):atom=self[atom]fors,ainenumerate(self.atom):ifa==atom:returnsraiseKeyError("Could not find `atom` in the list of atoms.")
specie_index=deprecation("specie_index is deprecated, use species_index instead.","0.15","0.16")(species_index)
[docs]defgroup_atom_data(self,data,axis=0):r"""Group data for each atom based on number of orbitals This is useful for grouping data that is orbitally resolved. This will return a list of length ``len(self)`` and with each item having the sub-slice of the data corresponding to the orbitals on the given atom. Examples -------- >>> atoms = Atoms([Atom(4, [0.1, 0.2]), Atom(6, [0.2, 0.3, 0.5])]) >>> orb_data = np.arange(10).reshape(2, 5) >>> atoms.group_data(orb_data, axis=1) [ [[0, 1], [2, 3]], [[4, 5, 6], [7, 8, 9]] ] Parameters ---------- data : numpy.ndarray data to be grouped axis : int, optional along which axis one should split the data """returnnp.split(data,self.lasto[:-1]+1,axis=axis)
[docs]@deprecate_argument("in_place","inplace","argument in_place has been deprecated in favor of inplace, please update your code.","0.15","0.16",)defreorder(self,inplace:bool=False):"""Reorders the atoms and species index so that they are ascending (starting with a species that exists) Parameters ---------- inplace : whether the re-order is done *in-place* """# Contains the minimum atomic index for a given speciesmin=_a.emptyi(len(self.atom))smin.fill(len(self))forainrange(len(self.atom)):lst=(self.species==a).nonzero()[0]iflen(lst)>0:smin[a]=lst.min()ifinplace:atoms=selfelse:atoms=self.copy()# Now swap indices into correct place# This will give the indices of the species# in the ascending orderisort=np.argsort(smin)ifnp.allclose(np.diff(isort),0):returnatomsatoms._atom[:]=[atoms._atom[i]foriinisort]atoms._species[:]=isort[atoms._species]atoms._update_orbitals()returnatoms
[docs]defformula(self,system="Hill"):"""Return the chemical formula for the species in this object Parameters ---------- system : {"Hill"}, optional which notation system to use Is not case-sensitive """# loop different speciesc=Counter()foratom,indicesinself.iter(species=True):iflen(indices)>0:c[atom.symbol]+=len(indices)# now we have all elements, and the counts of themsysteml=system.lower()ifsysteml=="hill":# sort lexographicallysymbols=sorted(c.keys())defparse(symbol_c):symbol,c=symbol_cifc==1:returnsymbolreturnf"{symbol}{c}"return"".join(map(parse,sorted(c.items())))raiseValueError(f"{self.__class__.__name__}.formula got unrecognized argument 'system' {system}")
[docs]@deprecate_argument("in_place","inplace","argument in_place has been deprecated in favor of inplace, please update your code.","0.15","0.16",)defreduce(self,inplace:bool=False):"""Returns a new `Atoms` object by removing non-used atoms"""ifinplace:atoms=selfelse:atoms=self.copy()atom=atoms._atomspecies=atoms._speciesrem=[]foriinrange(len(self.atom)):ifnp.all(species!=i):rem.append(i)# Remove the atomsforiinrem[::-1]:atom.pop(i)species=np.where(species>i,species-1,species)atoms._atom=atomatoms._species=speciesatoms._update_orbitals()returnatoms
[docs]defswap_atom(self,a,b):"""Swap species index positions"""speciesa=self.species_index(a)speciesb=self.species_index(b)idx_a=(self._species==speciesa).nonzero()[0]idx_b=(self._species==speciesb).nonzero()[0]atoms=self.copy()atoms._atom[speciesa],atoms._atom[speciesb]=(atoms._atom[speciesb],atoms._atom[speciesa],)atoms._species[idx_a]=speciesbatoms._species[idx_b]=speciesaatoms._update_orbitals()returnatoms
[docs]defreverse(self,atoms=None):"""Returns a reversed geometry Also enables reversing a subset of the atoms. """copy=self.copy()ifatomsisNone:copy._species=self._species[::-1]else:copy._species[atoms]=self._species[atoms[::-1]]copy._update_orbitals()returncopy
def__str__(self):"""Return the `Atoms` in str"""s=f"{self.__class__.__name__}{{species: {len(self._atom)},\n"fora,idxinself.iter(True):s+=" {1}: {0},\n".format(len(idx),str(a).replace("\n","\n "))returnf"{s}}}"def__repr__(self):returnf"<{self.__module__}.{self.__class__.__name__} nspecies={len(self._atom)}, na={len(self)}, no={self.no}>"def__len__(self):"""Return number of atoms in the object"""returnlen(self._species)
[docs]defiter(self,species=False):"""Loop on all atoms This iterator may be used in two contexts: 1. `species` is ``False``, this is the slowest method and will yield the `Atom` per contained atom. 2. `species` is ``True``, which yields a tuple of `(Atom, list)` where ``list`` contains all indices of atoms that has the `Atom` species. This is much faster than the first option. Parameters ---------- species : bool, optional If ``True`` loops only on different species and yields a tuple of (Atom, list) Else yields the atom for the equivalent index. """ifspecies:fors,atominenumerate(self._atom):yieldatom,(self.species==s).nonzero()[0]else:forsinself.species:yieldself._atom[s]
def__iter__(self):"""Loop on all atoms with the same species in order of atoms"""yield fromself.iter()def__contains__(self,key):"""Determine whether the `key` is in the unique atoms list"""returnkeyinself.atomdef__getitem__(self,key):"""Return an `Atom` object corresponding to the key(s)"""ifisinstance(key,slice):sl=key.indices(len(self))return[self.atom[self._species[s]]forsinrange(sl[0],sl[1],sl[2])]elifisinstance(key,Integral):returnself.atom[self._species[key]]elifisinstance(key,str):foratinself.atom:ifat.tag==key:returnatreturnNonekey=np.asarray(key)ifkey.ndim==0:returnself.atom[self._species[key]]return[self.atom[i]foriinself._species[key]]def__setitem__(self,key,value):"""Overwrite an `Atom` object corresponding to the key(s)"""# If key is a string, we replace the atom that matches 'key'ifisinstance(key,str):self.replace_atom(self[key],value)return# Convert to arrayifisinstance(key,slice):sl=key.indices(len(self))key=_a.arangei(sl[0],sl[1],sl[2])else:key=_a.asarrayi(key).ravel()iflen(key)==0:ifvaluenotinself:self._atom.append(value)return# Create new atoms object to iterateother=Atoms(value,na=len(key))# Append the new Atom objectsforatom,s_iinother.iter(True):ifatomnotinself:self._atom.append(atom)self._species[key[s_i]]=self.species_index(atom)self._update_orbitals()
[docs]defreplace(self,index,atom):"""Replace all atomic indices `index` with the atom `atom` (in-place) This is the preferred way of replacing atoms in geometries. Parameters ---------- index : list of int or Atom the indices of the atoms that should be replaced by the new atom. If an `Atom` is passed, this routine defers its call to `replace_atom`. atom : Atom the replacement atom. """ifisinstance(index,Atom):self.replace_atom(index,atom)returnifnotisinstance(atom,Atom):raiseTypeError(f"{self.__class__.__name__}.replace requires input arguments to ""be of the type Atom")index=_a.asarrayi(index).ravel()# Be sure to add the atomifatomnotinself.atom:self._atom.append(atom)# Get species index of the atomspecies=self.species_index(atom)# Loop unique species and check that we have the correct number of orbitalsforiusinnp.unique(self._species[index]):a=self._atom[ius]ifa.no!=atom.no:a1=" "+str(a).replace("\n","\n ")a2=" "+str(atom).replace("\n","\n ")info(f"Substituting atom\n{a1}\n->\n{a2}\nwith a different number of orbitals!")self._species[index]=species# Update orbital counts...self._update_orbitals()
[docs]defreplace_atom(self,atom_from:Atom,atom_to:Atom):"""Replace all atoms equivalent to `atom_from` with `atom_to` (in-place) I.e. this is the preferred way of adapting all atoms of a specific type with another one. If the two atoms does not have the same number of orbitals a warning will be raised. Parameters ---------- atom_from : Atom the atom that should be replaced, if not found in the current list of atoms, nothing will happen. atom_to : Atom the replacement atom. Raises ------ KeyError if `atom_from` does not exist in the list of atoms UserWarning if the atoms does not have the same number of orbitals. """ifnotisinstance(atom_from,Atom):raiseTypeError(f"{self.__class__.__name__}.replace_atom requires input arguments to ""be of the class Atom")ifnotisinstance(atom_to,Atom):raiseTypeError(f"{self.__class__.__name__}.replace_atom requires input arguments to ""be of the class Atom")# Get index of `atom_from`idx_from=self.species_index(atom_from)try:idx_to=self.species_index(atom_to)ifidx_from==idx_to:raiseKeyError("")# Decrement indices of the atoms that are# changed to one already thereself._species[self.species==idx_from]=idx_toself._species[self.species>idx_from]-=1# Now delete the old index, we replace, so we *have* to remove itself._atom.pop(idx_from)exceptKeyError:# The atom_to is not in the list# Simply changeself._atom[idx_from]=atom_toifatom_from.no!=atom_to.no:a1=" "+str(atom_from).replace("\n","\n ")a2=" "+str(atom_to).replace("\n","\n ")info(f"Replacing atom\n{a1}\n->\n{a2}\nwith a different number of orbitals!")# Update orbital counts...self._update_orbitals()
[docs]defhassame(self,other,R=True):"""True if the contained atoms are the same in the two lists Notes ----- This does not necessarily mean that the order, nor the number of atoms are the same. Parameters ---------- other : Atoms the list of atoms to check against R : bool, optional if True also checks that the orbital radius are the same See Also -------- equal : explicit check of the indices *and* the contained atoms """iflen(self.atom)!=len(other.atom):returnFalseforAinself.atom:is_in=FalseforBinother.atom:ifA.equal(B,R):is_in=Truebreakifnotis_in:returnFalsereturnTrue
[docs]defequal(self,other,R=True):"""True if the contained atoms are the same in the two lists (also checks indices) Parameters ---------- other : Atoms the list of atoms to check against R : bool, optional if True also checks that the orbital radius are the same See Also -------- hassame : only check whether the two atoms are contained in both """iflen(self.atom)>len(other.atom):foriA,Ainenumerate(self.atom):is_in=-1foriB,Binenumerate(other.atom):ifA.equal(B,R):is_in=iBbreakifis_in==-1:returnFalse# We should check that they also have the same indicesifnotnp.all(np.nonzero(self.species==iA)[0]==np.nonzero(other.species==is_in)[0]):returnFalseelse:foriB,Binenumerate(other.atom):is_in=-1foriA,Ainenumerate(self.atom):ifB.equal(A,R):is_in=iAbreakifis_in==-1:returnFalse# We should check that they also have the same indicesifnotnp.all(np.nonzero(other.species==iB)[0]==np.nonzero(self.species==is_in)[0]):returnFalsereturnTrue
def__eq__(self,b):"""Returns true if the contained atoms are the same"""returnself.equal(b)# Create pickling routinesdef__getstate__(self):"""Return the state of this object"""return{"atom":self.atom,"species":self.species}def__setstate__(self,d):"""Re-create the state of this object"""self.__init__()self._atom=d["atom"]self._species=d["species"]