From 196931010f9d0acf75390674f3bee67d02ca2e4f Mon Sep 17 00:00:00 2001 From: Pol Date: Sat, 8 Aug 2020 11:06:24 +0200 Subject: [PATCH 1/6] enh: reading orbitals params from string as an static method --- sisl/orbital.py | 124 +++++++++++++++++++++++++++++------------------- 1 file changed, 75 insertions(+), 49 deletions(-) diff --git a/sisl/orbital.py b/sisl/orbital.py index 8bdbf3b786..5afb923943 100644 --- a/sisl/orbital.py +++ b/sisl/orbital.py @@ -773,55 +773,7 @@ def __init__(self, *args, **kwargs): # String specification of the atomic orbital s = args.pop(0) - _n = {'s': 1, 'p': 2, 'd': 3, 'f': 4, 'g': 5} - _l = {'s': 0, 'p': 1, 'd': 2, 'f': 3, 'g': 4} - _m = {'s': 0, - 'pz': 0, 'px': 1, 'py': -1, - 'dxy': -2, 'dyz': -1, 'dz2': 0, 'dxz': 1, 'dx2-y2': 2, - 'fy(3x2-y2)': -3, 'fxyz': -2, 'fz2y': -1, 'fz3': 0, - 'fz2x': 1, 'fz(x2-y2)': 2, 'fx(x2-3y2)': 3, - 'gxy(x2-y2)': -4, 'gzy(3x2-y2)': -3, 'gz2xy': -2, 'gz3y': -1, 'gz4': 0, - 'gz3x': 1, 'gz2(x2-y2)': 2, 'gzx(x2-3y2)': 3, 'gx4+y4': 4, - } - - # First remove a P for polarization - P = 'P' in s - s = s.replace('P', '') - - # Try and figure out the input - # 2s => n=2, l=0, m=0, z=1, P=False - # 2sZ2P => n=2, l=0, m=0, z=2, P=True - # 2pxZ2P => n=2, l=0, m=0, z=2, P=True - # By default a non-"n" specification takes the lowest value allowed - # s => n=1 - # p => n=2 - # ... - try: - n = int(s[0]) - # Remove n specification - s = s[1:] - except: - n = _n.get(s[0]) - - # Get l - l = _l.get(s[0]) - - # Get number of zeta shell - iZ = s.find('Z') - if iZ >= 0: - # Currently we know that we are limited to 9 zeta shells. - # However, for now we assume this is enough (could easily - # be extended by a reg-exp) - try: - Z = int(s[iZ+1]) - # Remove Z + int - s = s[:iZ] + s[iZ+2:] - except: - Z = 1 - s = s[:iZ] + s[iZ+1:] - - # We should be left with m specification - m = _m.get(s) + n, l, m, Z, P = self.orb_name_to_params(s, force_n=True, n=n, l=l, m=m, Z=Z, P=P) # Now we should figure out how the spherical orbital # has been passed. @@ -898,6 +850,80 @@ def __init__(self, *args, **kwargs): self.orb = Orbital(self.R) self.R = self.orb.R + + @staticmethod + def orb_name_to_params(orb_name, force_n=False, n=None, l=None, m=None, Z=None, P=None): + """ + Gets the quantum numbers, Z shell and polarization corresponding to an orbital. + + Parameters + ------------ + orb_name: str + the name for which we want to retreive the parameters + force_n: bool, optional + whether n should be forced. + + This means that if the first character is not a number, it will be attempted to read from + {'s': 1, 'p': 2, 'd': 3, 'f': 4, 'g': 5}. + + Returns + -------- + n, l, m, Z, P + """ + + _n = {'s': 1, 'p': 2, 'd': 3, 'f': 4, 'g': 5} + _l = {'s': 0, 'p': 1, 'd': 2, 'f': 3, 'g': 4} + _m = {'s': 0, + 'pz': 0, 'px': 1, 'py': -1, + 'dxy': -2, 'dyz': -1, 'dz2': 0, 'dxz': 1, 'dx2-y2': 2, + 'fy(3x2-y2)': -3, 'fxyz': -2, 'fz2y': -1, 'fz3': 0, + 'fz2x': 1, 'fz(x2-y2)': 2, 'fx(x2-3y2)': 3, + 'gxy(x2-y2)': -4, 'gzy(3x2-y2)': -3, 'gz2xy': -2, 'gz3y': -1, 'gz4': 0, + 'gz3x': 1, 'gz2(x2-y2)': 2, 'gzx(x2-3y2)': 3, 'gx4+y4': 4, + } + + # First remove a P for polarization + P = 'P' in orb_name + orb_name = orb_name.replace('P', '') + + # Try and figure out the input + # 2s => n=2, l=0, m=0, z=1, P=False + # 2sZ2P => n=2, l=0, m=0, z=2, P=True + # 2pxZ2P => n=2, l=0, m=0, z=2, P=True + # By default a non-"n" specification takes the lowest value allowed + # s => n=1 + # p => n=2 + # ... + try: + n = int(orb_name[0]) + # Remove n specification + orb_name = orb_name[1:] + except: + if force_n: + n = _n.get(orb_name[0], n) + + if orb_name: + # Get l + l = _l.get(orb_name[0], l) + + # Get number of zeta shell + iZ = orb_name.find('Z') + if iZ >= 0: + # Currently we know that we are limited to 9 zeta shells. + # However, for now we assume this is enough (could easily + # be extended by a reg-exp) + try: + Z = int(orb_name[iZ+1]) + # Remove Z + int + orb_name = orb_name[:iZ] + orb_name[iZ+2:] + except: + Z = 1 + orb_name = orb_name[:iZ] + orb_name[iZ+1:] + + # We should be left with m specification + m = _m.get(orb_name, m) + + return n, l, m, Z, P @property def f(self): From 67a2d05ab51a9465f8445d2e2076fee1a3e3312e Mon Sep 17 00:00:00 2001 From: Pol Date: Sat, 8 Aug 2020 11:40:35 +0200 Subject: [PATCH 2/6] maint: make sure tests pass for orbital --- sisl/orbital.py | 45 ++++++++++++++------------------------ sisl/tests/test_orbital.py | 3 +++ 2 files changed, 19 insertions(+), 29 deletions(-) diff --git a/sisl/orbital.py b/sisl/orbital.py index 5afb923943..2efb42eb84 100644 --- a/sisl/orbital.py +++ b/sisl/orbital.py @@ -773,20 +773,7 @@ def __init__(self, *args, **kwargs): # String specification of the atomic orbital s = args.pop(0) - n, l, m, Z, P = self.orb_name_to_params(s, force_n=True, n=n, l=l, m=m, Z=Z, P=P) - - # Now we should figure out how the spherical orbital - # has been passed. - # There are two options: - # 1. The radial function is passed as two arrays: r, f - # 2. The SphericalOrbital-class is passed which already contains - # the relevant information. - # Figure out if it is a sphericalorbital - if len(args) > 0: - if isinstance(args[0], SphericalOrbital): - self.orb = args.pop(0) - else: - self.orb = SphericalOrbital(l, args.pop(0)) + n, l, m, Z, P = self.orb_name_to_params(s, force_n=True, n=n, l=l, m=m, Z=Z, P=kwargs.get('P')) else: # Arguments *have* to be @@ -810,13 +797,6 @@ def __init__(self, *args, **kwargs): if isinstance(args[0], bool): P = args.pop(0) - # Figure out if it is a sphericalorbital - if len(args) > 0: - if isinstance(args[0], SphericalOrbital): - self.orb = args.pop(0) - else: - self.orb = SphericalOrbital(l, args.pop(0)) - # Still if n is None, we assign the default (lowest) quantum number if n is None: n = l + 1 @@ -833,13 +813,19 @@ def __init__(self, *args, **kwargs): if abs(self.m) > self.l: raise ValueError(self.__class__.__name__ + ' requires |m| <= l.') - # Retrieve user-passed spherical orbital - s = kwargs.get('spherical', None) - + # Now we should figure out how the spherical orbital + # has been passed. + # There are two options: + # 1. The radial function is passed as two arrays: r, f + # 2. The SphericalOrbital-class is passed which already contains + # the relevant information. + # Figure out if it is a sphericalorbital + s = kwargs.get('spherical') if s is None: - # Expect the orbital to already be set - pass - elif isinstance(s, Orbital): + if len(args) > 0: + s = args.pop(0) + + if isinstance(s, SphericalOrbital) or s is None: self.orb = s else: self.orb = SphericalOrbital(l, s) @@ -882,8 +868,9 @@ def orb_name_to_params(orb_name, force_n=False, n=None, l=None, m=None, Z=None, 'gz3x': 1, 'gz2(x2-y2)': 2, 'gzx(x2-3y2)': 3, 'gx4+y4': 4, } - # First remove a P for polarization - P = 'P' in orb_name + if P is None: + # First remove a P for polarization + P = 'P' in orb_name orb_name = orb_name.replace('P', '') # Try and figure out the input diff --git a/sisl/tests/test_orbital.py b/sisl/tests/test_orbital.py index 9a79b5d894..63531c884e 100644 --- a/sisl/tests/test_orbital.py +++ b/sisl/tests/test_orbital.py @@ -287,6 +287,9 @@ def test_init1(self): a.append(AtomicOrbital('pzP', f)) a.append(AtomicOrbital('pzP', rf)) a.append(AtomicOrbital('2pzP', rf)) + a.append(AtomicOrbital('2', rf, l=1, m=0, Z=1, P=True)) + a.append(AtomicOrbital('2p', m=0, Z=1, P=True, spherical=f)) + a.append(AtomicOrbital('2p', "dummy", m=0, Z=1, P=True, spherical=rf)) for i in range(len(a) - 1): for j in range(i+1, len(a)): assert a[i] == a[j] and a[i].equal(a[j], psi=True, radial=True) From 46639d4ac43e154bfd5326dcb2b8a079916b2a6d Mon Sep 17 00:00:00 2001 From: Pol Date: Sat, 8 Aug 2020 11:43:41 +0200 Subject: [PATCH 3/6] maint: some whitespaces --- sisl/orbital.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/sisl/orbital.py b/sisl/orbital.py index 2efb42eb84..811fee9e28 100644 --- a/sisl/orbital.py +++ b/sisl/orbital.py @@ -820,7 +820,7 @@ def __init__(self, *args, **kwargs): # 2. The SphericalOrbital-class is passed which already contains # the relevant information. # Figure out if it is a sphericalorbital - s = kwargs.get('spherical') + s = kwargs.get('spherical') if s is None: if len(args) > 0: s = args.pop(0) @@ -836,7 +836,7 @@ def __init__(self, *args, **kwargs): self.orb = Orbital(self.R) self.R = self.orb.R - + @staticmethod def orb_name_to_params(orb_name, force_n=False, n=None, l=None, m=None, Z=None, P=None): """ @@ -870,7 +870,7 @@ def orb_name_to_params(orb_name, force_n=False, n=None, l=None, m=None, Z=None, if P is None: # First remove a P for polarization - P = 'P' in orb_name + P = 'P' in orb_name orb_name = orb_name.replace('P', '') # Try and figure out the input From c3988022aa83aa5bad646e83e59cfbdd9d3897e3 Mon Sep 17 00:00:00 2001 From: Pol Date: Sat, 8 Aug 2020 12:08:59 +0200 Subject: [PATCH 4/6] maint: remove force_n from orbital name to params --- sisl/orbital.py | 17 +++-------------- 1 file changed, 3 insertions(+), 14 deletions(-) diff --git a/sisl/orbital.py b/sisl/orbital.py index 811fee9e28..9de1cf8b3d 100644 --- a/sisl/orbital.py +++ b/sisl/orbital.py @@ -773,7 +773,7 @@ def __init__(self, *args, **kwargs): # String specification of the atomic orbital s = args.pop(0) - n, l, m, Z, P = self.orb_name_to_params(s, force_n=True, n=n, l=l, m=m, Z=Z, P=kwargs.get('P')) + n, l, m, Z, P = self.orb_name_to_params(s, n=n, l=l, m=m, Z=Z, P=kwargs.get('P')) else: # Arguments *have* to be @@ -838,7 +838,7 @@ def __init__(self, *args, **kwargs): self.R = self.orb.R @staticmethod - def orb_name_to_params(orb_name, force_n=False, n=None, l=None, m=None, Z=None, P=None): + def orb_name_to_params(orb_name, n=None, l=None, m=None, Z=None, P=None): """ Gets the quantum numbers, Z shell and polarization corresponding to an orbital. @@ -846,18 +846,12 @@ def orb_name_to_params(orb_name, force_n=False, n=None, l=None, m=None, Z=None, ------------ orb_name: str the name for which we want to retreive the parameters - force_n: bool, optional - whether n should be forced. - - This means that if the first character is not a number, it will be attempted to read from - {'s': 1, 'p': 2, 'd': 3, 'f': 4, 'g': 5}. Returns -------- n, l, m, Z, P """ - _n = {'s': 1, 'p': 2, 'd': 3, 'f': 4, 'g': 5} _l = {'s': 0, 'p': 1, 'd': 2, 'f': 3, 'g': 4} _m = {'s': 0, 'pz': 0, 'px': 1, 'py': -1, @@ -877,17 +871,12 @@ def orb_name_to_params(orb_name, force_n=False, n=None, l=None, m=None, Z=None, # 2s => n=2, l=0, m=0, z=1, P=False # 2sZ2P => n=2, l=0, m=0, z=2, P=True # 2pxZ2P => n=2, l=0, m=0, z=2, P=True - # By default a non-"n" specification takes the lowest value allowed - # s => n=1 - # p => n=2 - # ... try: n = int(orb_name[0]) # Remove n specification orb_name = orb_name[1:] except: - if force_n: - n = _n.get(orb_name[0], n) + pass if orb_name: # Get l From dee95f7c52e4728197984d89a694ddf50b755846 Mon Sep 17 00:00:00 2001 From: Pol Date: Sat, 8 Aug 2020 12:11:29 +0200 Subject: [PATCH 5/6] maint: checking if spherical orbital is None in the wrong place --- sisl/orbital.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/sisl/orbital.py b/sisl/orbital.py index 9de1cf8b3d..36c93cd648 100644 --- a/sisl/orbital.py +++ b/sisl/orbital.py @@ -825,9 +825,9 @@ def __init__(self, *args, **kwargs): if len(args) > 0: s = args.pop(0) - if isinstance(s, SphericalOrbital) or s is None: + if isinstance(s, SphericalOrbital): self.orb = s - else: + elif s is not None: self.orb = SphericalOrbital(l, s) if self.orb is None: From fef390479d28f5c6492d8791bd8d3450613f6a15 Mon Sep 17 00:00:00 2001 From: Pol Date: Sat, 8 Aug 2020 12:15:44 +0200 Subject: [PATCH 6/6] maint: completed docs for orb_name_to_params --- sisl/orbital.py | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/sisl/orbital.py b/sisl/orbital.py index 36c93cd648..672d786cf1 100644 --- a/sisl/orbital.py +++ b/sisl/orbital.py @@ -846,6 +846,17 @@ def orb_name_to_params(orb_name, n=None, l=None, m=None, Z=None, P=None): ------------ orb_name: str the name for which we want to retreive the parameters + n: int, optional + the default value for n. + l: int, optional + the default value for n. + m: int, optional + the default value for n. + Z: int, optional + the default value for Z. + P: bool, optional + if provided, the value for P. Note that this is not the default, but will + be enforced. Returns --------