
customtkinter 原生不支持 gif 动画播放,需手动提取帧并配合 `after()` 实现循环刷新;本文提供可复用的 `giflabel` 自定义组件,支持自动加载、缩放与无缝播放。
在 CustomTkinter 中显示静态图片(如 PNG、JPEG)非常简单,直接使用 CTkImage + CTkLabel 即可。但若要播放 GIF 动画,则必须自行处理帧序列和定时刷新逻辑——因为 CTkImage 仅保存单帧图像对象,无法自动解析 GIF 的多帧时序信息。
以下是一个完整、健壮的解决方案:通过继承 customtkinter.CTkLabel 创建自定义 GIFLabel 类,它会:
- 自动读取 GIF 文件所有帧;
- 将每帧转换为适配尺寸的 CTkImage 对象;
- 利用 widget.after() 实现非阻塞的逐帧切换;
- 支持自定义播放延迟(默认采用 GIF 内置 duration);
- 自动适配标签宽高,避免拉伸失真。
import customtkinter as ctk
from PIL import Image
class GIFLabel(ctk.CTkLabel):
def __init__(self, master, image_path, **kwargs):
self._gif_image = Image.open(image_path)
# 自动设置标签尺寸为 GIF 原始尺寸(可被 kwargs 覆盖)
kwargs.setdefault("width", self._gif_image.width)
kwargs.setdefault("height", self._gif_image.height)
kwargs.setdefault("text", "") # 隐藏文字
# 提取播放延迟(毫秒),优先使用传入的 duration,否则取 GIF info
self._duration = kwargs.pop("duration", None) or self._gif_image.info.get("duration", 100)
super().__init__(master, **kwargs)
# 预加载所有帧为 CTkImage(注意:size 必须显式指定,否则缩放失效)
self._frames = []
for i in range(self._gif_image.n_frames):
self._gif_image.seek(i)
frame = self._gif_image.copy()
self._frames.append(
ctk.CTkImage(light_image=frame, dark_image=frame,
size=(int(self.cget("width")), int(self.cget("height"))))
)
# 启动动画循环
self._animate()
def _animate(self, idx=0):
self.configure(image=self._frames[idx])
self.after(self._duration, self._animate, (idx + 1) % len(self._frames))✅ 使用示例:
app = ctk.CTk()
app.title("GIF in CustomTkinter")
app.geometry("1300x750")
# 直接传入 GIF 路径,自动适配尺寸
gif_label = GIFLabel(app, "resources/images/background.gif")
gif_label.pack(pady=20)
# 如需缩放,可显式指定 width/height(保持宽高比建议用 PIL 先处理)
# gif_label = GIFLabel(app, "logo.gif", width=400, height=225, duration=80)
app.mainloop()⚠️ 注意事项:
- 性能提示:大尺寸 GIF(如 1282×720)帧数较多时,内存占用较高,建议提前用 PIL 缩放或裁剪;也可在 __init__ 中添加 self._gif_image = self._gif_image.resize((w, h), Image.LANCZOS) 优化。
- 深色模式兼容:本例中 light_image 与 dark_image 使用同一帧(GIF 无明暗区分),如需主题适配,应准备两套 GIF 或动态生成反色帧。
- MP4 播放说明:CustomTkinter 不支持视频解码;如需播放 MP4,必须借助外部库(如 opencv-python + PIL 提取帧,再按 GIF 方式逐帧渲染),或改用 tkinter 原生 VideoPlayer 第三方组件(非 CustomTkinter 生态)。
该方案已在 CustomTkinter v5.2+ 和 Pillow v10+ 中验证稳定运行,是目前最轻量、可嵌入、免依赖的 GIF 播放实践方式。










