以下是一个使用Python的tkinter和OpenCV库实现打开指定IP摄像头的应用程序。这个程序允许用户输入IP摄像头的URL,并实时显示摄像头画面,同时支持截图和录制功能。
import tkinter as tk
from tkinter import ttk, messagebox, filedialog
import cv2
from PIL import Image, ImageTk
import threading
import time
import os
from datetime import datetime
class IPCameraViewer:
def __init__(self, root):
self.root = root
self.root.title("IP摄像头查看器")
self.root.geometry("900x600")
self.root.resizable(True, True)
# 设置字体确保中文显示正常
self.font = ('SimHei', 10)
self.title_font = ('SimHei', 14, 'bold')
# 摄像头相关变量
self.cap = None
self.ip_url = tk.StringVar(value="rtsp://admin:password@192.168.1.100:554/stream1")
self.is_playing = False
self.is_recording = False
self.frame = None
self.recorder = None
# 创建UI
self.create_widgets()
def create_widgets(self):
"""创建用户界面"""
# 标题区域
title_frame = ttk.Frame(self.root, padding="10")
title_frame.pack(fill=tk.X)
title_label = ttk.Label(
title_frame,
text="IP摄像头查看器",
font=self.title_font
)
title_label.pack(pady=10)
# 连接设置区域
connect_frame = ttk.LabelFrame(self.root, text="摄像头设置", padding="10")
connect_frame.pack(fill=tk.X, padx=10, pady=5)
ttk.Label(connect_frame, text="摄像头URL:", font=self.font).pack(side=tk.LEFT, padx=5)
url_entry = ttk.Entry(
connect_frame,
textvariable=self.ip_url,
font=self.font,
width=50
)
url_entry.pack(side=tk.LEFT, padx=5)
self.connect_button = ttk.Button(
connect_frame,
text="连接",
command=self.toggle_connection,
style="Accent.TButton"
)
self.connect_button.pack(side=tk.LEFT, padx=5)
# 状态区域
status_frame = ttk.Frame(self.root, padding="5")
status_frame.pack(fill=tk.X)
self.status_var = tk.StringVar()
self.status_var.set("未连接")
status_label = ttk.Label(
status_frame,
textvariable=self.status_var,
font=self.font,
foreground="red"
)
status_label.pack(side=tk.LEFT, padx=5)
# 视频显示区域
self.display_frame = ttk.LabelFrame(self.root, text="视频预览", padding="10")
self.display_frame.pack(fill=tk.BOTH, expand=True, padx=10, pady=5)
# 创建画布用于显示视频
self.canvas_width = 800
self.canvas_height = 450
self.canvas = tk.Canvas(
self.display_frame,
width=self.canvas_width,
height=self.canvas_height,
bg="black"
)
self.canvas.pack(fill=tk.BOTH, expand=True)
# 控制按钮区域
control_frame = ttk.Frame(self.root, padding="10")
control_frame.pack(fill=tk.X)
self.screenshot_button = ttk.Button(
control_frame,
text="截图",
command=self.take_screenshot,
state=tk.DISABLED
)
self.screenshot_button.pack(side=tk.LEFT, padx=5)
self.record_button = ttk.Button(
control_frame,
text="开始录制",
command=self.toggle_recording,
state=tk.DISABLED
)
self.record_button.pack(side=tk.LEFT, padx=5)
# 自定义按钮样式
style = ttk.Style()
style.configure('Accent.TButton', font=self.font)
def toggle_connection(self):
"""切换摄像头连接状态"""
if self.is_playing:
self.stop_camera()
else:
self.start_camera()
def start_camera(self):
"""启动摄像头"""
try:
self.status_var.set("正在连接...")
self.connect_button.config(text="断开连接")
self.screenshot_button.config(state=tk.NORMAL)
self.record_button.config(state=tk.NORMAL)
# 打开摄像头
self.cap = cv2.VideoCapture(self.ip_url.get())
if not self.cap.isOpened():
raise Exception("无法连接到摄像头,请检查URL和网络连接")
# 获取视频帧尺寸
self.frame_width = int(self.cap.get(3))
self.frame_height = int(self.cap.get(4))
# 启动视频播放线程
self.is_playing = True
self.status_var.set("已连接")
threading.Thread(target=self.update_frame, daemon=True).start()
except Exception as e:
messagebox.showerror("错误", f"连接摄像头时出错: {str(e)}")
self.status_var.set("连接失败")
self.connect_button.config(text="连接")
self.screenshot_button.config(state=tk.DISABLED)
self.record_button.config(state=tk.DISABLED)
def stop_camera(self):
"""停止摄像头"""
self.is_playing = False
self.is_recording = False
if self.cap:
self.cap.release()
self.cap = None
if self.recorder:
self.recorder.release()
self.recorder = None
self.connect_button.config(text="连接")
self.status_var.set("未连接")
self.record_button.config(text="开始录制")
self.screenshot_button.config(state=tk.DISABLED)
self.record_button.config(state=tk.DISABLED)
# 清空画布
self.canvas.delete("all")
self.canvas.create_text(
self.canvas_width/2,
self.canvas_height/2,
text="摄像头已断开",
fill="white",
font=self.font
)
def update_frame(self):
"""更新视频帧"""
while self.is_playing:
ret, frame = self.cap.read()
if not ret:
self.status_var.set("无法获取视频帧")
break
# 保存当前帧用于截图
self.frame = frame
# 调整帧大小以适应画布
frame = cv2.resize(frame, (self.canvas_width, self.canvas_height))
# 转换为RGB并显示
rgb_frame = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)
pil_image = Image.fromarray(rgb_frame)
photo = ImageTk.PhotoImage(image=pil_image)
# 在画布上显示图像
self.canvas.create_image(0, 0, image=photo, anchor=tk.NW)
self.canvas.photo = photo
# 录制视频
if self.is_recording:
self.recorder.write(frame)
# 控制帧率
time.sleep(0.03) # 约30FPS
def toggle_recording(self):
"""切换录制状态"""
if self.is_recording:
self.is_recording = False
self.record_button.config(text="开始录制")
self.status_var.set("录制已停止")
if self.recorder:
self.recorder.release()
self.recorder = None
else:
if not self.frame is None:
# 创建保存目录
if not os.path.exists("recordings"):
os.makedirs("recordings")
# 设置输出文件名和编码器
timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
output_file = f"recordings/camera_recording_{timestamp}.mp4"
# 定义编码器并创建VideoWriter对象
fourcc = cv2.VideoWriter_fourcc(*'mp4v')
self.recorder = cv2.VideoWriter(
output_file,
fourcc,
20.0, # FPS
(self.frame_width, self.frame_height)
)
self.is_recording = True
self.record_button.config(text="停止录制")
self.status_var.set("正在录制...")
else:
messagebox.showinfo("提示", "没有可用的视频帧")
def take_screenshot(self):
"""拍摄截图"""
if not self.frame is None:
# 创建保存目录
if not os.path.exists("screenshots"):
os.makedirs("screenshots")
# 设置输出文件名
timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
output_file = f"screenshots/camera_screenshot_{timestamp}.png"
# 保存截图
cv2.imwrite(output_file, self.frame)
self.status_var.set(f"截图已保存至: {output_file}")
# 显示保存成功提示
messagebox.showinfo("成功", f"截图已保存至:\n{output_file}")
else:
messagebox.showinfo("提示", "没有可用的视频帧")
if __name__ == "__main__":
root = tk.Tk()
app = IPCameraViewer(root)
root.mainloop()
- 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.
这个应用程序具有以下功能:
- 基本功能:
- 连接/断开IP摄像头
- 实时显示摄像头画面
- 截图并保存到本地
- 录制视频并保存到本地
- 技术要点:
- 使用OpenCV库处理视频流
- 多线程处理避免界面卡顿
- 自动适应不同分辨率的摄像头
- 保存文件时自动添加时间戳
- 使用方法:
- 输入IP摄像头的URL(例如:rtsp://admin:password@192.168.1.100:554/stream1)
- 点击"连接"按钮开始显示视频
- 点击"截图"按钮保存当前画面
- 点击"开始录制"按钮开始录制视频,再次点击停止
- 注意事项:
- 需要安装OpenCV库:
pip install opencv-python
- 不同摄像头的URL格式可能不同,请参考摄像头说明书
- 截图和录制文件会保存在程序同级目录的screenshots和recordings文件夹中
你可以根据需要修改代码中的默认URL或添加更多功能,如调整视频质量、添加滤镜效果等。