-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathDOS.py
519 lines (411 loc) · 26 KB
/
DOS.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
"""
Script for Visualizing the VASP DOSCAR files.
Reads a DOSCAR input file and returns the DOS plot, while printing the bandgap, VBM and CBM.
The DOSCAR() class contains the general plot() function that accepts arguments of the style ["specie", "color"],
as for example: dos.plot(["M","red"], ["X","blue"], ["T","green"]) to plot the Metal (M), X atom and Terminaton (T) in the specified colours.
Diego Ontiveros
"""
import os
os.environ['OPENBLAS_NUM_THREADS'] = '1'
import numpy as np
import matplotlib.pyplot as plt
from VASP import MX
def getSpin(out):
if len(out[6]) == 3: return "nsp"
elif len(out[6]) == 5: return "sp"
else: raise ValueError("Not able to deduce the spin-polarity of the file")
def isMXene():
###Yet to implement
pass
def getBandGap(E,T,Ef):
"""Takes Energy and Total DOS arrays and returns BandGap, VBM and CBM"""
E = E+Ef
Ef = round(Ef,3)
dE = E[1] - E[0]
searchVBM,searchCBM = False, False
for i,e in enumerate(E):
if e == round(Ef-5*dE,3) or e == round(Ef-5*dE,3)+0.001 or e == round(Ef-5*dE,3)-0.001:
searchVBM = True
if searchVBM:
if T[i] == 0:
VBM = round(E[i-1],3)
searchCBM, searchVBM = True, False
if T[i] != 0 and e > Ef+dE: searchVBM = False
if searchCBM:
if T[i] != 0:
CBM = round(E[i],3)
searchCBM = False
break
else: VBM,CBM = Ef,Ef
bandgap = round(CBM-VBM,3)
return bandgap, VBM, CBM
class DOSCAR():
def __init__(self,path:str,short:bool=False):
self.path = path
self.file = path.split("/")[-1]
self.short = short
##Data gathering and celaning to out list
self.out = self.getData() if not short else self.getDataShort()
self.spin = getSpin(self.out)
self.bandgap = None
def getData(self):
"""Gathers the data from the DOSCAR file."""
out = []
with open(self.path,"r") as infile:
for n,line in enumerate(infile):
rline = line.strip() #Removes inecessary whitespaces at the begining and end of line
line = rline.split() #Splits the line by the whitespaces into sublists
for i,nl in enumerate(line): #Floating the numbers of the lines,
try: nl = float(nl) #except when tey are strings
except ValueError: pass
line[i] = nl
if n == 0: self.nAtoms = int(line[0]) #Number of atoms in cell
if n == 4: self.mx = MX(rline) #Name of the MXene, specified by SYSTEM = Mn+1XnT2 in the INCAR
if n == 5: EMAX, EMIN, self.NEDOS, self.Ef = line[0], line[1], int(line[2]), line[3]
out.append(line)
self.out = out
return self.out
def getDataShort(self):
"""Gathers the firts part of the data from DOSCAR (total DOS)"""
out = []
self.NEDOS = 10
with open(self.path,"r") as inFile:
for n, line in enumerate(inFile):
rline = line.strip() #Removes inecessary whitespaces at the begining and end of line
line = rline.split() #Splits the line by the whitespaces into sublists
for i,nl in enumerate(line): #Floating the numbers of the lines,
try: nl = float(nl) #except when tey are strings
except ValueError: pass
line[i] = nl
if n == 0: self.nAtoms = int(line[0]) #Number of atoms in cell
if n == 4: self.mx = MX(rline) #Name of the MXene, specified by SYSTEM = Mn+1XnT2 in the INCAR
if n == 5: EMAX, EMIN, self.NEDOS, self.Ef = line[0], line[1], int(line[2]), line[3]
if n > self.NEDOS+6: break
out.append(line)
self.out = out
return self.out
def getBandgap(self,return_arrays=False):
"""Computes bandgap from data."""
out = self.out
Ef,nAtoms,NEDOS = self.Ef,self.nAtoms,self.NEDOS
data = out[6:NEDOS+6]
if self.spin == "nsp":
##Loop for obtaining the Total DOS data, taking into account that in the DOSCAR are distributed as E|Tα|Tβ|iTa|iTb
E,T,iT = [np.array([]) for i in range(3)] #Each paramater goes to a dedicated list with its name
for line in data[1:]:
E = np.append(E,line[0]-Ef) #Energy points (corrected by the fermi level, Ef)
T = np.append(T,line[1]) #Total DOS
iT = np.append(iT,line[2]) #Total integration
bandgap, VBM, CBM = getBandGap(E,T,Ef)
self.bandgap = (bandgap, VBM, CBM)
if return_arrays: return bandgap, VBM, CBM, E, T
else: return bandgap, VBM, CBM
elif self.spin == "sp":
##Loop for obtaining the Total DOS data, taking into account that in the DOSCAR are distributed as E|Tα|Tβ|iTa|iTb
E,T,iT,Ta,Tb,iTb,iTa = [np.array([]) for i in range(7)] #Each paramater goes to a dedicated list with its name
for line in data[1:]:
E = np.append(E,line[0]-Ef) #Energy points (corrected by the fermi level, Ef)
Ta = np.append(Ta,line[1]); Tb = np.append(Tb,line[2]) #Total α (Ta) and Total β (Tb) DOS contributions
iTa = np.append(iTa,line[3]); iTb = np.append(iTb,line[4]) #Total DOS integrations for a and b
T = np.append(T,line[1]+line[2]) #Total DOS (Ta+Tb)
iT = np.append(iT,line[3]+line[4]) #Total integration (iTa,iTb)
bandgap, VBM, CBM = getBandGap(E,T,Ef)
self.bandgap = (bandgap, VBM, CBM)
if return_arrays: return bandgap, VBM, CBM, E, T
else: return bandgap, VBM, CBM
def saveImage(self,out_path,params):
"""Saves the generated plot as an image to the given path."""
fmt = params.get("format","png")
extension = "_sp" if self.spc else ""
if not self.mantain_name: out_name = f"{self.file}_{self.mx.name}{extension}.{fmt}"
else: out_name = f"{self.file}{extension}.{fmt}"
file_path = f"{out_path}{out_name}"
outFiles = os.listdir(out_path)
if out_name in outFiles: #In case there are repeated names
file_path = file_path.replace(f".{fmt}",f"(d).{fmt}")
plt.savefig(file_path,format=fmt,dpi=params.get("dpi",1200))
else: plt.savefig(file_path,format=fmt,dpi=params.get("dpi",1200))
# def plot(self, *args, spin=False, xlabel="Energy (eV)", ylabel="DOS", linewidth=1, figsize=(), show=False):
def plot(self,*args, out_path:str=None,mantain_name=False,**params, ):
"""Plot method for MXene DOSCARS. It chooses spin or non-spin polarised automatically. Returns Eg, VBM, CBM \n
`out_path` : Folder Path where the DOS plot will be saved
`*Args` : Specify list with which atoms/orbitals (M,X,Term) you want to see plotted and color [atom,color].\n
`**Params` : Specify the plot parameters (xlabel,ylabel,xlim,ylim,figsize,format,dpi), also, specify if the spin contributions are plotted for with spc=True. """
if self.short: raise ValueError("You cannot use the plot() method when short=True was set!")
if out_path is None: self.out_path = self.path.split(self.file)[0]
else: self.out_path = out_path
if not out_path.endswith("/"): self.out_path += "/"
self.mantain_name = mantain_name
self.params = params
plt.rcParams["figure.autolayout"] = True #To autoajust plot if figsize is changed
# plt.rcParams["font.family"] = "Times New Roman" #Cooler and more formal font
##Gathering the parameters to plot and their respective colour
toPlot = [] ##Podria ser DICT!
colors = []
toPlot.append("T"); colors.append("black") #Spin sp
for p,c in args:
toPlot.append(p)
colors.append(c)
self.toPlot,self.colors = toPlot,colors
if self.spin == "nsp": self.DOS_nsp()
elif self.spin == "sp": self.DOS_sp()
else: raise ValueError("Spin-polarity not detected")
#Añadir caso general // self.spin and mx
def DOS_nsp(self):
"""Plots DOS of MXene from a NON spin-polarized DOSCAR file and returns Bandgap, VBM and CVM.\n
Make sure the file contains SYSTEM name as Mn+1XnT2! (i.e. Ti2C1O2)\n
Args: Specify list with which atoms you want to see plotted and color [line,color].\n
Lines: M, Ms,Mp,Md (total lines are always drawn)."""
#Variable assignation
fname,out = self.file,self.out
Ef,nAtoms,NEDOS = self.Ef,self.nAtoms,self.NEDOS
mx, toPlot,colors, params = self.mx, self.toPlot, self.colors, self.params
# print(f"{mx.mxName}: Ef = {Ef}eV") #Prints the Fermi level of the MXene studied
##List with embeded lists of each atom information [Total,At1,At2,At3,...]
data = [] #Gathers the points for each section of the data in diferent lists
for at in range(nAtoms+1):
data.append(out[(at)*NEDOS+6+(at):(at+1)*NEDOS+6+(at)])
##Loop for obtaining the Total DOS data, taking into account that in the DOSCAR are distributed as E|Tα|Tβ|iTa|iTb
E,T,iT = [np.array([]) for i in range(3)] #Each paramater goes to a dedicated list with its name
for line in data[0][1:]:
E = np.append(E,line[0]-Ef) #Energy points (corrected by the fermi level, Ef)
T = np.append(T,line[1]) #Total DOS
iT = np.append(iT,line[2]) #Total integration
bandgap, VBM, CBM = getBandGap(E,T,Ef)
print(f"{self.mx.mxName}: Eg = {bandgap:.3f} VBM = {VBM:.3f} CBM = {CBM:.3f}",flush=True)
##Loop for obtaining the orbitalic contribution to DOS for each atom. In DOSCAR distributed as the variables order
#s, py,pz,px, dxy,dyz,dxz,dz2,dx2y2 (9A)
#All the orbitals are generated in the orb list, for easier manipulation
orb = [[[] for i in range(nAtoms)] for j in range(9)]
#Separates coulumns by their orbitalic components
for i,atom in enumerate(data[1:]): #For each atom DOS dataset
for line in atom[1:]: #For every line in each atom dataset
for oa in range(len(orb)): #For every atomic orbital (oa) in orb
orb[oa][i].append(line[oa+1]) #Appends DOS coumn to its corresponding OA
for oa in range(9): orb[oa] = np.array(orb[oa]) #Transforma las listas en arrays de numpy for better manipulation
#Each orbital will be a list with the contributions of each atom for that orbital
#It is important to name the variables, as they will serve as the general name to use in the *args for which parameters to plot
s, py,pz,px, dxy,dyz,dxz,dz2,dx2y2 = [orb[i] for i in range(9)]
##Viariable assignment for the different general orbital components (s,p,d). a and b mean alpha and beta
#Since numpy arrays are used, the embeded lists can be added up and each list component will be added to
s,p,d = s, px+py+pz, dxy+dyz+dxz+dz2+dx2y2 #the corresponding one in the adjacent list (only if the
atT = sum(orb) #Total DOS for each atom #lists have the same range)
##Creates arrays for the metal (M) contributions
M, Ms,Mp,Md = [np.zeros(NEDOS-1) for i in range(4)]
for i in range(mx.n+1): #To take the first n+1 lists (where the M data is)
M += atT[i]
Ms += s[i]; Mp += p[i]; Md += d[i]
##Creates arrays for the carbide/nitride (X) contributions
X,Xs,Xp,Xd = [np.zeros(NEDOS-1) for i in range(4)]
for i in range(mx.n+1,2*mx.n+1): #To take the n+1 to 2n+1 lists (where the X data is)
X += atT[i]
Xs += s[i]; Xp += p[i]; Xd += d[i]
##Creates arrays for the termination (Term) contributions (only if there is termination)
Term,Terms,Termp,Termd= [np.zeros(NEDOS-1) for i in range(4)]
Term2,Term2s,Term2p,Term2d= [np.zeros(NEDOS-1) for i in range(4)]
if mx.terminal:
if mx.T_AB: #OH termination not implemented. Works for single atom terminations
for i in range(-4,-2):
Term += atT[i]
Terms += s[i]; Termp += p[i]; Termd += d[i]
for i in range(-2,0):
Term2 += atT[i]
Term2s += s[i]; Term2p += p[i]; Term2d += d[i]
else: #To take the last two lists (where the Term data is)
for i in range(-2,0):
Term += atT[i]
Terms += s[i]; Termp += p[i]; Termd += d[i]
##Loop that draws each specified line (contribution) in *args
fig,plot = plt.subplots(figsize = params.get("figsize", (5,2)))
for i,param in enumerate(toPlot):
#locals()[p] searches local variables of the name p and returns its value, in this case returning the DOS lists
pname = param #Paramater name
param = locals()[param] #For all parameters
#Label assesment for legend. Changing M,X,T for their corresponding element
if pname == "T": pname = "Total"
if "M" in pname: pname = pname.replace("M",mx.atoms[0]+" ")
if "X" in pname: pname = pname.replace("X",mx.atoms[1]+" ")
if mx.terminal and "Term" in pname and not "Term2" in pname:
pname = pname.replace("Term",mx.atoms[2]+" ")
elif "Term" in pname and not "Term2" in pname: continue
if mx.terminal and "Term2" in pname and mx.T_AB: # OH
pname = pname.replace("Term2",mx.atoms[3]+" ")
elif "Term2" in pname: continue
#PLOT. Plots the corresponding parameter introduced in *args with the energy
plot.plot(E,param, label = f"{pname}", color = colors[i], linewidth = 1)
#Plot configuration (title, axis, x inversion, ...)
plot.axhline(0,color = "black", lw = 1)
plot.axvline(0,color = "black", lw = 1, linestyle = "--")
# plot.set_title(f"DOS {mx.mxName}")
plot.set_xlabel(params.get("xlabel","Energy (eV)")) #Energy (eV)
plot.set_ylabel(params.get("ylabel","DOS")) #DOS
plot.set_xlim(params.get("xlim",(-10,10)))
plot.set_ylim(params.get("ylim",(0,20)))
plot.legend(frameon=False,fontsize = "x-small")
#Saves Plot
self.saveImage(self.out_path,params)
if params.get("show",False): plt.show() #In case the plots want to be shown on screen as they are created
plt.close(fig)
def DOS_sp(self):
"""Plots DOS of MXene from a spin-polarized DOSCAR file and returns Bandgap, VBM and CVM.\n
Make sure the file contains SYSTEM name as Mn+1XnT2! (i.e. Ti2C1O2)\n
Specify if you want the plot with spin contribution (α/β) or total (spin = False).\n
Args: Specify list with which atoms you want to see plotted and color [line,color].\n
Lines: M,Ma,Mb Ms,Mp,Md Msa/Msb,Mpa/Mpb,Mda,Mdb (total lines are always drawn)."""
#Variable assignation
fname,out = self.file,self.out
Ef,nAtoms,NEDOS = self.Ef,self.nAtoms,self.NEDOS
mx, toPlot,colors, params = self.mx, self.toPlot, self.colors, self.params
spc = params.get("spc",False)
self.spc = spc
##Gathering of the parameters to plot and their respective colour
if spc == True:
toPlot.pop(0);colors.pop(0)
toPlot.insert(0,"Ta");toPlot.insert(1,"Tb")
colors.insert(0,"red");colors.insert(1,"blue")
#print(f"{mx.mxName}: Ef = {Ef}eV") #Prints the Fermi level of the MXene studied
##List with embeded lists of each atom information [Total,At1,At2,At3,...]
data = [] #Gathers the points for each section of the data in diferent lists
for at in range(nAtoms+1):
data.append(out[(at)*NEDOS+6+(at):(at+1)*NEDOS+6+(at)])
##Loop for obtaining the Total DOS data, taking into account that in the DOSCAR are distributed as E|Tα|Tβ|iTa|iTb
E,T,iT,Ta,Tb,iTb,iTa = [np.array([]) for i in range(7)] #Each paramater goes to a dedicated list with its name
for line in data[0][1:]:
E = np.append(E,line[0]-Ef) #Energy points (corrected by the fermi level, Ef)
Ta = np.append(Ta,line[1]); Tb = np.append(Tb,line[2]) #Total α (Ta) and Total β (Tb) DOS contributions
iTa = np.append(iTa,line[3]); iTb = np.append(iTb,line[4]) #Total DOS integrations for a and b
T = np.append(T,line[1]+line[2]) #Total DOS (Ta+Tb)
iT = np.append(iT,line[3]+line[4]) #Total integration (iTa,iTb)
bandgap, VBM, CBM = getBandGap(E,T,Ef)
print(f"{self.mx.mxName}: Eg = {bandgap:.3f} VBM = {VBM:.3f} CBM = {CBM:.3f}",flush=True)
##Loop for obtaining the orbitalic contribution to DOS for each atom. In DOSCAR distributed as the variables order
#sa,sb, pya,pyb,pza,pzb,pxa,pxb, dxya,dxyb,dyza,dyzb,dxza,dxzb,dz2a,dz2b,dx2y2a,dx2y2b (18 OA)
#All the orbitals are generated in the orb list, for easier manipulation
orb = [[[] for i in range(nAtoms)] for j in range(18)]
#Separates coulumns by their orbitalic components
for i,atom in enumerate(data[1:]): #For each atom DOS dataset
for line in atom[1:]: #For every line in each atom dataset
for oa in range(len(orb)): #For every atomic orbital (oa) in orb
orb[oa][i].append(line[oa+1]) #Appends DOS coumn to its corresponding OA
for oa in range(18): orb[oa] = np.array(orb[oa]) #Transforma las listas en arrays de numpy for better manipulation
#Each orbital will be a list with the contributions of each atom for that orbital
#It is important to namme the variables, as they will serve as the general name to use in the *args for which parameters to plot
sa,sb, pya,pyb,pza,pzb,pxa,pxb, dxya,dxyb,dyza,dyzb,dxza,dxzb,dz2a,dz2b,dx2y2a,dx2y2b = [orb[i] for i in range(18)]
##Viariable assignment for the different general orbital components (s,p,d). a and b mean alpha and beta
pa,pb = pya+pza+pxa,pyb+pzb+pxb #Since numpy arrays are used, the embeded lists can
da,db = dxya+dyza+dz2a+dxza+dx2y2a,dxyb+dyzb+dz2b+dxzb+dx2y2b #be added up and each list component will be added to
s,p,d = sa+sb,pa+pb,da+db #the corresponding one in the adjacent list (only if the
atTa = sa+pa+da #Total DOSa for each atom #lists have the same range)
atTb = sb+pb+db #Total DOSb for each atom
atT = sum(orb) #Total DOS for each atom
##Creates arrays for the metal (M) contributions
M,Ma,Mb, Ms,Mp,Md, Msa,Mpa,Mda,Msb,Mpb,Mdb = [np.zeros(NEDOS-1) for i in range(12)]
for i in range(mx.n+1): #To take the first n+1 lists (where the M data is)
M += atT[i]
Ma += atTa[i]; Mb += atTb[i]
Ms += s[i]; Mp += p[i]; Md += d[i]
Msa += sa[i]; Mpa += pa[i]; Mda += da[i]
Msb += sb[i]; Mpb += pb[i]; Mdb += db[i]
##Creates arrays for the carbide/nitride (X) contributions
X,Xa,Xb, Xs,Xp,Xd, Xsa,Xpa,Xda,Xsb,Xpb,Xdb = [np.zeros(NEDOS-1) for i in range(12)]
for i in range(mx.n+1,2*mx.n+1): #To take the n+1 to 2n+1 lists (where the X data is)
X += atT[i]
Xa += atTa[i]; Xb += atTb[i]
Xs += s[i]; Xp += p[i]; Xd += d[i]
Xsa += sa[i]; Xpa += pa[i]; Xda += da[i]
Xsb += sb[i]; Xpb += pb[i]; Xdb += db[i]
##Creates arrays for the termination (Term) contributions (only if there is termination)
Term,Terma,Termb, Terms,Termp,Termd, Termsa,Termpa,Termda,Termsb,Termpb,Termdb = [np.zeros(NEDOS-1) for i in range(12)]
Term2,Term2a,Term2b, Term2s,Term2p,Term2d, Term2sa,Term2pa,Term2da,Term2sb,Term2pb,Term2db = [np.zeros(NEDOS-1) for i in range(12)]
if mx.terminal:
if mx.T_AB: #OH termination not implemented. Works for single atom terminations
for i in range(-4,-2): # O termination
Term += atT[i]
Terma += atTa[i]; Termb += atTb[i]
Terms += s[i]; Termp += p[i]; Termd += d[i]
Termsa += sa[i]; Termpa += pa[i]; Termda += da[i]
Termsb += sb[i]; Termpb += pb[i]; Termdb += db[i]
for i in range(-2,0): # H termination
Term2 += atT[i]
Term2a += atTa[i]; Term2b += atTb[i]
Term2s += s[i]; Term2p += p[i]; Term2d += d[i]
Term2sa += sa[i]; Term2pa += pa[i]; Term2da += da[i]
Term2sb += sb[i]; Term2pb += pb[i]; Term2db += db[i]
else: #To take the last two lists (where the Term data is)
for i in range(-2,0):
Term += atT[i]
Terma += atTa[i]; Termb += atTb[i]
Terms += s[i]; Termp += p[i]; Termd += d[i]
Termsa += sa[i]; Termpa += pa[i]; Termda += da[i]
Termsb += sb[i]; Termpb += pb[i]; Termdb += db[i]
if spc: fig,plot = plt.subplots(figsize = params.get("figsize", (5,3)))
else: fig,plot = plt.subplots(figsize = params.get("figsize", (5,2)))
for i,param in enumerate(toPlot):
#locals()[p] searches local variables of the name p and returns its value, in this case returning the DOS lists
pname = param #Paramater name
if pname[-1] == "b": param = -locals()[param] #For spin b parameters
else: param = locals()[param] #For spin a parameters
#Label assesment for legend. Changing M,X,T for their corresponding element
pname = pname.replace("a","α")
pname = pname.replace("b","β")
if pname == "T": pname = "Total"
if pname == "Tα": pname = "Total α"
if pname == "Tβ": pname = "Total β"
if "M" in pname: pname = pname.replace("M",mx.atoms[0]+" ")
if "X" in pname: pname = pname.replace("X",mx.atoms[1]+" ")
if mx.terminal and "Term" in pname and not "Term2" in pname:
pname = pname.replace("Term",mx.atoms[2]+" ")
elif "Term" in pname and not "Term2" in pname: continue
if mx.terminal and "Term2" in pname and mx.T_AB: # OH
pname = pname.replace("Term2",mx.atoms[3]+" ")
elif "Term2" in pname: continue
#! ADD ABC case and correct AB case
#PLOT. Plots the corresponding parameter introduced in *args with the energy
plot.plot(E,param, label = f"{pname}", color = colors[i], linewidth = 1)
#Plot configuration (title, axis, x inversion, ...)
plot.axhline(0,color = "black", lw = 1)
plot.axvline(0,color = "black", lw = 1, linestyle = "--")
# plot.set_title(f"DOS {mx.mxName}")
plot.set_xlabel(params.get("xlabel","Energy (eV)")) #Energy (eV)
plot.set_ylabel(params.get("ylabel","DOS")) #DOS
plot.set_xlim(params.get("xlim",(-10,10)))
if not spc: plot.set_ylim(params.get("ylim",(0,10)))
elif spc: plot.set_ylim([-15,15])
plot.legend(frameon=False,fontsize = "x-small")
#Saves Plot as .png
self.saveImage(self.out_path,params)
if params.get("show",False): plt.show() #In case the plots want to be shown on screen as they are created
plt.close(fig)
def DOS_general(self):
#Yet to implement
pass
### -------------------------------------------- MAIN PROGRAM ---------------------------------------------------- ###
###--------------------------------------------------------------------------------------------------------------- ###
if __name__ == "__main__":
# Creates DOSout folder where the plots will save
try: os.mkdir("DOSout")
except FileExistsError: pass
#List of input files to open, in DOS in folder
inFiles = os.listdir("./DOSin")
inFiles = ["./DOSin/" + f for f in inFiles if os.path.isfile("./DOSin/"+f)]
#Loop for each input file in DOSin
for file in inFiles:
dos = DOSCAR(file)
print(dos.getBandgap())
# Plots PDOS (non-spin polarized)
dos.plot(
["M","red"],["X","blue"],["Term","green"],["Term2","pink"],
spc = False,
out_path = "DOSout",
mantain_name=True
)
continue
# Plots spin contributions (if spin polarized)
if dos.spin=="sp":
dos.plot(
["Ma","orange"], ["Mb","cyan"],["Xa","pink"],["Xb","violet"],
["Terma","yellow"],["Termb","grey"],
spc = True,
out_path = "DOSout",
mantain_name=True
)