吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 11687|回复: 402
上一主题 下一主题
收起左侧

[原创工具] 视频批量横转竖

    [复制链接]
跳转到指定楼层
楼主
是谁的大海 发表于 2024-6-12 20:10 回帖奖励
本帖最后由 是谁的大海 于 2024-6-14 14:18 编辑

简介
视频处理器 v1.3 是一款由是貔貅呀开发的视频编辑和处理工具,提供高效便捷的视频批量横转竖,主要功能:
  • 导入与删除文件:轻松导入多个视频文件,删除不必要的文件。
  • 暂停与继续处理:随时暂停和继续处理。
  • 设置输出文件夹:选择视频处理后的存放文件夹。
  • FFmpeg 配置:自定义输出格式、视频编码器、音频编码器和线程数。
  • 日志记录:实时记录处理日志。
使用步骤
  • 导入文件
  • 设置输出文件夹
  • 配置FFmpeg
  • 开始处理文件
  • 查看处理进度
  • 暂停与继续
特点
  • 用户友好界面
  • 多格式支持
  • 高效多线程处理
  • 实时日志显示
  • 运行截图

  • 完成视频

修复功能:
  • 修复全选删除后程序出问题
  • 修复了初始4条线程改为初始1条
增加功能:
  • 新增直接拖拽视频到列表功能
  • 新增高斯模糊功能
  • 新增菜单栏(单纯忘了
  • 新增作者名(单纯忘了
待上线新功能
  • FFmpeg配置将进行修改,感觉挺鸡肋
  • 批量竖转横(只能增加黑边或者高斯模糊,如要完整视频则只能拉长视频)
  • 批量修改文件名(看需求量多不多
  • 图片批量修改大小,无ai辅助只是单纯放大或者缩小因为本人接触不深,(看需求量多不多

代码:
[Asm] 纯文本查看 复制代码
import os
import ttkbootstrap as ttk
from ttkbootstrap.constants import *
from tkinter import filedialog, messagebox, END, Text, StringVar, IntVar, BooleanVar, Menu
from concurrent.futures import ThreadPoolExecutor, as_completed
import subprocess
import threading
import psutil
import re
import sys

from tkinterdnd2 import TkinterDnD, DND_FILES

def resource_path(relative_path):
    """ Get absolute path to resource, works for dev and for PyInstaller """
    try:
        # PyInstaller creates a temp folder and stores path in _MEIPASS
        base_path = sys._MEIPASS
    except Exception:
        base_path = os.path.abspath(".")

    return os.path.join(base_path, relative_path)

class VideoProcessor:
    def __init__(self, master):
        self.master = master
        self.master.title("视频处理器  吾爱作者:是谁的大海(是貔貅呀) 版本:1.3")
        
        self.input_files = []
        self.output_folder = ""
        self.process_thread = None
        self.pause_event = threading.Event()
        self.pause_event.set()  # Start in the unpaused state
        self.ffmpeg_processes = []  # List to keep track of all ffmpeg processes
        self.is_closing = False
        
        self.output_format = StringVar(value="mp4")
        self.video_codec = StringVar(value="libx264")
        self.audio_codec = StringVar(value="aac")
        self.thread_count = IntVar(value=1)  # Default to 1 threads
        self.apply_blur = BooleanVar(value=False)  # Boolean to check if blur should be applied
        
        self.create_widgets()
        self.create_menu()
        self.master.protocol("WM_DELETE_WINDOW", self.on_closing)
        
    def create_widgets(self):
        frame = ttk.Frame(self.master, padding=10)
        frame.pack(fill=BOTH, expand=YES)

        self.file_list_frame = ttk.Frame(frame)
        self.file_list_frame.pack(fill=BOTH, expand=YES, pady=5)

        columns = ('序号', '文件夹名字', '进度')
        self.file_tree = ttk.Treeview(self.file_list_frame, columns=columns, show='headings')
        self.file_tree.heading('序号', text='序号')
        self.file_tree.heading('文件夹名字', text='文件夹名字')
        self.file_tree.heading('进度', text='进度')
        
        self.file_tree.column('序号', width=100, anchor='center')
        self.file_tree.column('文件夹名字', width=400, anchor='w')
        self.file_tree.column('进度', width=100, anchor='center')
        
        self.file_tree.pack(side=LEFT, fill=BOTH, expand=YES)
        
        scrollbar = ttk.Scrollbar(self.file_list_frame, orient="vertical", command=self.file_tree.yview)
        self.file_tree.configure(yscrollcommand=scrollbar.set)
        scrollbar.pack(side=RIGHT, fill=Y)

        self.log_text = Text(frame, height=5, state='disabled')
        self.log_text.pack(fill=BOTH, expand=YES, pady=5)

        button_frame = ttk.Frame(frame)
        button_frame.pack(fill=BOTH, expand=YES)

        self.add_files_button = ttk.Button(button_frame, text="导入文件", command=self.add_files, bootstyle=PRIMARY)
        self.add_files_button.pack(side=LEFT, padx=5, pady=5)
        
        self.remove_files_button = ttk.Button(button_frame, text="删除文件", command=self.remove_files, bootstyle=DANGER)
        self.remove_files_button.pack(side=LEFT, padx=5, pady=5)
        
        self.pause_button = ttk.Button(button_frame, text="暂停/继续", command=self.toggle_pause, bootstyle=WARNING)
        self.pause_button.pack(side=LEFT, padx=5, pady=5)
        
        self.open_output_folder_button = ttk.Button(button_frame, text="打开文件", command=self.open_output_folder, bootstyle=SUCCESS)
        self.open_output_folder_button.pack(side=LEFT, padx=5, pady=5)
        
        self.set_output_folder_button = ttk.Button(button_frame, text="导出文件", command=self.set_output_folder, bootstyle=SUCCESS)
        self.set_output_folder_button.pack(side=LEFT, padx=5, pady=5)
        
        self.process_button = ttk.Button(button_frame, text="开始处理文件", command=self.start_processing, bootstyle=INFO)
        self.process_button.pack(side=RIGHT, padx=5, pady=5)
        
        config_frame = ttk.LabelFrame(frame, text="FFmpeg 配置")
        config_frame.pack(fill=BOTH, expand=YES, pady=5)
        
        ttk.Label(config_frame, text="输出格式:").pack(side=LEFT, padx=5, pady=5)
        ttk.OptionMenu(config_frame, self.output_format, "mp4", "mp4", "avi", "mov").pack(side=LEFT, padx=5, pady=5)
        
        ttk.Label(config_frame, text="视频编码器:").pack(side=LEFT, padx=5, pady=5)
        ttk.OptionMenu(config_frame, self.video_codec, "libx264", "libx264", "libx265", "mpeg4").pack(side=LEFT, padx=5, pady=5)
        
        ttk.Label(config_frame, text="音频编码器:").pack(side=LEFT, padx=5, pady=5)
        ttk.OptionMenu(config_frame, self.audio_codec, "aac", "aac", "mp3", "ac3").pack(side=LEFT, padx=5, pady=5)
        
        ttk.Label(config_frame, text="线程数:").pack(side=LEFT, padx=5, pady=5)
        ttk.Entry(config_frame, textvariable=self.thread_count).pack(side=LEFT, padx=5, pady=5)

        self.blur_checkbox = ttk.Checkbutton(config_frame, text="应用高斯模糊效果", variable=self.apply_blur)
        self.blur_checkbox.pack(side=LEFT, padx=5, pady=5)

        # Set up drag and drop
        self.master.drop_target_register(DND_FILES)
        self.master.dnd_bind('<<Drop>>', self.drop_files)
    
    def create_menu(self):
        menu_bar = Menu(self.master)
        self.master.config(menu=menu_bar)

        help_menu = Menu(menu_bar, tearoff=0)
        menu_bar.add_cascade(label="帮助", menu=help_menu)
        
        help_menu.add_command(label="使用说明", command=self.show_usage_instructions)
        help_menu.add_command(label="软件具体说明", command=self.show_software_details)
    
    def show_usage_instructions(self):
        instructions = (
            "使用说明:\n"
            "1. 导入文件:点击“导入文件”按钮,选择需要处理的视频文件,或将视频文件拖拽到软件窗口中。\n"
            "2. 设置输出文件夹:点击“导出文件”按钮,选择一个文件夹作为输出文件夹。\n"
            "3. 配置FFmpeg参数:在“FFmpeg 配置”区域,选择输出格式、视频编码器、音频编码器、线程数,并可选择是否应用高斯模糊效果。\n"
            "4. 开始处理:点击“开始处理文件”按钮,开始批量处理视频文件。处理过程中可以查看处理进度和日志信息。\n"
            "5. 查看输出文件:点击“打开文件”按钮,打开输出文件夹查看处理完成的视频文件。\n"
            "6. 删除文件:选择文件列表中的文件,点击“删除文件”按钮删除不需要处理的文件。\n"
        )
        messagebox.showinfo("使用说明", instructions)
    
    def show_software_details(self):
        details = (
            "仅供学习,切勿使用到其他用途\n"
            "1. 输出格式:支持MP4、AVI和MOV等常见格式,用户可自定义选择。\n"
            "2. 视频压缩:默认使用libx264视频编码器和aac音频编码器,支持高效视频压缩,用户可自定义选择其他编码器。\n"
            "3. 视频裁剪:适用于将1920x1080横屏视频裁剪成9:16竖屏视频,不会变形。\n"
            "4. 高斯模糊:可选应用高斯模糊效果,适用于特殊视频效果需求。\n"
            "5. 多线程处理:支持多线程并发处理,用户可自定义线程数,提高处理效率。\n"

        )
        messagebox.showinfo("软件具体说明", details)
        
    def drop_files(self, event):
        files = self.master.tk.splitlist(event.data)
        for file in files:
            if file not in self.input_files and file.lower().endswith(('.mp4', '.avi', '.mov')):
                self.input_files.append(file)
                self.file_tree.insert('', END, values=(len(self.input_files), os.path.basename(file), "未处理"))
                self.log(f"导入文件: {file}")
            else:
                messagebox.showwarning("警告", f"文件已存在或不支持的文件类型: {os.path.basename(file)}")
    
    def add_files(self):
        files = filedialog.askopenfilenames(title="选择视频文件", filetypes=[("视频文件", "*.mp4 *.avi *.mov")])
        for file in files:
            if file not in self.input_files:
                self.input_files.append(file)
                self.file_tree.insert('', END, values=(len(self.input_files), os.path.basename(file), "未处理"))
                self.log(f"导入文件: {file}")
            else:
                messagebox.showwarning("警告", f"文件已存在: {os.path.basename(file)}")
        
    def remove_files(self):
        selected_items = self.file_tree.selection()
        indices_to_remove = []
        
        for item in selected_items:
            values = self.file_tree.item(item, 'values')
            if values:
                index = int(values[0]) - 1
                indices_to_remove.append(index)
                self.file_tree.delete(item)
        
        # 删除索引列表中的元素(倒序删除避免索引问题)
        for index in sorted(indices_to_remove, reverse=True):
            del self.input_files[index]
        
        self.log("删除选中文件")
        self.refresh_file_list()
    
    def refresh_file_list(self):
        for item in self.file_tree.get_children():
            self.file_tree.delete(item)
        for index, file in enumerate(self.input_files):
            self.file_tree.insert('', END, values=(index + 1, os.path.basename(file), "未处理"))

    def set_output_folder(self):
        self.output_folder = filedialog.askdirectory(title="选择输出文件夹")
        self.log(f"设置输出文件夹: {self.output_folder}")
        
    def start_processing(self):
        if not self.input_files or not self.output_folder:
            messagebox.showerror("错误", "请添加文件并设置输出文件夹。")
            return
        
        self.process_thread = threading.Thread(target=self.process_videos_concurrently)
        self.process_thread.start()
        
    def toggle_pause(self):
        if self.pause_event.is_set():
            self.pause_event.clear()
            self.log("处理暂停")
            for process in self.ffmpeg_processes:
                proc = psutil.Process(process.pid)
                proc.suspend()
        else:
            self.pause_event.set()
            self.log("处理继续")
            for process in self.ffmpeg_processes:
                proc = psutil.Process(process.pid)
                proc.resume()
    
    def open_output_folder(self):
        if self.output_folder:
            os.startfile(self.output_folder)
            self.log(f"打开输出文件夹: {self.output_folder}")
        else:
            messagebox.showerror("错误", "请先设置输出文件夹。")
    
    def log(self, message):
        if not self.is_closing:
            self.master.after(0, self._log, message)
    
    def _log(self, message):
        if not self.is_closing:
            self.log_text.configure(state='normal')
            self.log_text.insert(END, message + '\n')
            self.log_text.configure(state='disabled')
            self.log_text.yview(END)
    
    def update_tree_status(self, index, status):
        if not self.is_closing:
            self.master.after(0, self._update_tree_status, index, status)
    
    def _update_tree_status(self, index, status):
        if not self.is_closing:
            self.file_tree.item(self.file_tree.get_children()[index], values=(index + 1, os.path.basename(self.input_files[index]), status))
    
    def process_videos_concurrently(self):
        max_workers = self.thread_count.get()
        with ThreadPoolExecutor(max_workers=max_workers) as executor:
            futures = [executor.submit(self.process_video, index, input_file) for index, input_file in enumerate(self.input_files)]
            for future in as_completed(futures):
                future.result()

    def process_video(self, index, input_file):
        ffmpeg_path = resource_path(os.path.join("ffmpeg_folder", "ffmpeg"))
        filename = os.path.basename(input_file)
        output_file = os.path.join(self.output_folder, f"processed_{filename}.{self.output_format.get()}")

        if os.path.exists(output_file):
            overwrite = messagebox.askyesno("文件已存在", f"{output_file} 已存在,是否覆盖?")
            if not overwrite:
                self.update_tree_status(index, "跳过")
                return

        if self.apply_blur.get():
            cmd = [
                ffmpeg_path,
                "-y",  # 自动覆盖输出文件
                "-i", input_file,
                "-vf", "split[a][b];[a]scale=1080:1920,boxblur=10:5[1];[b]scale=1080:ih*1080/iw[2];[1][2]overlay=0:(H-h)/2",
                "-c:v", self.video_codec.get(),
                "-crf", "18",
                "-preset", "veryfast",
                "-aspect", "9:16",
                "-c:a", self.audio_codec.get(),
                output_file
            ]
        else:
            cmd = [
                ffmpeg_path,
                "-y",  # 自动覆盖输出文件
                "-i", input_file,
                "-vf", "scale='if(gt(iw/ih,9/16),1080,-2)':'if(gt(iw/ih,9/16),-2,1920)',pad=1080:1920:(1080-iw)/2:(1920-ih)/2",
                "-c:v", self.video_codec.get(),
                "-crf", "18",
                "-preset", "veryfast",
                "-c:a", self.audio_codec.get(),
                output_file
            ]

        self.log(f"开始处理: {filename}")
        self.update_tree_status(index, "处理中")

        try:
            startupinfo = subprocess.STARTUPINFO()
            startupinfo.dwFlags |= subprocess.STARTF_USESHOWWINDOW

            process = subprocess.Popen(cmd, stderr=subprocess.PIPE, universal_newlines=True, encoding='utf-8', startupinfo=startupinfo)
            self.ffmpeg_processes.append(process)

            for line in process.stderr:
                if self.is_closing:
                    break
                progress = self.parse_progress(line)
                if progress:
                    self.update_tree_status(index, progress)

            process.wait()
        except Exception as e:
            self.log(f"处理文件时出错: {filename} - {str(e)}")
            self.update_tree_status(index, "处理失败")
            return

        if self.is_closing:
            self.update_tree_status(index, "未完成")
        else:
            self.log(f"完成处理: {filename}")
            self.update_tree_status(index, "已完成")
            self.ffmpeg_processes.remove(process)
    
    def parse_progress(self, line):
        match = re.search(r'time=(\d+:\d+:\d+\.\d+)', line)
        if match:
            return f"进度: {match.group(1)}"
        return None
    
    def on_closing(self):
        self.is_closing = True
        for process in self.ffmpeg_processes:
            proc = psutil.Process(process.pid)
            proc.terminate()
        self.master.destroy()

if __name__ == "__main__":
    root = TkinterDnD.Tk()
    root.title("视频处理器")
    root.geometry("870x520")  # Set the window size to 870x520
    root.resizable(False, False)  # Make the window non-resizable
    app = VideoProcessor(master=root)
    root.mainloop()



https://wwl.lanzn.com/b03c97wtuj
密码:9oq8

如有错误反馈一下,我好修

免费评分

参与人数 128吾爱币 +124 热心值 +115 收起 理由
catoo1 + 1 鼓励转贴优秀软件安全工具和文档!
viconly + 1 + 1 谢谢@Thanks!
面面春风 + 1 + 1 谢谢@Thanks!
苍白之白 + 1 + 1 谢谢@Thanks!
Abing888 + 1 谢谢@Thanks!
yuanshixuan + 1 谢谢@Thanks!
砺心缘起 + 1 + 1 我很赞同!
VanYoung + 1 谢谢@Thanks!
wycbs + 1 + 1 我很赞同!
之之呦 + 1 + 1 我很赞同!
wangyongdesign + 1 + 1 谢谢@Thanks!
maiwens + 1 + 1 谢谢@Thanks!
boliuhesi + 1 + 1 热心回复!
SSF3002 + 1 用心讨论,共获提升!
xiaomr1990 + 1 + 1 谢谢@Thanks!
lkqscqaz20 + 1 + 1 谢谢@Thanks!
gyrryg + 1 + 1 谢谢@Thanks!
ly147258 + 1 + 1 谢谢@Thanks!
xxpj + 1 + 1 谢谢@Thanks!
zzp8974 + 1 + 1 谢谢@Thanks!
龙蟒钛业 + 1 + 1 这个很实用
zpzwz + 1 + 1 谢谢@Thanks!
yyxqf + 1 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
skydust018 + 1 + 1 我很赞同!
juanzi123 + 1 + 1 谢谢@Thanks!
0angel0 + 1 + 1 谢谢@Thanks!
wyytxml6 + 2 + 1 我很赞同!
jnzj4811 + 1 + 1 谢谢@Thanks!
cs2555 + 1 + 1 鼓励转贴优秀软件安全工具和文档!
niroso + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
canty胖胖 + 1 + 1 用心讨论,共获提升!
pxj1983 + 1 + 1 谢谢@Thanks!
chj01 + 1 + 1 用心讨论,共获提升!
ljm57 + 1 + 1 用心讨论,共获提升!
Tiniaual + 1 + 1 我很赞同!
card628 + 1 热心回复!
红黑之橙 + 1 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
5Ekko + 1 + 1 我很赞同!
rayx + 1 + 1 我很赞同!
wari01 + 1 + 1 欢迎分析讨论交流,吾爱破解论坛有你更精彩!
Tori97 + 1 热心回复!
HaiWang2019 + 1 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
天空的板砖fly + 1 + 1 用心讨论,共获提升!
zmllxh + 1 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
zhangdexi + 1 + 1 用心讨论,共获提升!
gym_168 + 1 + 1 谢谢@Thanks!
dachui666 + 1 + 1 谢谢@Thanks!
wpx977 + 1 + 1 热心回复!
sunlei658 + 1 + 1 谢谢@Thanks!
tianjianggouxia + 1 + 1 我很赞同!
hubohang + 1 我很赞同!
Kanye + 1 我很赞同!
liuyuanliu + 1 + 1 我很赞同!
xjd2000 + 1 + 1 用心讨论,共获提升!
lyslxx + 1 + 1 我很赞同!
wenye + 1 我很赞同!
awen131972 + 1 + 1 谢谢@Thanks!
smallchop + 1 + 1 谢谢@Thanks!
tiantianxiaoxue + 1 + 1 热心回复!
iTMZhang + 1 用心讨论,共获提升!
燃病软 + 1 + 1 谢谢@Thanks!
qingfeng1214 + 1 + 1 热心回复!
a85401234 + 1 + 1 谢谢@Thanks!
JackyChen + 1 + 1 我很赞同!
aabbcc123123 + 1 + 1 谢谢@Thanks!
fusonni + 1 + 1 热心回复!
kyokusanagi2000 + 1 谢谢@Thanks!
LS888 + 1 热心回复!
xyftcgr + 1 + 1 热心回复!
开心熊猫741 + 1 + 1 我很赞同!
aventadorchan + 1 + 1 我很赞同!
buddingwolf + 1 + 1 谢谢@Thanks!
tfl1 + 1 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
七年后有你 + 1 我很赞同!
ljhttxs123 + 1 + 1 我很赞同!
NicholasN4 + 1 看把孩子急的 分给你了
xljh888 + 1 + 1 谢谢@Thanks!
精妹 + 1 我很赞同!
papapo + 1 + 1 我很赞同!
moonrabbit + 1 + 1 谢谢@Thanks!
silenter6speake + 1 热心回复!
Zhaofeiyan + 1 没有必要搞的这么大
前笨笨 + 1 + 1 谢谢@Thanks!
763742228 + 1 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
橙子不洒脱 + 1 + 1 谢谢@Thanks!
limit496 + 1 + 1 谢谢@Thanks!
wczsdwczsd + 1 我很赞同!
rset + 1 + 1 我很赞同!
tocabd + 1 + 1 谢谢@Thanks!
追逐飞翔 + 1 + 1 谢谢@Thanks!
後天 + 1 + 1 热心回复!
confiant + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
pdcba + 1 + 1 谢谢@Thanks!
YYL7535 + 1 + 1 谢谢@Thanks!
zixiu520 + 1 谢谢@Thanks!
wangmin + 1 谢谢@Thanks!
qsj521521 + 1 + 1 谢谢@Thanks!
bao1024 + 1 + 1 鼓励转贴优秀软件安全工具和文档!
xhb6263 + 1 + 1 如果有横竖都能相互转换的就更好了
外星文明 + 1 谢谢@Thanks!

查看全部评分

本帖被以下淘专辑推荐:

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

来自 #
 楼主| 是谁的大海 发表于 2024-6-15 11:11 |楼主
我搞了个对应蒙版合并的工具,适合营销号合并视频用
就是上下规格可能需要计算一下,不然中间会留一些空白
如果需要我还可以搞一个配音进去跟方便了

免费评分

参与人数 5吾爱币 +5 热心值 +3 收起 理由
weiyixuetang + 1 + 1 这个有了吗?
meissen + 1 + 1 我很赞同!
hs10301 + 1 + 1 我很赞同!
MmxHand + 1 我很赞同!
mogaga + 1 这个功能相当实用,建议加蒙版后,等比缩放自动填满

查看全部评分

来自 #
 楼主| 是谁的大海 发表于 2024-6-20 12:41 |楼主
是谁的大海 发表于 2024-6-15 11:11
我搞了个对应蒙版合并的工具,适合营销号合并视频用
就是上下规格可能需要计算一下,不然中间会留一些空白 ...

新增了预览功能方便调整蒙版图层的大小
推荐
 楼主| 是谁的大海 发表于 2024-6-13 10:23 |楼主
如果需要其他功能,可以反馈,我有时候把可以做的做出来

免费评分

参与人数 2吾爱币 +5 热心值 +1 收起 理由
Zhaofeiyan + 2 我很赞同!
缺牙巴 + 3 + 1 能增加个竖转横吗

查看全部评分

推荐
sunboyzh 发表于 2024-6-12 22:11
谢谢分享 有没有竖转横的?
推荐
 楼主| 是谁的大海 发表于 2024-6-13 09:43 |楼主
ws001980 发表于 2024-6-13 09:33
功能很实用谢谢分享 有没有竖转横的?

这只能两边补黑边吧,那样看着感觉好奇怪,要是竖的视频还是黑边的更奇怪了
推荐
yctx2050 发表于 2024-6-12 23:31
我也想要批量竖转横的,一个个手动转太麻烦了
推荐
Boom018 发表于 2024-6-12 22:10
竖着的不好看不习惯
4#
dx163 发表于 2024-6-12 21:58
这个自动取中间的画面么?
5#
yan999 发表于 2024-6-12 22:17
感谢分享!!!
6#
cta019177 发表于 2024-6-12 22:18
能做个竖转横的吗?
7#
liphee 发表于 2024-6-12 22:19
这个功能很实用
8#
柒點 发表于 2024-6-12 22:32
后期党狂喜
9#
Tortrix 发表于 2024-6-12 22:42
这是用Python写的呀
10#
wwd2018 发表于 2024-6-12 22:44
一直都想一个这样的东西,可以满足我了谢谢
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2025-1-15 16:42

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表