Atomic Positions Reading & Outputing

2 minute read

Published:

only work for a & b axes perpendicular to c axis now.

PWscf format

# -*- coding: utf-8 -*-
# Date 2024Dec22
# =============================

import numpy as np
import matplotlib
import matplotlib.pyplot as plt
from matplotlib.font_manager import FontProperties
from matplotlib.ticker import MultipleLocator
import os
# =========================== need change ========================================
input_file = ''
work_dir = input_file[:len(input_file)-input_file[::-1].index('/')]
input_file = input_file[len(input_file)-input_file[::-1].index('/'):]
os.chdir(work_dir)
# =========================== need change ========================================

fdata=open(input_file,'r')

with open(input_file,'r') as f:
    elelist = []
    atompos = []
    marknum = 0
    atomnum = 0
    for line_index, line in enumerate(f):
        if marknum != 0:
            try:
                float(line.split()[1])
            except ValueError:
                break
            except IndexError:
                break
            p=line.split()
            try:
                elelist.index(p[0])
            except ValueError:
                elelist.append(p[0])
                atompos.append([])
            atompos[elelist.index(p[0])].append([float(p[1]),float(p[2]),float(p[3])])
            atomnum+=1
        try:
            line.split().index('ATOMIC_POSITIONS')
        except ValueError:
            continue
        except IndexError:
            continue
        marknum=line_index


for index_ele, ele in enumerate(atompos):
    atompos[index_ele] = np.array(atompos[index_ele])
    atompos[index_ele] = atompos[index_ele][np.argsort(atompos[index_ele][:,2])]
for i in range(len(elelist)):
    for j in range(len(atompos[i])):
        pos=atompos[i][j]
        print(f"{elelist[i]:<6}  {pos[0]:12.8f}      {pos[1]:12.8f}      {pos[2]:12.8f}")
print(f"atom number: {atomnum:.0f}")

VESTA format

# -*- coding: utf-8 -*-
# Date 2024Dec22
# =============================

import numpy as np
import matplotlib
import matplotlib.pyplot as plt
from matplotlib.font_manager import FontProperties
from matplotlib.ticker import MultipleLocator
import os

# =========================== need change ========================================
input_file = ''
work_dir = input_file[:len(input_file)-input_file[::-1].index('/')]
input_file = input_file[len(input_file)-input_file[::-1].index('/'):]
os.chdir(work_dir)
fdata=open(input_file,'r')
# =========================== need change ========================================

with open(input_file,'r') as f:
	cell = []
	elelist = []
	atompos = []
	cellnum = 0
	markcellnum = 0
	strucnum = 0
	atomnum = 0
	for line_index, line in enumerate(f):
	    if cellnum != 0 and strucnum == 0 and markcellnum == 0:
		try:
		    float(line.split()[0])
		except ValueError:
		    continue
		except IndexError:
		    continue
		p=line.split()
		cell.append([float(p[0]),0,0])
		cell.append([float(p[1])*np.sin(np.pi/180*(90-float(p[5]))),float(p[1])*np.cos(np.pi/180*(90-float(p[5]))),0])
		cell.append([0,0,float(p[2])])
		markcellnum = 1
	    elif cellnum != 0 and strucnum != 0:
		try:
		    float(line.split()[0])
		except ValueError:
		    break
		except IndexError:
		    break
		if float(line.split()[0]) != 0:
		    p=line.split()
		    atomnum = int(p[0])
		    try:
		        elelist.index(p[1])
		    except ValueError:
		        elelist.append(p[1])
		        atompos.append([])
		    atompos[elelist.index(p[1])].append([float(p[5]),float(p[4]),float(p[6])])
	    if cellnum == 0:
		try:
		    line.split()[0].index('CELL')
		except ValueError:
		    continue
		except IndexError:
		    continue
		cellnum=line_index
	    else:
		try:
		    line.split().index('STRUC')
		except ValueError:
		    continue
		except IndexError:
		    continue
		strucnum=line_index

cell = np.array(cell)
for index_ele, ele in enumerate(atompos):
    pos=np.array(ele)
    atompos[index_ele] = np.dot(pos,cell)
    atompos[index_ele] = atompos[index_ele][np.argsort(atompos[index_ele][:,2])]
for i in range(len(elelist)):
    for j in range(len(atompos[i])):
        pos=atompos[i][j]
        print(f"{elelist[i]:<6}  {pos[0]:12.8f}      {pos[1]:12.8f}      {pos[2]:12.8f}")
print(f"atom number: {atomnum:.0f}")

Output

Si        0.00000000        0.00000000        4.48626210
Si       -0.00000155        1.78939174        7.01424621
Si        0.00000000       -0.00000000        9.54366290
Si        1.54965948        0.89469453       12.07164700
C         0.00000000       -0.00000000        2.58684602
C        -0.00000155        1.78939174        5.10863375
C         0.00000000        0.00000000        7.64424681
C         1.54965948        0.89469453       10.16603455
atom number: 8