前言

最近在写文章需要绘制一些一维的能量曲线(energy profile)和抽象的二维和三维的网格来表示晶体用来描述自己的算法,于是自己在之前的脚本的基础上进行了整改写成了只提供接口的Python库,基本思想就是封装了matplotlib中相关接口,方便快速搭建和定制自己的能量曲线和网格结构, 代码托管在GitHub上并上传至PyPI。对于研究晶体材料的同学如果想通过python来绘制简单的晶格图像可以参考一下。

正文

首先还是介绍一下这个程序的用途,目前主要是提供三个主要的模块来绘制三方面的内容:

1. 绘制抽象的二维网格结构

catplot提供了丰富的接口用来定制所需要的任何二维网格并进行周期性扩展,如下图是一个通过当个重复单元扩展出来的抽象(100)晶面的二维网格结构:

2. 绘制抽象的三维网格结构

同理只不过这次是在三维画布中进行绘制并进行重复单元的周期性扩展,扩展的效果如下图:

3. 通过插值算法实现绘制”顺滑”的energy profile

实现过程基本是通过对matplotlib提供的绘图组件和接口进一步封装成可以快速搭建上面三个类型图像的组件。

采用二次插值结合样条插值方法绘制 energy profile

energy profile可以理解成在势能面(Potential Energy Surface)上沿着某个特定的方向(反应坐标方向)上能量的变化,

下面我就上一个简单的例子来画一条顺滑的energy profile, 更多具体的例子我都已经jupyter notebook的形式放在的github上(https://github.com/PytLab/catplot/tree/master/examples)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
# 从catplot中导入绘制所需的组件: 画布 和 线
from catplot.ep_components.ep_canvas import EPCanvas
from catplot.ep_components.ep_lines import ElementaryLine

# 创建一个用于绘制energy profile的画布
canvas = EPCanvas()

# 创建一条能量曲线,提供的三个值分别是三个状态下的能量数值
line = ElementaryLine([0.0, 1.2, 0.8])

# 将这条线添加到画布中
canvas.add_line(line)

# 绘制
canvas.draw()
canvas.figure.show()

插值方法

为了能将能量最高点沿着横坐标任意位置移动,我先将顶点的两边用二次函数进行插值,获取两个不同的二次函数形式,然后根据二次函数的形式在左右两边插上5个点,为了能让分开插值的两部分看起来连续,在将上面的10个新插的点和之前的3个点进行一次spline插值即可。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
# 顶点两侧进行二次插值的算法
def quadratic_connect_interp(x1, y1, x2, y2):
A = np.matrix([[x1**2, x1, 1],
[x2**2, x2, 1],
[2*x2, 1, 0]])

b = np.matrix([[y1], [y2], [0]])
x = A.I * b
x.shape = (1, -1)
a, b, c = x.tolist()[0]

poly_func = lambda x: a*x**2 + b*x + c

return poly_func

与插值相关的方法参考:https://github.com/PytLab/catplot/blob/master/catplot/interpolate.py

丰富的接口

除了上面最简单的例子,catplot还提供了丰富的接口来定制和操作energy profile,比如拼接,合并,平移,添加阴影、改变颜色, 辅助线, 修改画布大小,导出插值数据等等。具体的例子参考: https://github.com/PytLab/catplot/tree/master/examples

绘制二维和三维抽象网格

晶格中的原子和键在catplot中被抽象成图中的node和edge,这样我们就可以通过创建图中的node和edge的方式搭建我们网格的重复单元,之后可以通过重复单元的扩展方法来将其扩展成nxn或者nxnxn的网格。

实现的基本方法就是通过matplotlib提供的Line2D, Arrow和scatter相关的接口来将相应node和edge的数据添加到maptlotlib的二维或者三维画布中然后进行绘制和显示。下面给分别给出两个绘制正交网格的绘制方法:

绘制5x5的二维网格

notebook版可以参见:https://github.com/PytLab/catplot/blob/master/examples/grid_2d_examples/expand_supercell.ipynb

创建nodes和edges

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
from catplot.grid_components.nodes import Node2D
from catplot.grid_components.edges import Edge2D

nodes, edges = [], []

# 创建重复单元中的nodes和edge
top = Node2D([0.0, 0.0], size=800, color="#2A6A9C")
t1 = Node2D([0.0, 1.0])
t2 = Node2D([1.0, 0.0])
nodes.append(top)

# 链接这三个node的edges
e1 = Edge2D(top, t1, width=4)
e2 = Edge2D(top, t2, width=4)
edges.extend([e1, e2])

# 中间的nodes
bridge1 = Node2D([0.0, 0.5], style="s", size=600, color="#5A5A5A", alpha=0.6)
bridge2 = Node2D([0.5, 0.0], style="s", size=600, color="#5A5A5A", alpha=0.6)
b1 = bridge1.clone([0.5, 0.5])
b2 = bridge2.clone([0.5, 0.5])
nodes.extend([bridge1, bridge2])

# 连接他们的edges
e1 = Edge2D(bridge1, b1)
e2 = Edge2D(bridge1, bridge2)
e3 = Edge2D(bridge2, b2)
e4 = Edge2D(b1, b2)
edges.extend([e1, e2, e3, e4])

# 正中间的node
h = Node2D([0.5, 0.5], style="h", size=700, color="#5A5A5A", alpha=0.3)
nodes.append(h)

好了,现在我们就创建一个重复单元中的所需的所有元素,可以绘制一下看看效果了

1
2
3
4
5
6
7
8
9
10
11
from catplot.grid_components.grid_canvas import Grid2DCanvas
from catplot.grid_components.supercell import SuperCell2D

canvas = Grid2DCanvas()
# 将上面的元素放到supercell中,后面我们将一supercell为单位进行展开
supercell = SuperCell2D(nodes, edges)

# 绘制效果
canvas.add_supercell(supercell)
canvas.draw()
canvas.figure

OK, 重复单元已经搭建成功,可以以他为单位进行扩展了, 下面我们将其沿着x和y轴方向各进行5次重复扩展。

1
2
# 很简单,就一行代码
expanded_supercell = supercell.expand(5, 5)

来看看效果:

1
2
3
4
canvas_big = Grid2DCanvas(figsize=(30, 20), dpi=60)  # 定制画布大小
canvas_big.add_supercell(expanded_supercell)
canvas_big.draw()
canvas_big.figure

是不是很直观和简单呢?

绘制三维网格

绘制三维网格,catplot中我都写了与二维绘制中相对应的类和接口,这里就不赘述了,可以参考项目中的examples: https://github.com/PytLab/catplot/tree/master/examples/grid_3d_examples/expand_3d_supercell.ipynb

是不是只能画正交的网格?

怎么可能,虽然所有的坐标都是在分数坐标系中定义的,但是在SuperCell类中我添加了分数坐标到笛卡尔坐标的转化,从而可以使得catplot绘制任意的网格。来个例子就知道了:

1
2
3
4
5
6
7
8
9
# 创建nodes和edges的代码与上面的部分完全相同
...

# 但是我们在定义supercell的时候可以修改cell_vectors参数来是重复单元发生形变
supercell = SuperCell2D(nodes, edges, cell_vectors=[[1.0, 0.0],
[0.5, 1.0]])
canvas.add_supercell(supercell)
canvas.draw()
canvas.figure.show()

来我们看看这时候的重复单元是什么样子:

然后我们再将其进行一次3x3的扩展看看

1
2
3
4
5
expanded_supercell = supercell.expand(3, 3)
canvas_big = Grid2DCanvas(figsize=(30, 20), dpi=60)
canvas_big.add_supercell(expanded_supercell)
canvas_big.draw()
canvas_big.figure.show()

所以基本上现在所有类型的晶格都可以通过CatPlot来绘制了。

总结

本来catplot这个库最初是自己用matplotlib来绘图的小脚本,由于现在写论文的情况下需要灵活的绘制网格图,所以进行了重写,现在写成了一个封装了matplotlib的python库方便使用者可以快速搭建自己想要的网格图和绘制漂亮的energy profile。代码和具体使用的notebook格式的例子均开源并放到了github上 (https://github.com/PytLab/catplot ),欢迎有需要的童鞋参考和使用。

Comments