init
This commit is contained in:
@@ -0,0 +1,492 @@
|
||||
"""VisPy 演示:加载预计算轨迹数据,驱动小球运动动画。
|
||||
|
||||
计算与显示完全分离:
|
||||
1. 先运行 compute.py → 生成 output/trajectory.txt(全量 NT 步轨迹)
|
||||
2. 再运行 sample.py → 从 output/trajectory.txt 抽帧生成 output/display.txt
|
||||
3. 本文件加载 output/display.txt,按帧播放动画
|
||||
|
||||
用法:
|
||||
python draw.py # 使用 dynamics 根目录下的 output/
|
||||
python draw.py examples/case01/output # 指定案例输出目录
|
||||
"""
|
||||
|
||||
import numpy as np
|
||||
import os
|
||||
import sys
|
||||
from vispy import app, scene
|
||||
from vispy.visuals.transforms import STTransform
|
||||
|
||||
sys.path.insert(0, os.path.dirname(os.path.abspath(__file__)))
|
||||
import compute
|
||||
|
||||
# ===========================================================================
|
||||
# 加载预计算轨迹
|
||||
# ===========================================================================
|
||||
script_dir = os.path.dirname(os.path.abspath(__file__))
|
||||
if len(sys.argv) > 1:
|
||||
# 用户指定了输出目录
|
||||
output_dir = os.path.abspath(sys.argv[1])
|
||||
else:
|
||||
output_dir = compute.get_output_dir(script_dir)
|
||||
os.environ["DYNAMICS_OUTPUT_DIR"] = output_dir
|
||||
disp_path = os.path.join(output_dir, "display.txt")
|
||||
|
||||
if not os.path.exists(disp_path):
|
||||
raise FileNotFoundError(
|
||||
f"找不到 display.txt!\n"
|
||||
f"期望路径: {disp_path}\n"
|
||||
f"请先运行 compute.py 计算轨迹,再运行 sample.py 生成显示数组。\n"
|
||||
f"用法: python draw.py [案例输出目录]"
|
||||
)
|
||||
|
||||
disp_data = compute.load_text_data(disp_path)
|
||||
|
||||
# 单原子数据(plot_atom:用于信息显示)
|
||||
DISP_X = disp_data["disp_x"]
|
||||
DISP_Y = disp_data["disp_y"]
|
||||
DISP_Z = disp_data["disp_z"]
|
||||
DISP_VX = disp_data["disp_vx"]
|
||||
DISP_VY = disp_data["disp_vy"]
|
||||
DISP_VZ = disp_data["disp_vz"]
|
||||
|
||||
# 全原子数据(用于多球绘制)
|
||||
DISP_ALL_X = disp_data["disp_all_x"] # (n_frames, n_atoms)
|
||||
DISP_ALL_Y = disp_data["disp_all_y"]
|
||||
DISP_ALL_Z = disp_data["disp_all_z"]
|
||||
DISP_ALL_VX = disp_data["disp_all_vx"]
|
||||
DISP_ALL_VY = disp_data["disp_all_vy"]
|
||||
DISP_ALL_VZ = disp_data["disp_all_vz"]
|
||||
|
||||
DISP_T = disp_data["disp_t"]
|
||||
DISP_STEP = disp_data["disp_step"]
|
||||
N_FRAMES = int(disp_data["n_frames"])
|
||||
NT = int(disp_data["NT"])
|
||||
DT = float(disp_data["DT"])
|
||||
NSTEP = int(disp_data["NSTEP"])
|
||||
|
||||
# 原子信息
|
||||
ATOM_IDS = disp_data.get("atom_ids", np.array([1]))
|
||||
ATOM_RADII = disp_data.get("atom_radii", np.array([float(disp_data["ball_radius"])]))
|
||||
N_ATOMS = len(ATOM_IDS)
|
||||
PLOT_ATOM_ROW = int(disp_data.get("plot_atom_row", 0))
|
||||
PLOT_ATOM_ID = int(disp_data.get("plot_atom_id", ATOM_IDS[0]))
|
||||
BOND_PAIRS = disp_data.get("bond_pairs", [])
|
||||
|
||||
if N_FRAMES <= 0:
|
||||
raise ValueError(
|
||||
"output/display.txt 中没有可播放的帧,请检查 sample_start/sample_end/NSTEP 配置。")
|
||||
|
||||
# 保留模拟边界常量(用于场景缩放、相机等),从 output/display.txt 中读取
|
||||
X_MIN = float(disp_data["X_MIN"]); X_MAX = float(disp_data["X_MAX"])
|
||||
Y_MIN = float(disp_data["Y_MIN"]); Y_MAX = float(disp_data["Y_MAX"])
|
||||
Z_MIN = float(disp_data["Z_MIN"]); Z_MAX = float(disp_data["Z_MAX"])
|
||||
X0 = float(disp_data["X0"]); Y0 = float(disp_data["Y0"]); Z0 = float(disp_data["Z0"])
|
||||
raw_alpha = disp_data["alpha"]
|
||||
if isinstance(raw_alpha, (list, tuple, np.ndarray)):
|
||||
alpha_list = [float(a) for a in raw_alpha]
|
||||
if len(alpha_list) != 6:
|
||||
raise ValueError(f"alpha 数组长度须为 6,实际为 {len(alpha_list)}")
|
||||
else:
|
||||
alpha_list = [float(raw_alpha)] * 6
|
||||
# 绘图参数
|
||||
ball_radius = float(disp_data["ball_radius"])
|
||||
ball_color_r = float(disp_data["ball_color_r"])
|
||||
ball_color_g = float(disp_data["ball_color_g"])
|
||||
ball_color_b = float(disp_data["ball_color_b"])
|
||||
box_color_r = float(disp_data["box_color_r"])
|
||||
box_color_g = float(disp_data["box_color_g"])
|
||||
box_color_b = float(disp_data["box_color_b"])
|
||||
|
||||
|
||||
# ===========================================================================
|
||||
# 图形界面无关的几何参数(不参与物理计算,仅用于场景外观)
|
||||
# ===========================================================================
|
||||
|
||||
info_margin = 36
|
||||
axis_length = 10.0
|
||||
|
||||
initial_camera = {
|
||||
"distance": 40.0,
|
||||
"elevation": 0,
|
||||
"azimuth": 0,
|
||||
"center": (0, 0, 0),
|
||||
}
|
||||
|
||||
|
||||
# ===========================================================================
|
||||
# 创建画布与相机
|
||||
# ===========================================================================
|
||||
canvas = scene.SceneCanvas(
|
||||
keys="interactive",
|
||||
size=(1000, 700),
|
||||
bgcolor=(0.08, 0.08, 0.10, 1.0),
|
||||
show=True,
|
||||
)
|
||||
|
||||
view = canvas.central_widget.add_view()
|
||||
view.camera = "turntable"
|
||||
view.camera.distance = initial_camera["distance"]
|
||||
view.camera.elevation = initial_camera["elevation"]
|
||||
view.camera.azimuth = initial_camera["azimuth"]
|
||||
view.camera.center = initial_camera["center"]
|
||||
|
||||
|
||||
# ===========================================================================
|
||||
# 场景对象
|
||||
# ===========================================================================
|
||||
axis_width = 3
|
||||
arrow_size = 14
|
||||
axes_visible = True
|
||||
axes_group = []
|
||||
|
||||
axes_group.append(scene.visuals.Arrow(
|
||||
pos=np.array([[0, 0, 0], [axis_length, 0, 0]], dtype=np.float32),
|
||||
color=(1.0, 0.2, 0.2, 1.0),
|
||||
width=axis_width,
|
||||
arrows=np.array([[0, 0, 0, axis_length, 0, 0]], dtype=np.float32),
|
||||
arrow_size=arrow_size,
|
||||
parent=view.scene,
|
||||
))
|
||||
axes_group.append(scene.visuals.Arrow(
|
||||
pos=np.array([[0, 0, 0], [0, axis_length, 0]], dtype=np.float32),
|
||||
color=(0.2, 1.0, 0.2, 1.0),
|
||||
width=axis_width,
|
||||
arrows=np.array([[0, 0, 0, 0, axis_length, 0]], dtype=np.float32),
|
||||
arrow_size=arrow_size,
|
||||
parent=view.scene,
|
||||
))
|
||||
axes_group.append(scene.visuals.Arrow(
|
||||
pos=np.array([[0, 0, 0], [0, 0, axis_length]], dtype=np.float32),
|
||||
color=(0.3, 0.6, 1.0, 1.0),
|
||||
width=axis_width,
|
||||
arrows=np.array([[0, 0, 0, 0, 0, axis_length]], dtype=np.float32),
|
||||
arrow_size=arrow_size,
|
||||
parent=view.scene,
|
||||
))
|
||||
|
||||
axes_group.append(scene.visuals.Text(text="x", color=(1.0, 0.2, 0.2, 1.0), font_size=18,
|
||||
pos=(axis_length + 0.2, 0, 0), anchor_x="left", anchor_y="center", parent=view.scene))
|
||||
axes_group.append(scene.visuals.Text(text="y", color=(0.2, 1.0, 0.2, 1.0), font_size=18,
|
||||
pos=(0, axis_length + 0.2, 0), anchor_x="left", anchor_y="bottom", parent=view.scene))
|
||||
axes_group.append(scene.visuals.Text(text="z", color=(0.3, 0.6, 1.0, 1.0), font_size=18,
|
||||
pos=(0, 0, axis_length + 0.2), anchor_x="left", anchor_y="bottom", parent=view.scene))
|
||||
|
||||
# 所有小球(每个原子一个球,不同颜色)
|
||||
TAB10_RGB = np.array([
|
||||
[0.1216, 0.4667, 0.7059], # 蓝
|
||||
[0.8902, 0.4667, 0.1137], # 橙
|
||||
[0.1725, 0.6275, 0.1725], # 绿
|
||||
[0.8392, 0.1529, 0.1569], # 红
|
||||
[0.5804, 0.4039, 0.7412], # 紫
|
||||
[0.5490, 0.3373, 0.2941], # 棕
|
||||
[0.8902, 0.4667, 0.7608], # 粉
|
||||
[0.4980, 0.4980, 0.4980], # 灰
|
||||
[0.7373, 0.7412, 0.1333], # 黄绿
|
||||
[0.0902, 0.7451, 0.8118], # 青
|
||||
])
|
||||
balls = []
|
||||
for i in range(N_ATOMS):
|
||||
r, g, b = TAB10_RGB[i % len(TAB10_RGB)]
|
||||
s = scene.visuals.Sphere(
|
||||
radius=float(ATOM_RADII[i]), method="latitude",
|
||||
color=(r, g, b, 1.0), edge_color=None, parent=view.scene)
|
||||
s.mesh.shading = "smooth"
|
||||
balls.append(s)
|
||||
|
||||
# 成键线(原子之间的连接)
|
||||
if len(BOND_PAIRS) > 0:
|
||||
n_bonds = len(BOND_PAIRS)
|
||||
bond_pos = np.zeros((n_bonds * 2, 3), dtype=np.float32)
|
||||
bond_lines = scene.visuals.Line(
|
||||
pos=bond_pos, color=(0.7, 0.7, 0.7, 0.8), width=3,
|
||||
connect="segments", parent=view.scene)
|
||||
else:
|
||||
bond_lines = None
|
||||
|
||||
# 六个面形成立方体边界(每个面独立透明度,alpha<=0 时隐藏该面)
|
||||
box_size = X_MAX - X_MIN
|
||||
faces = [
|
||||
((X_MAX, 0, 0), "-x"),
|
||||
((X_MIN, 0, 0), "+x"),
|
||||
((0, Y_MAX, 0), "-y"),
|
||||
((0, Y_MIN, 0), "+y"),
|
||||
((0, 0, Z_MAX), "-z"),
|
||||
((0, 0, Z_MIN), "+z"),
|
||||
]
|
||||
for f_idx, (pos, direction) in enumerate(faces):
|
||||
a = alpha_list[f_idx]
|
||||
if a <= 0:
|
||||
continue
|
||||
face_color = (box_color_r, box_color_g, box_color_b, a)
|
||||
plane = scene.visuals.Plane(
|
||||
width=box_size, height=box_size, width_segments=1, height_segments=1,
|
||||
direction=direction, color=face_color, parent=view.scene)
|
||||
plane.set_gl_state(depth_test=False, blend=True)
|
||||
plane.transform = STTransform(translate=pos)
|
||||
|
||||
# 右上角:相机信息
|
||||
camera_info = scene.visuals.Text(
|
||||
text="", color="white", font_size=14,
|
||||
pos=(0, 0), anchor_x="right", anchor_y="top", parent=canvas.scene)
|
||||
|
||||
# 左上角:小球信息
|
||||
ball_info = scene.visuals.Text(
|
||||
text="", color=(0.2, 1.0, 0.2, 1.0), font_size=28,
|
||||
pos=(0, 0), anchor_x="left", anchor_y="top",
|
||||
face="黑体", bold=True, parent=canvas.scene)
|
||||
|
||||
# 左上角:reset 按钮(在 info 上方)
|
||||
reset_btn_size = (60, 30)
|
||||
reset_button = scene.visuals.Rectangle(
|
||||
center=(reset_btn_size[0] / 2 + 8, reset_btn_size[1] / 2 + 8),
|
||||
width=reset_btn_size[0], height=reset_btn_size[1],
|
||||
radius=6, color=(0.18, 0.35, 0.65, 0.85),
|
||||
border_color="white", parent=canvas.scene)
|
||||
reset_button_label = scene.visuals.Text(
|
||||
text="reset", color="white", font_size=16,
|
||||
pos=(reset_btn_size[0] / 2 + 8, reset_btn_size[1] / 2 + 8),
|
||||
anchor_x="center", anchor_y="center",
|
||||
bold=True, parent=canvas.scene)
|
||||
|
||||
# 信息显示/隐藏 切换按钮(左上角小方块,在 reset 下方)
|
||||
info_btn_size = (60, 30)
|
||||
info_toggle_visible = True
|
||||
info_button = scene.visuals.Rectangle(
|
||||
center=(info_btn_size[0] / 2 + 8, info_btn_size[1] / 2 + 8),
|
||||
width=info_btn_size[0], height=info_btn_size[1],
|
||||
radius=6, color=(0.9, 0.3, 0.3, 0.9),
|
||||
border_color="white", parent=canvas.scene)
|
||||
info_button_label = scene.visuals.Text(
|
||||
text="info", color="white", font_size=16,
|
||||
pos=(info_btn_size[0] / 2 + 8, info_btn_size[1] / 2 + 8),
|
||||
anchor_x="center", anchor_y="center",
|
||||
bold=True, parent=canvas.scene)
|
||||
|
||||
# 强制在初始化时定位到左上角(canvas.scene坐标系:左下角为原点)
|
||||
_cw, _ch = canvas.size
|
||||
reset_button.center = (reset_btn_size[0] / 2 + 8, _ch - reset_btn_size[1] / 2 - 8 - info_btn_size[1] - 4)
|
||||
reset_button_label.pos = reset_button.center
|
||||
info_button.center = (info_btn_size[0] / 2 + 8, _ch - info_btn_size[1] / 2 - 8)
|
||||
info_button_label.pos = info_button.center
|
||||
|
||||
# axes 显示/隐藏 按钮(在 info 下方)
|
||||
axes_btn_size = (60, 30)
|
||||
axes_button = scene.visuals.Rectangle(
|
||||
center=(axes_btn_size[0] / 2 + 8, axes_btn_size[1] / 2 + 8),
|
||||
width=axes_btn_size[0], height=axes_btn_size[1],
|
||||
radius=6, color=(0.3, 0.7, 0.3, 0.9),
|
||||
border_color="white", parent=canvas.scene)
|
||||
axes_button_label = scene.visuals.Text(
|
||||
text="axes", color="white", font_size=16,
|
||||
pos=(axes_btn_size[0] / 2 + 8, axes_btn_size[1] / 2 + 8),
|
||||
anchor_x="center", anchor_y="center",
|
||||
bold=True, parent=canvas.scene)
|
||||
axes_button.center = (axes_btn_size[0] / 2 + 8, _ch - axes_btn_size[1] / 2 - 8 - info_btn_size[1] - 4 - axes_btn_size[1] - 4)
|
||||
axes_button_label.pos = axes_button.center
|
||||
|
||||
|
||||
# ===========================================================================
|
||||
# 回调函数
|
||||
# ===========================================================================
|
||||
|
||||
def update_camera_info(event=None):
|
||||
c = view.camera
|
||||
camera_info.text = (
|
||||
"Camera\n"
|
||||
f"center = ({c.center[0]:.2f}, {c.center[1]:.2f}, {c.center[2]:.2f})\n"
|
||||
f"distance = {c.distance:.2f}\n"
|
||||
f"elevation = {c.elevation:.2f}\n"
|
||||
f"azimuth = {c.azimuth:.2f}"
|
||||
)
|
||||
|
||||
|
||||
def update_ball_info(frame_idx, x, y, z, vx, vy, vz):
|
||||
step = int(DISP_STEP[frame_idx])
|
||||
t = float(DISP_T[frame_idx])
|
||||
ball_info.text = (
|
||||
f"原子 {PLOT_ATOM_ID} (共 {N_ATOMS} 个原子)\n"
|
||||
f"frame {frame_idx+1}/{N_FRAMES} | saved step {step}/{NT-1}\n"
|
||||
f"t = {t:.2f} s | dt = {DT:.3f} s | nstep = {NSTEP}\n"
|
||||
f"Position: ({x:.2f}, {y:.2f}, {z:.2f})\n"
|
||||
f"Velocity: ({vx:.2f}, {vy:.2f}, {vz:.2f})\n"
|
||||
)
|
||||
|
||||
|
||||
def reposition_camera_info(event=None):
|
||||
width, height = canvas.size
|
||||
camera_info.pos = (width - 20, height - 20)
|
||||
# reset → info → axes 三按钮纵向排列
|
||||
gap = 4
|
||||
info_y = info_btn_size[1] / 2 + 8
|
||||
reset_y = info_y + info_btn_size[1] + gap
|
||||
axes_y = info_y + info_btn_size[1] + gap + axes_btn_size[1] + gap
|
||||
reset_button.center = (reset_btn_size[0] / 2 + 8, height - reset_y)
|
||||
reset_button_label.pos = reset_button.center
|
||||
info_button.center = (info_btn_size[0] / 2 + 8, height - info_y)
|
||||
info_button_label.pos = info_button.center
|
||||
axes_button.center = (axes_btn_size[0] / 2 + 8, height - axes_y)
|
||||
axes_button_label.pos = axes_button.center
|
||||
# info 文字放在所有按钮下方
|
||||
buttons_bottom = height - (axes_y + axes_btn_size[1] / 2)
|
||||
ball_info.pos = (info_margin, buttons_bottom - 10)
|
||||
update_ball_info(frame_idx, DISP_X[frame_idx], DISP_Y[frame_idx], DISP_Z[frame_idx],
|
||||
DISP_VX[frame_idx], DISP_VY[frame_idx], DISP_VZ[frame_idx])
|
||||
update_camera_info()
|
||||
|
||||
|
||||
def handle_view_interaction(event):
|
||||
update_camera_info()
|
||||
|
||||
|
||||
def rotate_about_screen_normal(angle):
|
||||
if hasattr(view.camera, "roll"):
|
||||
view.camera.roll = (view.camera.roll + angle) % 360
|
||||
else:
|
||||
view.camera.azimuth = (view.camera.azimuth + angle) % 360
|
||||
update_camera_info()
|
||||
|
||||
|
||||
def handle_key_press(event):
|
||||
key_name = ""
|
||||
if getattr(event, "text", None):
|
||||
key_name = event.text.lower()
|
||||
elif getattr(event, "key", None) is not None:
|
||||
key_name = str(event.key).lower()
|
||||
if key_name == "q":
|
||||
rotate_about_screen_normal(-90)
|
||||
elif key_name == "e":
|
||||
rotate_about_screen_normal(90)
|
||||
|
||||
|
||||
def reset_camera_view():
|
||||
global frame_idx
|
||||
frame_idx = 0
|
||||
# 立即复位所有小球到第 0 帧
|
||||
for i in range(N_ATOMS):
|
||||
balls[i].transform = STTransform(translate=(
|
||||
float(DISP_ALL_X[frame_idx, i]),
|
||||
float(DISP_ALL_Y[frame_idx, i]),
|
||||
float(DISP_ALL_Z[frame_idx, i]),
|
||||
))
|
||||
if bond_lines is not None and len(BOND_PAIRS) > 0:
|
||||
pos = np.zeros((len(BOND_PAIRS) * 2, 3), dtype=np.float32)
|
||||
for b_idx, (i, j) in enumerate(BOND_PAIRS):
|
||||
pos[b_idx * 2] = [DISP_ALL_X[frame_idx, i], DISP_ALL_Y[frame_idx, i], DISP_ALL_Z[frame_idx, i]]
|
||||
pos[b_idx * 2 + 1] = [DISP_ALL_X[frame_idx, j], DISP_ALL_Y[frame_idx, j], DISP_ALL_Z[frame_idx, j]]
|
||||
bond_lines.set_data(pos=pos)
|
||||
# 复位相机
|
||||
view.camera.distance = initial_camera["distance"]
|
||||
view.camera.elevation = initial_camera["elevation"]
|
||||
view.camera.azimuth = initial_camera["azimuth"]
|
||||
view.camera.center = initial_camera["center"]
|
||||
if hasattr(view.camera, "roll"):
|
||||
view.camera.roll = 0
|
||||
update_camera_info()
|
||||
|
||||
|
||||
def handle_mouse_press(event):
|
||||
global info_toggle_visible, axes_visible
|
||||
if event.pos is None:
|
||||
return
|
||||
ex, ey = event.pos
|
||||
|
||||
# reset 按钮
|
||||
bx, by = reset_button.center
|
||||
lw = reset_btn_size[0] / 2; lh = reset_btn_size[1] / 2
|
||||
if (bx - lw) <= ex <= (bx + lw) and (by - lh) <= ey <= (by + lh):
|
||||
reset_camera_view()
|
||||
return
|
||||
|
||||
# info 按钮
|
||||
bx, by = info_button.center
|
||||
if (bx - info_btn_size[0]/2) <= ex <= (bx + info_btn_size[0]/2) and (by - info_btn_size[1]/2) <= ey <= (by + info_btn_size[1]/2):
|
||||
info_toggle_visible = not info_toggle_visible
|
||||
ball_info.visible = info_toggle_visible
|
||||
camera_info.visible = info_toggle_visible
|
||||
return
|
||||
|
||||
# axes 按钮
|
||||
bx, by = axes_button.center
|
||||
if (bx - axes_btn_size[0]/2) <= ex <= (bx + axes_btn_size[0]/2) and (by - axes_btn_size[1]/2) <= ey <= (by + axes_btn_size[1]/2):
|
||||
axes_visible = not axes_visible
|
||||
for axis in axes_group:
|
||||
axis.visible = axes_visible
|
||||
|
||||
|
||||
# ===========================================================================
|
||||
# 动画初始化
|
||||
# ===========================================================================
|
||||
frame_idx = 0
|
||||
|
||||
# 初始帧:摆放所有小球并刷新 UI
|
||||
for i in range(N_ATOMS):
|
||||
balls[i].transform = STTransform(translate=(
|
||||
float(DISP_ALL_X[frame_idx, i]),
|
||||
float(DISP_ALL_Y[frame_idx, i]),
|
||||
float(DISP_ALL_Z[frame_idx, i]),
|
||||
))
|
||||
# 初始帧:更新成键线
|
||||
if bond_lines is not None and len(BOND_PAIRS) > 0:
|
||||
pos = np.zeros((len(BOND_PAIRS) * 2, 3), dtype=np.float32)
|
||||
for b_idx, (i, j) in enumerate(BOND_PAIRS):
|
||||
pos[b_idx * 2] = [DISP_ALL_X[frame_idx, i], DISP_ALL_Y[frame_idx, i], DISP_ALL_Z[frame_idx, i]]
|
||||
pos[b_idx * 2 + 1] = [DISP_ALL_X[frame_idx, j], DISP_ALL_Y[frame_idx, j], DISP_ALL_Z[frame_idx, j]]
|
||||
bond_lines.set_data(pos=pos)
|
||||
reposition_camera_info()
|
||||
update_ball_info(frame_idx,
|
||||
float(DISP_X[frame_idx]), float(DISP_Y[frame_idx]), float(DISP_Z[frame_idx]),
|
||||
float(DISP_VX[frame_idx]), float(DISP_VY[frame_idx]), float(DISP_VZ[frame_idx]))
|
||||
|
||||
print(f"[draw] 加载 output/display.txt: {N_FRAMES} 帧, {N_ATOMS} 个原子, NT={NT}, DT={DT}, NSTEP={NSTEP}")
|
||||
print(f"[draw] 绘图参数: ball_radius={ball_radius}, box_color=({box_color_r:.2f},{box_color_g:.2f},{box_color_b:.2f}), alpha={alpha_list}")
|
||||
|
||||
|
||||
# ===========================================================================
|
||||
# 每帧回调:仅推进帧索引,从预存数组读取位置,零物理计算
|
||||
# ===========================================================================
|
||||
def update(event):
|
||||
global frame_idx
|
||||
frame_idx = (frame_idx + 1) % N_FRAMES # 循环播放
|
||||
|
||||
# 更新所有小球位置
|
||||
for i in range(N_ATOMS):
|
||||
x = float(DISP_ALL_X[frame_idx, i])
|
||||
y = float(DISP_ALL_Y[frame_idx, i])
|
||||
z = float(DISP_ALL_Z[frame_idx, i])
|
||||
balls[i].transform = STTransform(translate=(x, y, z))
|
||||
|
||||
# 更新成键线
|
||||
if bond_lines is not None and len(BOND_PAIRS) > 0:
|
||||
pos = np.zeros((len(BOND_PAIRS) * 2, 3), dtype=np.float32)
|
||||
for b_idx, (i, j) in enumerate(BOND_PAIRS):
|
||||
pos[b_idx * 2] = [DISP_ALL_X[frame_idx, i], DISP_ALL_Y[frame_idx, i], DISP_ALL_Z[frame_idx, i]]
|
||||
pos[b_idx * 2 + 1] = [DISP_ALL_X[frame_idx, j], DISP_ALL_Y[frame_idx, j], DISP_ALL_Z[frame_idx, j]]
|
||||
bond_lines.set_data(pos=pos)
|
||||
|
||||
# 信息面板显示 plot_atom 的数据
|
||||
x = float(DISP_X[frame_idx])
|
||||
y = float(DISP_Y[frame_idx])
|
||||
z = float(DISP_Z[frame_idx])
|
||||
vx = float(DISP_VX[frame_idx])
|
||||
vy = float(DISP_VY[frame_idx])
|
||||
vz = float(DISP_VZ[frame_idx])
|
||||
update_ball_info(frame_idx, x, y, z, vx, vy, vz)
|
||||
update_camera_info()
|
||||
|
||||
|
||||
# ===========================================================================
|
||||
# 事件绑定与启动
|
||||
# ===========================================================================
|
||||
timer = app.Timer(interval=0.02, connect=update, start=True)
|
||||
canvas.events.mouse_move.connect(handle_view_interaction)
|
||||
canvas.events.mouse_press.connect(handle_mouse_press)
|
||||
canvas.events.mouse_wheel.connect(handle_view_interaction)
|
||||
canvas.events.resize.connect(reposition_camera_info)
|
||||
canvas.events.key_press.connect(handle_key_press)
|
||||
|
||||
if hasattr(canvas, "native") and hasattr(canvas.native, "setFocus"):
|
||||
canvas.native.setFocus()
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
app.run()
|
||||
Reference in New Issue
Block a user