python批量为视频添加文字水印和图片水印的程序

如题,代码如下,可设置多个图片水印及它们的移动位置

功能为:可以添加多个动态移动的水印,还可以设置水印的大小以及移动速度,也可以增加文字水印,重点是这个是批量执行的,可以对目录下的所有视频文件批量添加水印

import cv2
import os
import numpy as np
from moviepy.editor import VideoFileClipdef add_dynamic_watermarks_with_audio(video_path, output_path, watermark_image_paths, watermark_sizes, watermark_speeds, add_text_watermark, text_watermark_text, text_watermark_color):# 读取视频video_clip = VideoFileClip(video_path)audio_clip = video_clip.audio# 获取视频的宽度和高度width, height = video_clip.size# 准备文本水印if add_text_watermark:# 创建用于绘制文本水印的画布text_watermark_canvas = np.zeros((height, width, 3), dtype=np.uint8)# 设置文本水印属性font_face = cv2.FONT_HERSHEY_SIMPLEXfont_scale = 1.5font_thickness = 2# 获取文本水印的大小(text_width, text_height), _ = cv2.getTextSize(text_watermark_text, font_face, font_scale, font_thickness)# 计算文本水印的位置text_x = int((width - text_width) / 2)text_y = int(height - text_height - 10)# 在画布上绘制文本水印cv2.putText(text_watermark_canvas, text_watermark_text, (text_x, text_y), font_face, font_scale, text_watermark_color, font_thickness, cv2.LINE_AA)# 在每一帧上添加水印watermark_positions = [(0, 0)] * len(watermark_image_paths)def process_frame(t, x):nonlocal watermark_positionsframe = video_clip.get_frame(t)# 在帧上添加图片水印for i in range(len(watermark_image_paths)):watermark_image_path = watermark_image_paths[i]watermark_size = watermark_sizes[i]watermark_speed = watermark_speeds[i]# 读取水印图片watermark_image = cv2.imread(watermark_image_path)# 调整水印图片大小watermark_height, watermark_width, _ = watermark_image.shapeif watermark_width > width or watermark_height > height:scale_factor = min(width / watermark_width, height / watermark_height)watermark_image = cv2.resize(watermark_image,(int(watermark_width * scale_factor), int(watermark_height * scale_factor)),interpolation=cv2.INTER_LINEAR,)# 更新水印位置if t % watermark_speed == 0:while True:x = np.random.randint(0, width - watermark_image.shape[1])y = np.random.randint(0, height - watermark_image.shape[0])too_close = False# 检查新位置与已选择水印位置之间的距离for j in range(i):dist = np.sqrt((x - watermark_positions[j][0]) ** 2 + (y - watermark_positions[j][1]) ** 2)if dist < watermark_image.shape[1] or dist < watermark_image.shape[0]:too_close = Truebreakif not too_close:watermark_positions[i] = (x, y)breakelse:x, y = watermark_positions[i]# 在帧上添加水印watermark_resized = cv2.resize(watermark_image, (int(watermark_size * watermark_width), int(watermark_size * watermark_height)))alpha = watermark_resized[:, :, 0] / 255.0for c in range(3):frame_copy = frame.copy()frame_copy[y : y + watermark_resized.shape[0], x : x + watermark_resized.shape[1], c] = (frame[y : y + watermark_resized.shape[0], x : x + watermark_resized.shape[1], c] * (1 - alpha)+ watermark_resized[:, :, c] * alpha)# 在帧上添加文本水印if add_text_watermark:frame_with_text = cv2.addWeighted(frame, 1, text_watermark_canvas, 0.7, 0)frame = frame_with_textcv2.imshow("Watermarked Video", frame)cv2.waitKey(1)return frame# 处理视频每一帧processed_clip = video_clip.fl(lambda gf, t: process_frame(t, gf))# 输出处理后的视频final_clip = processed_clip.set_audio(audio_clip)final_clip.write_videofile(output_path, codec='libx264', audio_codec="aac")return "水印添加完成!"def batch_add_watermarks_in_directory_with_audio(directory, output_directory, watermark_image_paths, watermark_sizes, watermark_speeds, add_text_watermark=False, text_watermark_text=None):# 获取目录中的所有文件file_list = os.listdir(directory)# 遍历文件列表for file_name in file_list:# 检查文件是否为视频文件if file_name.lower().endswith(('.avi', '.mp4', '.mov', '.mkv')):file_path = os.path.join(directory, file_name)# 设置输出文件路径output_file_path = os.path.join(output_directory, file_name)# 添加水印add_dynamic_watermarks_with_audio(file_path, output_file_path, watermark_image_paths, watermark_sizes, watermark_speeds, add_text_watermark, text_watermark_text if add_text_watermark else None, (255, 255, 255))return "批量处理完成!"# 使用示例
if __name__ == '__main__':directory = input("请输入视频文件所在目录路径:")  # 视频文件所在目录output_directory = input("请输入输出视频文件目录路径:")  # 输出视频文件目录# 检查并创建输出目录os.makedirs(output_directory, exist_ok=True)watermark_image_paths = []watermark_sizes = []watermark_speeds = []num_watermarks = int(input("请输入要添加的水印数量:"))for i in range(num_watermarks):watermark_image_path = input(f"请输入第{i+1}个水印图片的路径:")watermark_size = float(input(f"请输入第{i+1}个水印图片的大小(0~1之间):"))watermark_speed = int(input(f"请输入第{i+1}个水印图片的速度:"))watermark_image_paths.append(watermark_image_path)watermark_sizes.append(watermark_size)watermark_speeds.append(watermark_speed)add_text_watermark_input = input("是否添加文本水印?(y/n) ")if add_text_watermark_input.lower() == 'y':add_text_watermark = Truetext_watermark_text = input("请输入文本水印内容:")else:add_text_watermark = Falsetext_watermark_text = Nonebatch_add_watermarks_in_directory_with_audio(directory, output_directory, watermark_image_paths, watermark_sizes, watermark_speeds, add_text_watermark, text_watermark_text)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/145174.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringBoot实现IP地址归属地查询

SpringBoot实现IP地址归属地查询 功能特性 标准化的数据格式 每个 IP 数据段的 region 信息都固定了格式&#xff1a; 国家|区域|省份|城市|ISP&#xff0c;只有中国的数据绝大部分精确到了城市&#xff0c;其他国家部分数据只能定位到国家&#xff0c;后前的选项全部是 0。…

React父组件怎么调用子组件的方法

调用方法&#xff1a;1、类组件中的调用可以利用React.createRef()、ref的函数式声明或props自定义onRef属性来实现&#xff1b;2、函数组件、Hook组件中的调用可以利用useImperativeHandle或forwardRef抛出子组件ref来实现。 【程序员必备开发工具推荐】Apifox一款免费API管理…

微服务架构之路7,Nacos配置中心的Pull原理,附源码

目录 一、本地配置二、配置中心1、以Nacos为例&#xff1a;2、Pull模式3、也可以通过Nacos实现注册中心 三、配置中心提供了哪些功能四、如何操作配置中心1、配置注册2、配置反注册3、配置查看4、配置变更订阅 五、主流的微服务注册中心有哪些&#xff0c;如何选择&#xff1f;…

【开源】基于JAVA的智能停车场管理系统

项目编号&#xff1a; S 005 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S005&#xff0c;文末获取源码。} 项目编号&#xff1a;S005&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容A. 车主端功能B. 停车工作人员功能C. 系…

Kubernetes基础知识了解

一、Kubernetes简介 Kubernetes是一个轻便的和可扩展的开源平台&#xff0c;用于管理容器化应用和服务。通过Kubernetes能够进行应用的自动化部署和扩缩容。在Kubernetes中&#xff0c;会将组成应用的容器组合成一个逻辑单元以更易管理和发现。Kubernetes积累了作为Google生产…

asp.net实验管理系统VS开发sqlserver数据库web结构c#编程web网页设计

一、源码特点 asp.net 实验管理系统 是一套完善的web设计管理系统&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为vs2010&#xff0c;数据库为sqlserver2008&#xff0c;使用c#语言开发。 asp.net实验管理系统1 应用技术&am…

SELinux零知识学习十二、SELinux策略语言之客体类别和许可(6)

接前一篇文章&#xff1a;SELinux零知识学习十一、SELinux策略语言之客体类别和许可&#xff08;5&#xff09; 一、SELinux策略语言之客体类别和许可 3. 有效的客体类别 &#xff08;3&#xff09;System V IPC客体队列 与IPC有关的客体类别代表System V IPC资源。下表总结…

JavaWeb-JavaWeb中的I/O(输入/输出)

JavaWeb-JavaWeb中的I/O(输入/输出) 一、JavaWeb中的I/O(输入/输出)1.1 什么是I/O1.2 JAVA中关于I/O的类库二、磁盘的I/O2.1 磁盘I/O的工作机制2.2 磁盘的物理结构2.3 磁盘的IO过程三、Java实现访问磁盘文件四、JAVA的序列化与反序列化五、网络编程5.1 Java Socket的工作机…

在IDEA中的DeBug调试技巧

一、条件断点 循环中经常用到这个技巧&#xff0c;例如&#xff1a;遍历1个List的过程中&#xff0c;想让断点停在某个特定值。 参考上图&#xff0c;在断点的位置&#xff0c;右击断点旁边的红点&#xff0c;会出来1个界面&#xff0c;在Condition这里填写断点条件即可&#…

SSM项目初始化流程与操作概念解释-SpringBoot简化版

文章目录 1.引入概念2.导入依赖3.项目配置4.依照SpringMVC框架构建项目 1.引入概念 例如某一个XX系统&#xff0c;该系统存在前台页面&#xff08;给用户直观看或使用&#xff09;&#xff0c;和后台页面&#xff08;给管理人员调整数据和权限&#xff09;。 这二个页面都通过…

【算法】复习搜索与图论

&#x1f34e; 博客主页&#xff1a;&#x1f319;披星戴月的贾维斯 &#x1f34e; 欢迎关注&#xff1a;&#x1f44d;点赞&#x1f343;收藏&#x1f525;留言 &#x1f347;系列专栏&#xff1a;&#x1f319; 蓝桥杯 &#x1f319;请不要相信胜利就像山坡上的蒲公英一样唾手…

C++二分查找算法:规划兼职工作

题目 你打算利用空闲时间来做兼职工作赚些零花钱。 这里有 n 份兼职工作&#xff0c;每份工作预计从 startTime[i] 开始到 endTime[i] 结束&#xff0c;报酬为 profit[i]。 给你一份兼职工作表&#xff0c;包含开始时间 startTime&#xff0c;结束时间 endTime 和预计报酬 pro…

Java集合List报错,java.lang.UnsupportedOperationException

目录 一、点击Arrays.asList源码&#xff0c;一探究竟二、习惯了Arrays.asList&#xff0c;就是想用.add()添加元素&#xff0c;怎么办&#xff1f;三、又有一个同事&#xff0c;是这样写的四、重新点击Arrays.asList源码&#xff0c;一探究竟五、全是坑&#xff0c;怎么办&…

Nginx反向代理与负载均衡与504错误

Nginx反向代理与负载均衡概念简介 关于代理 什么是代理 类似中介 在没有代理模式的情况下&#xff0c;客户端和Nginx服务端&#xff0c;都是客户端直接请求服务端&#xff0c;服务端直接响应客户端。 那么在互联网请求里面&#xff0c;客户端往往无法直接向服务端发起请求…

“技能兴鲁”职业技能大赛-网络安全赛项-学生组初赛 WP

Crypto BabyRSA 共模攻击 题目附件&#xff1a; from gmpy2 import * from Crypto.Util.number import *flag flag{I\m not gonna tell you the FLAG} # 这个肯定不是FLAG了&#xff0c;不要交这个咯p getPrime(2048) q getPrime(2048) m1 bytes_to_long(bytes(flag.e…

Unity中Shader矩阵的乘法

文章目录 前言一、矩阵乘以标量二、矩阵和矩阵相乘1、第一个矩阵的列数必须 与 第二个矩阵的行数相等&#xff0c;否则无法相乘&#xff01;2、相乘的结果矩阵&#xff0c;行数由第一个矩阵的行数决定&#xff0c;列数由第二个矩阵的列数决定&#xff01; 三、单位矩阵四、矩阵…

Docker push的 http 413问题处理

1. 环境&#xff1a; 1) 内网docker服务器&#xff1a;192.168.123.1&#xff0c;操作系统为&#xff1a;redhat 7.9&#xff0c;运行了registry&#xff0c;端口18088 2) 代理服务器(可通外网)&#xff1a;192.168.110.2&#xff0c;操作系统为&#xff1a;redhat 7.92.docke…

VIVADO+FPGA调试记录

vivadoFPGA调试记录 vitis编译vivado导出的硬件平台&#xff0c;提示xxxx.h file cant findVITIS内定义的头文件找不到 vitis编译vivado导出的硬件平台&#xff0c;提示’xxxx.h file cant find’ 此硬件平台中&#xff0c;包含有AXI接口类型的ip。在vitis编译硬件平台时&…

Linux_Docker修改Docker Root Dir

今天遇到需求&#xff0c;要修改一下docker容器和镜像的存储位置&#xff0c;默认位置为/var/lib/docker目录下&#xff0c;要修改到/new/dockerFile目录下。 停止docker服务 sudo service docker stop 备份docker容器镜像 移动/var/lib/docker目录下的文件到/dockerFile目录…

IntelliJ IDE 插件开发 |(一)快速入门

前言 IntelliJ IDEA 作为 Java 开发的首选 IDE&#xff0c;其强大、方便之处不必多说。不过&#xff0c;由于个人或者团队的个性化需求&#xff0c;我们或多或少会想对其功能进行拓展&#xff0c;这时就需要开发插件&#xff08;在 IntelliJ 平台下的所有 IDE 均可运行&#x…