Как я гифку с помощью ИИ сжимал

обложка

обложка

Вступление

Привет, Хабр! Я графический дизайнер. Занимаюсь созданием сайтов, иллюстраций, немного работаю с видео и в качестве хобби увлекаюсь 3D. Я никогда не считал себя программистом. Да, я умею читать код, понимаю его логику, но вот так, чтобы самостоятельно сесть и написать что‑то с нуля… до недавнего времени это казалось мне чем‑то запредельным.

Проблема

Редко, но прилетают задачи сделать гифку из видео для рекламных площадок или для чего‑нибудь ещё. В этот раз попросили сделать новогоднее видео (баннер). Я отрисовал подарки на снегу в Стилистике сайта, добавил немного свечения и снег (Новый год же всё‑таки). Получилось зацикленное видео 320×200 пикселей на 10 секунд весом в 5 мегабайт. Всем всё понравилось, видео согласовали, но теперь нужно сделать из него гифку до 300 килобайт. Окей. Иду на ezgif.com, и после примерно пяти попыток уменьшения настроек получаю гифку размером 1.6 мегабайта. И тут я замечаю, что в настройках метода сжатия используется FFMPEG. А он у меня установлен (уже и не помню, зачем изначально я его себе ставил).

интерфейс сайта ezgif.com

интерфейс сайта ezgif.com

Идея

И тут меня осенило:, а что если я у нейронки попрошу помочь с конвертацией MP4 > GIF, и тоже с помощью FFMPEG?

Тут нужно небольшое отступление. Я очень люблю автоматизировать рутинные процессы порой даже ценой больших усилий (когда сама автоматизация занимает больше времени, чем процесс, который я пытался автоматизировать). Но зато, когда процесс повторяется и у меня есть для этого готовое решение, я каждый раз радуюсь как ребенок.

Иду к нейронке с вопросом: «а мы можем MP4 конвертировать в GIF с помощью FFMPEG?» Она ответила положительно, чему я был несказанно рад. Запросил у неё все возможные настройки в FFMPEG, которые влияют на конечный размер гифки. Писать решил на Python, так как он у меня тоже установлен (нужен, чтобы запускать нейронки для генерации картинок локально на компе). Она довольно быстро накидала мне рабочий код (который работал в консоли). И тут, не очень‑то надеясь на положительный ответ, я спросил:, а можем ли мы сюда добавить интерфейс со всеми настройками и ползуночками (сделать полноценную программу)? И нейронка ответила положительно. От этого я офигел во второй раз.

Описание программы

В итоге у меня получился мой софт по запросу. И не столь важно, как он работает, хотя работает он идеально. Важнее то, что это решает мои задачи ну и то что, я сам его сделал! Интерфейс получился простым и понятным.
Складываем видео в папку, выбираем параметры сжатия (FPS, количество цветов, дизеринг, разрешение, пресет), нажимаем на кнопку и получаем гифки!

Вот такая красота получилась

Вот такая красота получилась

Технологии

Итак, мой помощник — это Gemini 2.0. До этого я сидел на платной подписке ChatGpt с момента основания, так как альтернатив особо не видел. Пробовал несколько раз Claude от Anthropic, но Chat подкупал своей экосистемой со своими приложениями под все устройства и потрясающим голосовым режимом. Но тут выходит Gemini 2.0 Flash Thinking Experimental. Не претендую на объективность своих тестов, но чисто по ощущениям я бы сравнил её с o1 от OpenAI, но бесплатная и безлимитная. Экосистема — это конечно, хорошо, но за 20$ в месяц можно и потерпеть работу в браузере. И вот уже месяц как я отказался от подписки чата и пользуюсь только Gemini.

Я просто задавал вопросы на русском простым языком, а она писала код. Это похоже на общение с очень умным другом, который хорошо погружён в твою тему и знает ответы на все возникающие в процессе работы технические вопросы. Я не лез в дебри программирования, я просто говорил: «Мне нужен код, который будет делать вот так». И она делала.

Разработка и первые шаги

Я задавал вопросы, получал ответы и, как правило, готовый рабочий код, запускал, и если что‑то шло не так — снова задавал вопросы, и так по кругу.

Самое интересное было то, что благодаря Gemini, я смог «подружиться» с FFMPEG. Это, если коротко, комбайн для работы с видео и аудио. Но в консоли, то есть графического интерфейса у него нет. А теперь у меня есть доступ ко всем его возможностям.

интерфейс gmini

интерфейс gmini

Раньше, чтобы сконвертировать какое‑то видео или уменьшить его размер, нужно было лезть в документацию, смотреть хелп, читать про те команды, которые можно и нужно использовать. А сейчас просто берёшь, просишь Gemini, и она тебе пишет команду, которая решает твой вопрос. И всё работает моментально, тебе не нужно ждать пока загрузится какой‑нибудь Adobe Media Encoder — FFMPEG часто видео конвертирует быстрее, чем Media Encoder открывается. То же самое касается и написания плагинов для Figma или экспрешенов и скриптов для After Effects.

Кстати, про плагины для фигмы, в комьюнити лежит файл с иконками от Paweł Kuna, у него почти 5000 потрясающих иконок и каждая проименованна например «corner‑up‑right», но когда мне нужно найти определенные иконки, то на поиск порой уходило слишком много времени, поэтому я написал плагин для фигмы, который название всех фреймов из этого файла сохранил в txt файл. Теперь, когда нужна очередная иконка, я закидываю в Gemini этот текстовый файл и прошу найти мне подходящие по смыслу иконки на нужные мне, например, пункты меню. И она находит, причём довольно точно находит, даже по ассоциациям. Тоже сильно сокращает время работы.

Извиняюсь, немного отвлёкся от GIF Конвертера.

Иконочки от Paweł Kuna

Иконочки от Paweł Kuna

Технические детали

За пару часов у меня получилось 600 строк кода. (Оставлю их все в конце текста) И это всё работает! Основные «фишки» моего конвертера: простой интерфейс, быстрая работа, пояснения ко всем настройкам, т.к. мне редко нужно пользоваться и через полгода я скорее всего забуду, зачем та или иная настройка, поэтому я попросил Gemini подписать каждую и оценка размера гифки — так я хоть примерно понимаю, что получится.

Видео не ускорено, на одну небольшую гифку действительно уходит меньше секунды, при пакетном преобразовании это очень сильно заметно.

Основные параметры:

FPS — это количество кадров в секунду. Чем больше кадров, тем плавнее гифка, но и размер будет больше.
MAX_COLORS — это количество цветов. Чем больше цветов, тем лучше качество, но опять же, размер будет расти.
DITHER — это такая штука, которая помогает сглаживать переходы между цветами.
SCALE — тут всё понятно, разрешение гифки.
PRESET — это настройка скорости конвертации, от очень быстрой до очень медленной (но и с более высоким качеством).

Результат и личные впечатления

На всё про всё у меня ушло где‑то 2–3 часа. В итоге у меня получилась программа, которая решает определенную задачу, и решает её так как мне нужно.

Будущее и новые горизонты

И самое главное — это только начало! Я вижу, в использовании нейросетей просто безграничные возможности. ИИ — это просто мощный инструмент, который может сделать нашу работу ещё более крутой. И, кстати, если у вас есть кейсы — как вам нейронки помогают в профессиональной деятельности, то будет очень интересно почитать.

Если вы хотите погрузиться в технические детали или просто попробовать запустить этот конвертор локально, то вот исходный код:

import os
import subprocess
from pathlib import Path
import json
import math
import tkinter as tk
from tkinter import ttk, messagebox
from PIL import Image, ImageTk

SETTINGS_FILE = "gif_settings.json"
INPUT_FOLDER = "input"

DEFAULT_SETTINGS = {
    "FPS": "10",
    "MAX_COLORS": "64",
    "DITHER": "floyd_steinberg",
    "SCALE": "316:-1",
    "PRESET": "medium"
}

DITHER_INFO = {
    "floyd_steinberg": "Баланс качества и размера, подходит для большинства случаев.",
    "bayer": "Структурированный дизеринг, может подойти для пиксельной графики.",
    "none": "Без дизеринга. Может уменьшить размер, но понизит плавность переходов.",
    "sierra2": "Альтернативный дизеринг, стоит попробовать, если другие не подходят.",
    "sierra2_4a": "Ещё один вариант экспериментального дизеринга."
}

PRESET_INFO = {
    "ultrafast": "Очень быстрая обработка, но может быть снижение качества.",
    "fast": "Быстрее среднего, чуть лучше качество чем ultrafast.",
    "medium": "Сбалансированный вариант по скорости и качеству.",
    "slow": "Медленнее, но может дать немного лучшее качество.",
    "veryslow": "Очень медленная обработка, чуть лучшее качество, но долго."
}

SCALE_HELP = """\
Введите разрешение в формате WxH.
Например:
- 320:240 для точного масштаба.
- 316:-1 чтобы ширина была 316, а высота подобралась автоматически для сохранения пропорций.
Если указать -1 для одной из сторон, мы вычислим итоговое разрешение, исходя из оригинальных пропорций видео.
"""

def load_settings():
    if os.path.exists(SETTINGS_FILE):
        with open(SETTINGS_FILE, "r", encoding="utf-8") as f:
            loaded = json.load(f)
            return {**DEFAULT_SETTINGS, **loaded}
    return DEFAULT_SETTINGS.copy()

def save_settings(settings):
    with open(SETTINGS_FILE, "w", encoding="utf-8") as f:
        json.dump(settings, f, ensure_ascii=False, indent=4)

def get_file_size(file_path):
    return round(file_path.stat().st_size / 1024, 2) if file_path.exists() else 0

def ffprobe_json(file_path):
    if not file_path.exists():
        return None
    cmd = ["ffprobe", "-v", "error", "-print_format", "json", "-show_format", "-show_streams", str(file_path)]
    proc = subprocess.run(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE, text=True)
    return json.loads(proc.stdout) if proc.returncode == 0 else None

def get_video_info(file_path):
    data = ffprobe_json(file_path)
    if not data:
        return None
    fmt = data.get("format", {})
    video_stream = next((s for s in data.get("streams", []) if s.get("codec_type") == "video"), None)
    if not video_stream:
        return None
    frame_rate_str = video_stream.get("avg_frame_rate", "0/0")
    frame_rate = float(frame_rate_str.split('/')[0]) / float(frame_rate_str.split('/')[1]) if frame_rate_str != "0/0" and frame_rate_str.split('/')[1] != '0' else 0.0
    return {
        "duration": float(fmt.get("duration", 0.0)),
        "width": video_stream.get("width", 0),
        "height": video_stream.get("height", 0),
        "codec": video_stream.get("codec_name", "unknown"),
        "frame_rate": frame_rate,
        "file_size_mb": float(fmt.get("size", 0.0)) / (1024*1024),
        "creation_date": fmt.get("tags", {}).get("creation_time", "Неизвестно")
    }

def parse_scale(scale_str):
    parts = scale_str.split(":")
    if len(parts) != 2:
        return -1, -1
    try:
        return int(parts[0]), int(parts[1])
    except ValueError:
        return -1, -1

class GifConverterApp:
    def __init__(self, master):
        self.master = master
        master.title("GIF Конвертер")
        self.settings = load_settings()
        self.output_folder = Path.cwd() / "converted_gifs"
        self.output_folder.mkdir(exist_ok=True)
        self.input_path = Path.cwd() / INPUT_FOLDER
        self.input_path.mkdir(exist_ok=True)
        self.files = self._load_files()
        self.selected_files = []
        self.current_duration = 0.0
        self.current_video_info = None
        self.last_selected_indices = ()
        self.preview_image = None

        self.style = ttk.Style(master)
        self.style.theme_use('clam')
        self.style.configure('TLabelFrame.Label', font=('Segoe UI', 10, 'bold'))
        self.style.configure('TButton', padding=5)
        self.style.configure('TCombobox', padding=5)
        self.style.configure('TScale', background='#f0f0f0')

        left_frame = ttk.Frame(master, padding=10)
        left_frame.pack(side=tk.LEFT, fill=tk.Y)

        self.refresh_button = ttk.Button(left_frame, text="Обновить", command=self._update_file_list)
        self.refresh_button.pack(anchor="w", pady=(0, 5))

        ttk.Label(left_frame, text="Список файлов:", font=('Segoe UI', 9)).pack(anchor="w")

        self.file_listbox = tk.Listbox(left_frame, height=10, selectmode=tk.EXTENDED, font=('Segoe UI', 9), borderwidth=1, relief="solid")
        self.file_listbox.pack(fill=tk.BOTH, expand=True)
        for f in self.files:
            self.file_listbox.insert(tk.END, f.name)
        self.file_listbox.bind("<>", self._on_file_select)

        self.preview_frame = ttk.Frame(left_frame, borderwidth=1, relief="solid", padding=5)
        self.preview_frame.pack(fill=tk.X, pady=5)
        ttk.Label(self.preview_frame, text="Предпросмотр:", font=('Segoe UI', 9)).pack(anchor="w")
        self.preview_label = ttk.Label(self.preview_frame)
        self.preview_label.pack()

        right_frame = ttk.Frame(master, padding=10)
        right_frame.pack(side=tk.RIGHT, fill=tk.BOTH, expand=True)

        settings_frame = ttk.LabelFrame(right_frame, text="Настройки конвертации", padding=10)
        settings_frame.pack(fill=tk.X, pady=10)

        ttk.Label(settings_frame, text="FPS (1-30):", font=('Segoe UI', 9)).grid(row=0, column=0, sticky="w", padx=5, pady=5)
        fps_frame = ttk.Frame(settings_frame)
        fps_frame.grid(row=0, column=1, padx=5, pady=5, sticky="ew")

        self.fps_entry = ttk.Spinbox(fps_frame, from_=1, to=30, width=5, command=self._on_fps_entry_change)
        self.fps_entry.insert(0, self.settings["FPS"])
        self.fps_entry.pack(side=tk.LEFT, padx=(0, 5))

        self.fps_slider = ttk.Scale(fps_frame, from_=1, to=30, orient="horizontal", command=self._on_fps_change)
        self.fps_slider.set(int(self.settings["FPS"]))
        self.fps_slider.pack(side=tk.LEFT, fill="x", expand=True)

        self.fps_info = ttk.Label(settings_frame, text="FPS: Количество кадров в секунду. Чем выше FPS, тем плавнее анимация, но больше размер файла.", wraplength=300, justify="left", anchor="nw", font=('Segoe UI', 9))
        self.fps_info.grid(row=1, column=0, columnspan=2, sticky="ew", padx=5, pady=5)

        ttk.Label(settings_frame, text="MAX COLORS (2-256):", font=('Segoe UI', 9)).grid(row=2, column=0, sticky="w", padx=5, pady=5)
        colors_frame = ttk.Frame(settings_frame)
        colors_frame.grid(row=2, column=1, padx=5, pady=5, sticky="ew")

        self.colors_entry = ttk.Spinbox(colors_frame, from_=2, to=256, width=5, command=self._on_colors_entry_change)
        self.colors_entry.insert(0, self.settings["MAX_COLORS"])
        self.colors_entry.pack(side=tk.LEFT, padx=(0, 5))

        self.colors_slider = ttk.Scale(colors_frame, from_=2, to=256, orient="horizontal", command=self._on_colors_change)
        self.colors_slider.set(int(self.settings["MAX_COLORS"]))
        self.colors_slider.pack(side=tk.LEFT, fill="x", expand=True)

        self.colors_info = ttk.Label(settings_frame, text="MAX COLORS: Чем больше цветов, тем лучше качество, но больше размер файла.", wraplength=300, justify="left", anchor="nw", font=('Segoe UI', 9))
        self.colors_info.grid(row=3, column=0, columnspan=2, sticky="ew", padx=5, pady=5)

        ttk.Label(settings_frame, text="DITHER:", font=('Segoe UI', 9)).grid(row=4, column=0, sticky="w", padx=5, pady=5)
        self.dither_dropdown = ttk.Combobox(settings_frame, values=list(DITHER_INFO.keys()))
        self.dither_dropdown.grid(row=4, column=1, padx=5, pady=5, sticky="ew")
        self.dither_dropdown.set(self.settings["DITHER"])
        self.dither_dropdown.bind("<>", self._on_dither_change)

        self.dither_info = ttk.Label(settings_frame, text=DITHER_INFO[self.settings["DITHER"]], wraplength=300, justify="left", anchor="nw", font=('Segoe UI', 9))
        self.dither_info.grid(row=5, column=0, columnspan=2, sticky="ew", padx=5, pady=5)

        ttk.Label(settings_frame, text="SCALE (WxH):", font=('Segoe UI', 9)).grid(row=6, column=0, sticky="w", padx=5, pady=5)
        scale_frame = ttk.Frame(settings_frame)
        scale_frame.grid(row=6, column=1, padx=5, pady=5, sticky="ew")

        self.scale_entry = ttk.Entry(scale_frame)
        self.scale_entry.insert(0, self.settings["SCALE"])
        self.scale_entry.pack(side=tk.LEFT, fill="x", expand=True, padx=(0, 5))
        self.scale_entry.bind("", self._preserve_selection)
        self.scale_entry.bind("", self._update_estimate)

        self.resolution_button = ttk.Button(scale_frame, text="Разрешение", command=self._apply_selected_resolution)
        self.resolution_button.pack(side=tk.LEFT, padx=(0, 5))

        help_btn = ttk.Button(scale_frame, text="?", command=self._show_scale_help)
        help_btn.pack(side=tk.LEFT)

        self.scale_info = ttk.Label(settings_frame, text="SCALE: Например, 316:-1 — ширина 316, высота будет рассчитана пропорционально.", wraplength=300, justify="left", anchor="nw", font=('Segoe UI', 9))
        self.scale_info.grid(row=7, column=0, columnspan=2, sticky="ew", padx=5, pady=5)

        ttk.Label(settings_frame, text="PRESET:", font=('Segoe UI', 9)).grid(row=8, column=0, sticky="w", padx=5, pady=5)
        self.preset_dropdown = ttk.Combobox(settings_frame, values=list(PRESET_INFO.keys()))
        self.preset_dropdown.grid(row=8, column=1, padx=5, pady=5, sticky="ew")
        self.preset_dropdown.set(self.settings["PRESET"])
        self.preset_dropdown.bind("<>", self._on_preset_change)

        self.preset_info_label = ttk.Label(settings_frame, text=PRESET_INFO[self.settings["PRESET"]], wraplength=300, justify="left", anchor="nw", font=('Segoe UI', 9))
        self.preset_info_label.grid(row=9, column=0, columnspan=2, sticky="ew", padx=5, pady=5)

        save_btn = ttk.Button(settings_frame, text="Сохранить настройки", command=self._save_current_settings)
        save_btn.grid(row=10, column=0, columnspan=2, padx=5, pady=5, sticky="ew")

        convert_btn = ttk.Button(right_frame, text="Конвертировать выбранные файлы", command=self._convert_selected_files)
        convert_btn.pack(pady=10, fill="x")

        self.results_text = tk.Text(right_frame, height=10, wrap="word", font=('Segoe UI', 9), borderwidth=1, relief="solid")
        self.results_text.pack(fill=tk.BOTH, expand=True, pady=5)
        self.results_text.config(state=tk.DISABLED)
        self.results_text.tag_configure("red", foreground="red")
        self.results_text.tag_configure("green", foreground="green")

        self.file_info_label = ttk.Label(right_frame, text="Нет выбранного файла", justify="left", anchor="nw", font=('Segoe UI', 9))
        self.file_info_label.pack(anchor="w", padx=5, pady=5)

        self.estimate_label = ttk.Label(right_frame, text="Примерная оценка размера GIF: недостаточно данных", wraplength=300, justify="left", anchor="nw", foreground="blue", font=('Segoe UI', 9, 'italic'))
        self.estimate_label.pack(anchor="w", padx=5, pady=5)

        self._update_estimate()

    def _load_files(self):
        return list(self.input_path.glob("*.mp4"))

    def _update_file_list(self):
        self._preserve_selection()
        self.file_listbox.delete(0, tk.END)
        self.files = self._load_files()
        for f in self.files:
            self.file_listbox.insert(tk.END, f.name)
        self._restore_selection()

    def _apply_selected_resolution(self):
        if self.current_video_info:
            resolution = f"{self.current_video_info['width']}:{self.current_video_info['height']}"
            self.scale_entry.delete(0, tk.END)
            self.scale_entry.insert(0, resolution)
            self._update_estimate()

    def _show_scale_help(self):
        messagebox.showinfo("Справка по SCALE", SCALE_HELP)

    def _on_file_select(self, event=None):
        selection = self.file_listbox.curselection()
        self.selected_files = [self.files[i] for i in selection]
        self.last_selected_indices = selection

        if self.selected_files:
            selected_file_path = self.selected_files[0]
            info = get_video_info(selected_file_path)
            if info:
                self.current_video_info = info
                self.current_duration = info["duration"]
                file_info_text = (f"Файл: {selected_file_path.name}\n"
                                  f"Разрешение: {info['width']}x{info['height']}\n"
                                  f"Продолжительность: {round(info['duration'],2)} сек\n"
                                  f"Размер: {round(info['file_size_mb'],2)} МБ\n"
                                  f"Кодек: {info['codec']}\n"
                                  f"Частота кадров: {round(info['frame_rate'],2)} к/с\n"
                                  f"Дата создания: {info['creation_date']}")
                self.file_info_label.config(text=file_info_text)
                self._generate_preview(selected_file_path)
                self.resolution_button.config(text=f"{info['width']}:{info['height']}")
            else:
                self._clear_preview()
                self.file_info_label.config(text="Не удалось получить информацию о файле")
                self.resolution_button.config(text="Разрешение")
        else:
            self._clear_preview()
            self.file_info_label.config(text="Нет выбранного файла")
            self.resolution_button.config(text="Разрешение")

        self._update_estimate()

    def _clear_preview(self):
        self.preview_label.config(image='')
        self.preview_image = None

    def _generate_preview(self, video_path):
        self._clear_preview()
        info = get_video_info(video_path)
        if not info or info['duration'] <= 0:
            return
        preview_time = info['duration'] / 2
        temp_filename = "preview.png"
        cmd = ["ffmpeg", "-i", str(video_path), "-ss", str(preview_time), "-vframes", "1", "-vf", "scale=200:-1", temp_filename]
        try:
            subprocess.run(cmd, check=True, capture_output=True)
            img = Image.open(temp_filename)
            self.preview_image = ImageTk.PhotoImage(img)
            self.preview_label.config(image=self.preview_image)
        except FileNotFoundError:
            messagebox.showerror("Ошибка", "FFmpeg не найден.")
        except subprocess.CalledProcessError as e:
            messagebox.showerror("Ошибка", f"Ошибка предпросмотра: {e.stderr.decode()}")
        except Exception as e:
            messagebox.showerror("Ошибка", f"Ошибка предпросмотра: {e}")
        finally:
            if os.path.exists(temp_filename):
                os.remove(temp_filename)

    def _on_fps_change(self, val):
        self.fps_entry.delete(0, tk.END)
        self.fps_entry.insert(0, str(int(float(val))))
        self._update_estimate()

    def _on_fps_entry_change(self):
        try:
            val = int(self.fps_entry.get())
            if 1 <= val <= 30:
                self.fps_slider.set(val)
                self._update_estimate()
        except ValueError:
            pass

    def _on_colors_change(self, val):
        self.colors_entry.delete(0, tk.END)
        self.colors_entry.insert(0, str(int(float(val))))
        self._update_estimate()

    def _on_colors_entry_change(self):
        try:
            val = int(self.colors_entry.get())
            if 2 <= val <= 256:
                self.colors_slider.set(val)
                self._update_estimate()
        except ValueError:
            pass

    def _on_dither_change(self, event=None):
        self.dither_info.config(text=DITHER_INFO[self.dither_dropdown.get()])
        self._update_estimate()

    def _on_preset_change(self, event=None):
        self.preset_info_label.config(text=PRESET_INFO[self.preset_dropdown.get()])
        self._update_estimate()

    def _save_current_settings(self):
        self.settings["FPS"] = str(int(float(self.fps_slider.get())))
        self.settings["MAX_COLORS"] = str(int(float(self.colors_slider.get())))
        self.settings["DITHER"] = self.dither_dropdown.get().strip()
        self.settings["SCALE"] = self.scale_entry.get().strip()
        self.settings["PRESET"] = self.preset_dropdown.get().strip()
        save_settings(self.settings)

    def _convert_selected_files(self):
        if not self.selected_files:
            self._append_result("Нет выбранных файлов для конвертации.", "red")
            return
        self._save_current_settings()
        fps = self.settings["FPS"]
        max_colors = self.settings["MAX_COLORS"]
        dither = self.settings["DITHER"]
        scale = self.settings["SCALE"]
        preset = self.settings["PRESET"]

        for input_file in self.selected_files:
            output_gif = self.output_folder / (input_file.stem + ".gif")
            palette_file = self.output_folder / (input_file.stem + "_palette.png")

            cmd_palette = ["ffmpeg", "-i", str(input_file), "-vf", f"fps={fps},scale={scale}:flags=lanczos,palettegen=max_colors={max_colors}", "-y", str(palette_file)]
            cmd_gif = ["ffmpeg", "-i", str(input_file), "-i", str(palette_file), "-lavfi", f"fps={fps},scale={scale}:flags=lanczos[x];[x][1:v]paletteuse=dither={dither}", "-preset", preset, "-y", str(output_gif)]

            proc_palette = subprocess.run(cmd_palette, stdout=subprocess.PIPE, stderr=subprocess.PIPE, text=True)
            if proc_palette.returncode != 0:
                self._append_result(f"{input_file.name}: Ошибка palettegen: {proc_palette.stderr}", "red")
                continue

            proc_gif = subprocess.run(cmd_gif, stdout=subprocess.PIPE, stderr=subprocess.PIPE, text=True)
            if proc_gif.returncode != 0:
                self._append_result(f"{input_file.name}: Ошибка конвертации в GIF: {proc_gif.stderr}", "red")
                continue

            try:
                palette_file.unlink()
            except FileNotFoundError:
                pass

            input_size = get_file_size(input_file)
            output_size = get_file_size(output_gif)
            reduction = f"{round(((output_size - input_size) / input_size) * 100, 2)}%" if input_size > 0 else "0%"
            color = "red" if output_size > input_size else "green"

            self._append_result(f"{input_file.name}: {input_size} KB → {output_size} KB ({reduction}).", color)

    def _append_result(self, text, color):
        self.results_text.config(state=tk.NORMAL)
        self.results_text.insert("1.0", text + "\n", (color,))
        self.results_text.config(state=tk.DISABLED)

    def _preserve_selection(self, event=None):
        self.last_selected_indices = self.file_listbox.curselection()

    def _restore_selection(self):
        if self.last_selected_indices:
            self.file_listbox.selection_clear(0, tk.END)
            for i in self.last_selected_indices:
                self.file_listbox.selection_set(i)

    def _update_estimate(self, event=None):
        try:
            fps = int(float(self.fps_slider.get()))
        except ValueError:
            fps = 10
        try:
            max_colors = int(float(self.colors_slider.get()))
        except ValueError:
            max_colors = 64

        scale_str = self.scale_entry.get().strip()
        w, h = parse_scale(scale_str)

        if not self.selected_files or not self.current_video_info or self.current_video_info['duration'] <= 0:
            self.estimate_label.config(text="Примерная оценка размера GIF: недостаточно данных", foreground="blue")
            return

        orig_w = self.current_video_info['width']
        orig_h = self.current_video_info['height']
        duration = self.current_video_info['duration']

        final_w, final_h = w, h
        if w == -1 and h == -1:
            self.estimate_label.config(text="Примерная оценка размера GIF: некорректный SCALE", foreground="blue")
            return
        elif w == -1:
            final_w = int(round((orig_w / orig_h) * h)) if orig_h != 0 else 0
        elif h == -1:
            final_h = int(round((orig_h / orig_w) * w)) if orig_w != 0 else 0

        if final_w <= 0 or final_h <= 0:
            self.estimate_label.config(text="Примерная оценка размера GIF: итоговое разрешение не рассчитано", foreground="blue")
            return

        frames = duration * fps
        bpp = math.log2(max(2, max_colors))
        pixels_per_frame = final_w * final_h
        size_approx_kb = (frames * pixels_per_frame * (bpp/8) * 0.2) / 1024
        self.estimate_label.config(text=f"Примерная оценка размера GIF: ~ {size_approx_kb:.2f} KB", foreground="blue")

if __name__ == "__main__":
    root = tk.Tk()
    app = GifConverterApp(root)
    root.lift()
    root.attributes("-topmost", True)
    root.after(0, root.attributes, "-topmost", False)
    root.mainloop()

© Habrahabr.ru