Compare commits

...

15 Commits

Author SHA1 Message Date
zsyg
d4afc5a5e7 修改版本号 2025-07-19 20:17:29 +08:00
zsyg
a445bbdc2d 提供avif格式转换 2025-07-19 20:13:18 +08:00
zsyg
0e90195f9a 提供更多文本转换模式 2025-07-19 20:13:01 +08:00
zsyg
11df47ff2e 改为cmake构建 2025-07-17 12:03:29 +08:00
zsyg
7d87d9be7e 添加更多格式 2025-07-14 14:05:13 +08:00
zsyg
5530c6413d 修改版本号 2025-07-14 14:01:31 +08:00
zsyg
b79db2e6bd 添加更多格式 2025-07-14 13:59:37 +08:00
zsyg
6056c984c0 改为静态编译 2025-07-14 11:10:15 +08:00
zsyg
2395c13b60 改为静态编译 2025-07-14 11:09:42 +08:00
zsyg
6624cee1a3 添加软件图标 2025-07-11 16:10:24 +08:00
zsyg
165703575b 修改版本号 2025-07-11 16:08:47 +08:00
zsyg
efc1498a25 添加应用卡片 2025-07-11 16:07:43 +08:00
zsyg
481fc24735 修改版本号 2025-07-11 08:17:57 +08:00
zsyg
bebd804b7a 美化ui 2025-07-11 08:17:26 +08:00
zsyg
4202013265 添加依赖 2025-07-11 08:16:11 +08:00
89 changed files with 2682 additions and 127 deletions

View File

@@ -51,7 +51,7 @@ namespace AppStore
// 初始化并添加应用信息 // 初始化并添加应用信息
infoLabel = new Label(); infoLabel = new Label();
infoLabel.Text = "kortapp-z\n版本: 1.2.9\n作者: zs-yg\n一个简单、开源的应用商店\nkortapp-z是完全免费\n基于.NET8和C/C++的软件"; infoLabel.Text = "kortapp-z\n版本: 1.3.3\n作者: zs-yg\n一个简单、开源的应用商店\nkortapp-z是完全免费\n基于.NET8和C/C++的软件";
infoLabel.Font = new Font("Microsoft YaHei", 12); infoLabel.Font = new Font("Microsoft YaHei", 12);
infoLabel.AutoSize = false; infoLabel.AutoSize = false;
infoLabel.Width = 300; infoLabel.Width = 300;

View File

@@ -35,6 +35,7 @@
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<PackageReference Include="SunnyUI" Version="3.8.6" />
<PackageReference Include="ZXing.Net" Version="0.16.9" /> <PackageReference Include="ZXing.Net" Version="0.16.9" />
</ItemGroup> </ItemGroup>

View File

@@ -8,14 +8,43 @@ using System.Text;
using System.Text.Json; using System.Text.Json;
using System.Diagnostics; using System.Diagnostics;
using AppStore; using AppStore;
using Sunny.UI;
using System.Runtime.InteropServices;
namespace AppStore namespace AppStore
{ {
/// <summary> public class MainForm : UIForm
/// 主窗体类,负责应用程序的主界面显示和交互
/// </summary>
public class MainForm : Form
{ {
[DllImport("Gdi32.dll", EntryPoint = "CreateRoundRectRgn")]
private static extern IntPtr CreateRoundRectRgn(
int nLeftRect,
int nTopRect,
int nRightRect,
int nBottomRect,
int nWidthEllipse,
int nHeightEllipse
);
[DllImport("dwmapi.dll")]
private static extern int DwmExtendFrameIntoClientArea(IntPtr hWnd, ref MARGINS pMarInset);
[DllImport("dwmapi.dll")]
private static extern int DwmSetWindowAttribute(IntPtr hwnd, int attr, ref int attrValue, int attrSize);
[DllImport("dwmapi.dll")]
private static extern int DwmIsCompositionEnabled(ref int pfEnabled);
private struct MARGINS
{
public int leftWidth;
public int rightWidth;
public int topHeight;
public int bottomHeight;
}
private const int DWMWA_WINDOW_CORNER_PREFERENCE = 33;
private const int DWMWCP_ROUND = 2;
private static readonly string CacheDir = Path.Combine( private static readonly string CacheDir = Path.Combine(
Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData),
"zsyg", "kortapp-z", ".cache"); "zsyg", "kortapp-z", ".cache");
@@ -139,10 +168,32 @@ namespace AppStore
// 设置窗体基本属性 // 设置窗体基本属性
// 窗体基本设置 // 窗体基本设置
this.Text = "kortapp-z"; this.Text = "kortapp-z";
this.Size = new Size(1430, 1050); // 增加窗体高度 this.Size = new Size(1430, 1050);
this.MinimumSize = new Size(600, 600); // 设置最小尺寸 this.MinimumSize = new Size(600, 600);
this.StartPosition = FormStartPosition.CenterScreen; this.StartPosition = FormStartPosition.CenterScreen;
this.Icon = new Icon("img/ico/icon.ico"); // 设置窗体图标 this.Icon = new Icon("img/ico/icon.ico");
this.Style = UIStyle.Custom;
this.FormBorderStyle = FormBorderStyle.None;
// 应用现代化圆角
this.Region = Region.FromHrgn(CreateRoundRectRgn(0, 0, Width, Height,
ThemeManager.FormRadius, ThemeManager.FormRadius));
// 启用窗口阴影
if (Environment.OSVersion.Version.Major >= 6)
{
int val = DWMWCP_ROUND;
DwmSetWindowAttribute(this.Handle, DWMWA_WINDOW_CORNER_PREFERENCE, ref val, sizeof(int));
MARGINS margins = new MARGINS()
{
leftWidth = 1,
rightWidth = 1,
topHeight = 1,
bottomHeight = 1
};
DwmExtendFrameIntoClientArea(this.Handle, ref margins);
}
// 初始化系统托盘 // 初始化系统托盘
trayMenu = new ContextMenuStrip(); trayMenu = new ContextMenuStrip();
@@ -190,13 +241,69 @@ namespace AppStore
// 现代化顶部导航栏 // 现代化顶部导航栏
Panel buttonPanel = new Panel(); Panel buttonPanel = new Panel();
buttonPanel.Dock = DockStyle.Top; buttonPanel.Dock = DockStyle.Top;
buttonPanel.Height = 70; buttonPanel.Height = 80;
buttonPanel.BackColor = ThemeManager.ControlBackgroundColor; buttonPanel.BackColor = ThemeManager.ControlBackgroundColor;
buttonPanel.Padding = new Padding(10, 15, 10, 0); buttonPanel.Padding = new Padding(15, 20, 15, 5);
buttonPanel.AutoScroll = true; buttonPanel.AutoScroll = true;
buttonPanel.AutoSize = true; buttonPanel.AutoSize = true;
buttonPanel.AutoSizeMode = AutoSizeMode.GrowAndShrink; buttonPanel.AutoSizeMode = AutoSizeMode.GrowAndShrink;
// 添加支持作者按钮
UIButton btnSupport = new UIButton();
btnSupport.Text = "支持作者";
btnSupport.Font = new Font("微软雅黑", 10F, FontStyle.Bold);
btnSupport.Size = new Size(120, 45);
btnSupport.Location = new Point(buttonPanel.Width - 140, 15);
btnSupport.Anchor = AnchorStyles.Top | AnchorStyles.Right;
btnSupport.Style = UIStyle.Custom;
btnSupport.FillColor = ThemeManager.AccentColor;
btnSupport.ForeColor = Color.White;
btnSupport.Click += (s, e) => {
// 使用SunnyUI的现代化MessageBox
var form = new UIForm();
form.Style = UIStyle.Custom;
form.Text = "支持作者";
form.Size = new Size(400, 200);
form.StartPosition = FormStartPosition.CenterParent;
var label = new Label();
label.Text = "您确定要前往GitHub支持作者吗";
label.Font = new Font("Microsoft YaHei", 10);
label.AutoSize = false;
label.Size = new Size(300, 40);
label.TextAlign = ContentAlignment.MiddleCenter;
label.Location = new Point(50, 40);
form.Controls.Add(label);
var btnOK = new UIButton();
btnOK.Text = "确定";
btnOK.Style = UIStyle.Custom;
btnOK.Size = new Size(80, 30);
btnOK.Location = new Point(120, 100);
btnOK.Click += (s, e) => {
form.DialogResult = DialogResult.OK;
form.Close();
};
form.Controls.Add(btnOK);
var btnCancel = new UIButton();
btnCancel.Text = "取消";
btnCancel.Style = UIStyle.Custom;
btnCancel.Size = new Size(80, 30);
btnCancel.Location = new Point(220, 100);
btnCancel.Click += (s, e) => {
form.DialogResult = DialogResult.Cancel;
form.Close();
};
form.Controls.Add(btnCancel);
if (form.ShowDialog(this) == DialogResult.OK)
{
Process.Start(new ProcessStartInfo("https://github.com/zs-yg/kortapp-z") { UseShellExecute = true });
}
};
buttonPanel.Controls.Add(btnSupport);
// 导航按钮样式 // 导航按钮样式
Action<Button> styleButton = (Button btn) => { Action<Button> styleButton = (Button btn) => {
btn.FlatStyle = FlatStyle.Flat; btn.FlatStyle = FlatStyle.Flat;
@@ -1144,6 +1251,31 @@ namespace AppStore
"https://ghproxy.net/https://github.com/vladelaina/Catime/releases/download/v1.1.1/catime_1.1.1.exe", "https://ghproxy.net/https://github.com/vladelaina/Catime/releases/download/v1.1.1/catime_1.1.1.exe",
"img/png/catime_resize.png")); "img/png/catime_resize.png"));
flowPanel.Controls.Add(CreateAppCard(
"Cataclysm-DDA",
"https://ghproxy.cn/https://github.com/CleverRaven/Cataclysm-DDA/releases/download/0.H-RELEASE/cdda-windows-with-graphics-and-sounds-x64-2024-11-23-1857.zip",
"img/png/Cataclysm-DDA.png"));
flowPanel.Controls.Add(CreateAppCard(
"gophish",
"https://ghproxy.cn/https://github.com/gophish/gophish/releases/download/v0.12.1/gophish-v0.12.1-windows-64bit.zip",
"img/png/gophish.png"));
flowPanel.Controls.Add(CreateAppCard(
"NoteGen",
"https://ghproxy.cn/https://github.com/codexu/note-gen/releases/download/note-gen-v0.19.3/NoteGen_0.19.3_x64-setup.exe",
"img/png/NoteGen.png"));
flowPanel.Controls.Add(CreateAppCard(
"hashcat",
"https://ghproxy.cn/https://github.com/hashcat/hashcat/releases/download/v6.2.6/hashcat-6.2.6.7z",
"img/png/hashcat.png"));
flowPanel.Controls.Add(CreateAppCard(
"rpg-cli",
"https://gh-proxy.com/https://github.com/facundoolano/rpg-cli/releases/download/1.2.0/rpg-cli-1.2.0-windows.exe",
""));
flowPanel.Controls.Add(CreateAppCard( flowPanel.Controls.Add(CreateAppCard(
"fluxy", "fluxy",
"https://ghproxy.net/https://github.com/alley-rs/fluxy/releases/download/v0.1.17/fluxy_0.1.17_x64-setup.exe", "https://ghproxy.net/https://github.com/alley-rs/fluxy/releases/download/v0.1.17/fluxy_0.1.17_x64-setup.exe",

View File

@@ -58,22 +58,26 @@ namespace AppStore
} }
// 浅色主题颜色 // 浅色主题颜色
private static readonly Color LightBackground = Color.FromArgb(255, 255, 255); private static readonly Color LightBackground = Color.FromArgb(250, 250, 250);
private static readonly Color LightControlBackground = Color.FromArgb(240, 240, 240); private static readonly Color LightControlBackground = Color.FromArgb(245, 245, 245);
private static readonly Color LightText = Color.FromArgb(30, 30, 30); private static readonly Color LightText = Color.FromArgb(40, 40, 40);
private static readonly Color LightButtonHover = Color.FromArgb(230, 230, 230); private static readonly Color LightButtonHover = Color.FromArgb(235, 235, 235);
private static readonly Color LightButtonActive = Color.FromArgb(220, 220, 220); private static readonly Color LightButtonActive = Color.FromArgb(225, 225, 225);
private static readonly Color LightAccent = Color.FromArgb(0, 120, 215);
private static readonly Color LightAccentLight = Color.FromArgb(0, 150, 245);
// 深色主题颜色 // 深色主题颜色
private static readonly Color DarkBackground = Color.FromArgb(30, 30, 30); private static readonly Color DarkBackground = Color.FromArgb(25, 25, 25);
private static readonly Color DarkControlBackground = Color.FromArgb(45, 45, 45); private static readonly Color DarkControlBackground = Color.FromArgb(40, 40, 40);
private static readonly Color DarkText = Color.FromArgb(240, 240, 240); private static readonly Color DarkText = Color.FromArgb(245, 245, 245);
private static readonly Color DarkButtonHover = Color.FromArgb(60, 60, 60); private static readonly Color DarkButtonHover = Color.FromArgb(55, 55, 55);
private static readonly Color DarkButtonActive = Color.FromArgb(70, 70, 70); private static readonly Color DarkButtonActive = Color.FromArgb(65, 65, 65);
private static readonly Color DarkBorder = Color.FromArgb(80, 80, 80); private static readonly Color DarkBorder = Color.FromArgb(70, 70, 70);
private static readonly Color DarkAccent = Color.FromArgb(0, 150, 245);
private static readonly Color DarkAccentLight = Color.FromArgb(0, 180, 255);
// 浅色主题边框颜色 // 浅色主题边框颜色
private static readonly Color LightBorder = Color.FromArgb(180, 180, 180); private static readonly Color LightBorder = Color.FromArgb(200, 200, 200);
public static event Action<ThemeMode> ThemeChanged = delegate {}; public static event Action<ThemeMode> ThemeChanged = delegate {};
@@ -109,6 +113,16 @@ namespace AppStore
public static Color BorderColor => public static Color BorderColor =>
_currentTheme == ThemeMode.Light ? LightBorder : DarkBorder; _currentTheme == ThemeMode.Light ? LightBorder : DarkBorder;
public static Color AccentColor =>
_currentTheme == ThemeMode.Light ? LightAccent : DarkAccent;
public static Color AccentLightColor =>
_currentTheme == ThemeMode.Light ? LightAccentLight : DarkAccentLight;
public static int ControlRadius => 8;
public static int FormRadius => 12;
public static void ApplyTheme(Control control) public static void ApplyTheme(Control control)
{ {
ApplyThemeToControl(control); ApplyThemeToControl(control);
@@ -125,6 +139,9 @@ namespace AppStore
button.FlatAppearance.BorderSize = 0; button.FlatAppearance.BorderSize = 0;
button.FlatAppearance.MouseOverBackColor = ButtonHoverColor; button.FlatAppearance.MouseOverBackColor = ButtonHoverColor;
button.FlatAppearance.MouseDownBackColor = ButtonActiveColor; button.FlatAppearance.MouseDownBackColor = ButtonActiveColor;
button.BackColor = ControlBackgroundColor;
button.Font = new Font(button.Font, FontStyle.Bold);
button.Padding = new Padding(10, 5, 10, 5);
} }
foreach (Control childControl in control.Controls) foreach (Control childControl in control.Controls)

BIN
img/png/Cataclysm-DDA.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 40 KiB

BIN
img/png/NoteGen.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 17 KiB

BIN
img/png/gophish.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

BIN
img/png/hashcat.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

View File

@@ -2,7 +2,7 @@
; 有关创建 Inno Setup 脚本文件的详细信息,请参阅帮助文档! ; 有关创建 Inno Setup 脚本文件的详细信息,请参阅帮助文档!
#define MyAppName "kortapp-z" #define MyAppName "kortapp-z"
#define MyAppVersion "1.2.9" #define MyAppVersion "1.3.3"
#define MyAppPublisher "zsyg" #define MyAppPublisher "zsyg"
#define MyAppURL "https://github.com/zs-yg/kortapp-z" #define MyAppURL "https://github.com/zs-yg/kortapp-z"
#define MyAppExeName "kortapp-z.exe" #define MyAppExeName "kortapp-z.exe"

View File

@@ -2,7 +2,7 @@
; 有关创建 Inno Setup 脚本文件的详细信息,请参阅帮助文档! ; 有关创建 Inno Setup 脚本文件的详细信息,请参阅帮助文档!
#define MyAppName "kortapp-z" #define MyAppName "kortapp-z"
#define MyAppVersion "1.2.9" #define MyAppVersion "1.3.3"
#define MyAppPublisher "zsyg" #define MyAppPublisher "zsyg"
#define MyAppURL "https://github.com/zs-yg/kortapp-z" #define MyAppURL "https://github.com/zs-yg/kortapp-z"
#define MyAppExeName "kortapp-z.exe" #define MyAppExeName "kortapp-z.exe"

View File

@@ -0,0 +1,77 @@
cmake_minimum_required(VERSION 3.10)
project(ImageFormatConverter)
# 设置静态编译
set(CMAKE_EXE_LINKER_FLAGS "-static")
set(BUILD_SHARED_LIBS OFF)
set(CMAKE_FIND_LIBRARY_SUFFIXES .a)
# 设置C++标准
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
# 设置MSYS2库搜索路径
set(MSYS2_LIB_DIR "C:/msys64/ucrt64/lib")
# 查找依赖库(静态版本)
find_package(PkgConfig REQUIRED)
pkg_check_modules(TIFF REQUIRED libtiff-4)
find_library(TIFF_STATIC_LIB NAMES libtiff.a PATHS ${TIFF_LIBRARY_DIRS})
pkg_check_modules(WEBP REQUIRED libwebp)
find_library(WEBP_STATIC_LIB NAMES libwebp.a PATHS ${WEBP_LIBRARY_DIRS})
pkg_check_modules(AVIF REQUIRED libavif)
find_library(AVIF_STATIC_LIB NAMES libavif.a PATHS ${AVIF_LIBRARY_DIRS})
# 使用find_package查找FLTK(静态版本)
find_package(FLTK REQUIRED)
find_library(FLTK_STATIC_LIB NAMES libfltk.a PATHS ${FLTK_LIBRARY_DIRS})
# 查找其他依赖库
find_library(JPEG_STATIC_LIB NAMES libjpeg.a PATHS ${MSYS2_LIB_DIR})
find_library(ZLIB_STATIC_LIB NAMES libz.a PATHS ${MSYS2_LIB_DIR})
find_library(ZSTD_STATIC_LIB NAMES libzstd.a PATHS ${MSYS2_LIB_DIR})
find_library(SHARPYUV_STATIC_LIB NAMES libsharpyuv.a PATHS ${MSYS2_LIB_DIR})
find_library(JBIG_STATIC_LIB NAMES libjbig.a PATHS ${MSYS2_LIB_DIR})
find_library(LZMA_STATIC_LIB NAMES liblzma.a PATHS ${MSYS2_LIB_DIR})
find_library(LIBDEFLATE_STATIC_LIB NAMES libdeflate.a PATHS ${MSYS2_LIB_DIR})
find_library(LERC_STATIC_LIB NAMES liblerc.a PATHS ${MSYS2_LIB_DIR})
find_library(AOM_STATIC_LIB NAMES libaom.a PATHS ${MSYS2_LIB_DIR})
find_library(YUV_STATIC_LIB NAMES libyuv.a PATHS ${MSYS2_LIB_DIR})
find_library(SVTAV1_ENC_STATIC_LIB NAMES libSvtAv1Enc.a PATHS ${MSYS2_LIB_DIR})
find_library(RAV1E_STATIC_LIB NAMES librav1e.a PATHS ${MSYS2_LIB_DIR})
find_library(DAV1D_STATIC_LIB NAMES libdav1d.a PATHS ${MSYS2_LIB_DIR})
# 包含头文件目录
include_directories(include ${TIFF_INCLUDE_DIRS} ${WEBP_INCLUDE_DIRS} ${FLTK_INCLUDE_DIRS} ${AVIF_INCLUDE_DIRS})
# 收集所有源文件
file(GLOB SOURCES "src/*.cpp")
# 创建可执行文件(设置为WIN32应用程序避免控制台窗口)
add_executable(ImageFormatConverter WIN32 ${SOURCES})
# 链接静态库
target_link_libraries(ImageFormatConverter
${TIFF_STATIC_LIB}
${WEBP_STATIC_LIB}
${FLTK_STATIC_LIB}
${JPEG_STATIC_LIB}
${ZLIB_STATIC_LIB}
${ZSTD_STATIC_LIB}
${SHARPYUV_STATIC_LIB}
${JBIG_STATIC_LIB}
${LZMA_STATIC_LIB}
${LIBDEFLATE_STATIC_LIB}
${LERC_STATIC_LIB}
${AVIF_STATIC_LIB}
${AOM_STATIC_LIB}
${YUV_STATIC_LIB}
${SVTAV1_ENC_STATIC_LIB}
${RAV1E_STATIC_LIB}
${DAV1D_STATIC_LIB}
-lstdc++ -lgcc -lwinpthread -lcomctl32 -lole32 -luuid -lws2_32 -lntdll)
# 设置输出目录
set(EXECUTABLE_OUTPUT_PATH ${CMAKE_BINARY_DIR}/bin)

View File

@@ -1,25 +0,0 @@
CXX = g++
CXXFLAGS = -I"C:/msys64/ucrt64/include" -I"./include" -std=c++17 -Wall -mwindows
LDFLAGS = -L"C:/msys64/ucrt64/lib" -lfltk -lfltk_images -ltiff -ljpeg -lz -lzstd -lwebp -llerc -ljbig -llzma -ldeflate -lsharpyuv -lcomctl32 -lgdi32 -lole32 -luuid -lws2_32 -lwinspool -lcomdlg32 -static
SRC = src/main.cpp src/gui.cpp src/utils.cpp src/stb_impl.cpp \
src/png_to_jpg.cpp src/jpg_to_png.cpp src/image_loader.cpp \
src/bmp_to_png.cpp src/bmp_to_jpg.cpp src/png_to_bmp.cpp \
src/jpg_to_bmp.cpp src/tiff_to_bmp.cpp src/bmp_to_tiff.cpp \
src/tiff_to_png.cpp src/png_to_tiff.cpp src/tiff_to_jpg.cpp \
src/jpg_to_tiff.cpp
OBJ = $(SRC:src/%.cpp=obj/%.o)
TARGET = image_converter
all: $(TARGET)
$(TARGET): $(OBJ)
$(CXX) $(CXXFLAGS) $^ -o $@ $(LDFLAGS)
obj/%.o: src/%.cpp
$(CXX) $(CXXFLAGS) -c $< -o $@
clean:
rm -f $(OBJ) $(TARGET)
.PHONY: all clean

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class AvifToBmpConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class AvifToJpegConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class AvifToJpgConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class AvifToPngConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class AvifToTiffConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class AvifToWebpConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class BmpToAvifConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class BmpToJpegConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class BmpToWebpConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -17,6 +17,8 @@ enum class ImageFormat {
PNG, PNG,
JPG, JPG,
TIFF, TIFF,
WEBP,
AVIF,
UNKNOWN UNKNOWN
}; };

View File

@@ -7,7 +7,7 @@ public:
static ImageData load(const std::string& path); static ImageData load(const std::string& path);
static bool save_png(const std::string& path, const ImageData& data); static bool save_png(const std::string& path, const ImageData& data);
static bool save_jpg(const std::string& path, const ImageData& data, int quality = 90); static bool save_jpg(const std::string& path, const ImageData& data, int quality = 90);
static void validate_image(const unsigned char* data, int width, int height);
private: private:
static void validate_image(const unsigned char* data, int width, int height);
}; };

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class JpegToAvifConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,12 @@
#pragma once
#include <string>
#include "common.hpp"
class JpegToBmpConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,12 @@
#pragma once
#include <string>
#include "common.hpp"
class JpegToPngConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,12 @@
#pragma once
#include <string>
#include "common.hpp"
class JpegToTiffConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class JpegToWebpConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class JpgToAvifConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class JpgToWebpConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class PngToAvifConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class PngToJpegConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class PngToWebpConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class TiffToAvifConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class TiffToJpegConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class TiffToWebpConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,11 @@
#pragma once
#include "converter_base.hpp"
#include "common.hpp"
class WebpToAvifConverter : public ConverterBase {
public:
bool convert(const std::string& input, const std::string& output) override;
protected:
bool validate(const ImageData& data) override;
};

View File

@@ -0,0 +1,12 @@
#pragma once
#include <string>
#include "common.hpp"
class WebpToBmpConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class WebpToJpegConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,13 @@
#pragma once
#include <string>
#include "common.hpp"
class WebpToJpgConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path,
int quality = 90);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,12 @@
#pragma once
#include <string>
#include "common.hpp"
class WebpToPngConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,12 @@
#pragma once
#include <string>
#include "common.hpp"
class WebpToTiffConverter {
public:
static bool convert(const std::string& input_path,
const std::string& output_path);
private:
static bool validate_input(const ImageData& data);
};

View File

@@ -0,0 +1,31 @@
#include "avif_to_bmp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <stb/stb_image_write.h>
bool AvifToBmpConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载AVIF图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 保存为BMP
return stbi_write_bmp(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get());
}
bool AvifToBmpConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,32 @@
#include "avif_to_jpeg.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <stb/stb_image_write.h>
bool AvifToJpegConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载AVIF图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 保存为JPEG(默认质量90)
return stbi_write_jpg(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
90);
}
bool AvifToJpegConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,32 @@
#include "avif_to_jpg.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <stb/stb_image_write.h>
bool AvifToJpgConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载AVIF图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 保存为JPG(默认质量90)
return stbi_write_jpg(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
90);
}
bool AvifToJpgConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,33 @@
#include "avif_to_png.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <stb/stb_image_write.h>
bool AvifToPngConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载AVIF图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 保存为PNG
return stbi_write_png(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
data.width * data.channels);
}
bool AvifToPngConverter::validate(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,65 @@
#include "avif_to_tiff.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <tiffio.h>
bool AvifToTiffConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载AVIF图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 创建TIFF文件
TIFF* tif = TIFFOpen(output_path.c_str(), "w");
if (!tif) {
return false;
}
// 设置TIFF标签
TIFFSetField(tif, TIFFTAG_IMAGEWIDTH, data.width);
TIFFSetField(tif, TIFFTAG_IMAGELENGTH, data.height);
TIFFSetField(tif, TIFFTAG_SAMPLESPERPIXEL, data.channels);
TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 8);
TIFFSetField(tif, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
TIFFSetField(tif, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
TIFFSetField(tif, TIFFTAG_PHOTOMETRIC,
(data.channels == 1) ? PHOTOMETRIC_MINISBLACK :
(data.channels == 3) ? PHOTOMETRIC_RGB : PHOTOMETRIC_RGB);
TIFFSetField(tif, TIFFTAG_COMPRESSION, COMPRESSION_LZW);
// 写入图像数据
tsize_t linebytes = data.width * data.channels;
unsigned char* buf = nullptr;
if (TIFFScanlineSize(tif) == linebytes) {
buf = (unsigned char*)_TIFFmalloc(linebytes);
} else {
buf = (unsigned char*)_TIFFmalloc(TIFFScanlineSize(tif));
}
for (uint32_t row = 0; row < data.height; row++) {
memcpy(buf, &data.pixels.get()[row * linebytes], linebytes);
if (TIFFWriteScanline(tif, buf, row, 0) < 0) {
_TIFFfree(buf);
TIFFClose(tif);
return false;
}
}
// 清理资源
_TIFFfree(buf);
TIFFClose(tif);
return true;
}
bool AvifToTiffConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 1 || data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,50 @@
#include "avif_to_webp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <webp/encode.h>
bool AvifToWebpConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载AVIF图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 保存为WEBP
FILE* f = fopen(output_path.c_str(), "wb");
if (!f) {
return false;
}
uint8_t* output = nullptr;
size_t size = 0;
if (data.channels == 3) {
size = WebPEncodeRGB(data.pixels.get(), data.width, data.height,
data.width * 3, 90, &output);
} else if (data.channels == 4) {
size = WebPEncodeRGBA(data.pixels.get(), data.width, data.height,
data.width * 4, 90, &output);
}
if (size == 0 || !output) {
fclose(f);
return false;
}
fwrite(output, 1, size, f);
fclose(f);
WebPFree(output);
return true;
}
bool AvifToWebpConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,71 @@
#include "bmp_to_avif.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <stb/stb_image.h>
bool BmpToAvifConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载BMP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 创建AVIF编码器
avifEncoder* encoder = avifEncoderCreate();
encoder->maxThreads = 4;
encoder->minQuantizer = 20;
encoder->maxQuantizer = 20;
encoder->speed = 6;
// 创建AVIF图像
avifImage* image = avifImageCreate(data.width, data.height, 8, AVIF_PIXEL_FORMAT_YUV420);
avifRGBImage rgbImage;
avifRGBImageSetDefaults(&rgbImage, image);
rgbImage.pixels = data.pixels.get();
rgbImage.rowBytes = data.width * data.channels;
rgbImage.format = (data.channels == 3) ? AVIF_RGB_FORMAT_RGB : AVIF_RGB_FORMAT_RGBA;
// 转换RGB到YUV
if (avifImageRGBToYUV(image, &rgbImage) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 编码AVIF图像
avifRWData output = AVIF_DATA_EMPTY;
if (avifEncoderWrite(encoder, image, &output) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 保存AVIF文件
FILE* f = fopen(output_path.c_str(), "wb");
if (!f) {
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
fwrite(output.data, 1, output.size, f);
fclose(f);
// 清理资源
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return true;
}
bool BmpToAvifConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,35 @@
#include "bmp_to_jpeg.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <stb/stb_image.h>
#include <stb/stb_image_write.h>
#include <string>
bool BmpToJpegConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载BMP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 保存为JPEG
return stbi_write_jpg(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
quality);
}
bool BmpToJpegConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,61 @@
#include "bmp_to_webp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/encode.h>
#include <string>
bool BmpToWebpConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载BMP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 编码为WebP
uint8_t* output = nullptr;
size_t output_size;
if (data.channels == 3) {
output_size = WebPEncodeRGB(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
} else {
output_size = WebPEncodeRGBA(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
}
if (output_size == 0) {
return false;
}
// 保存WebP文件
FILE* file = fopen(output_path.c_str(), "wb");
if (!file) {
WebPFree(output);
return false;
}
fwrite(output, 1, output_size, file);
fclose(file);
WebPFree(output);
return true;
}
bool BmpToWebpConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -12,6 +12,34 @@
#include "png_to_tiff.hpp" #include "png_to_tiff.hpp"
#include "tiff_to_jpg.hpp" #include "tiff_to_jpg.hpp"
#include "jpg_to_tiff.hpp" #include "jpg_to_tiff.hpp"
#include "webp_to_jpg.hpp"
#include "webp_to_png.hpp"
#include "webp_to_bmp.hpp"
#include "webp_to_tiff.hpp"
#include "jpg_to_webp.hpp"
#include "png_to_webp.hpp"
#include "bmp_to_webp.hpp"
#include "tiff_to_webp.hpp"
#include "jpeg_to_png.hpp"
#include "jpeg_to_bmp.hpp"
#include "jpeg_to_tiff.hpp"
#include "jpeg_to_webp.hpp"
#include "png_to_jpeg.hpp"
#include "bmp_to_jpeg.hpp"
#include "tiff_to_jpeg.hpp"
#include "webp_to_jpeg.hpp"
#include "avif_to_png.hpp"
#include "avif_to_jpg.hpp"
#include "avif_to_jpeg.hpp"
#include "avif_to_bmp.hpp"
#include "avif_to_tiff.hpp"
#include "avif_to_webp.hpp"
#include "png_to_avif.hpp"
#include "jpg_to_avif.hpp"
#include "jpeg_to_avif.hpp"
#include "bmp_to_avif.hpp"
#include "tiff_to_avif.hpp"
#include "webp_to_avif.hpp"
#include <FL/Fl_File_Chooser.H> #include <FL/Fl_File_Chooser.H>
#include <FL/fl_ask.H> #include <FL/fl_ask.H>
#include <stdexcept> #include <stdexcept>
@@ -45,6 +73,34 @@ MainWindow::MainWindow(int w, int h, const char* title)
format_choice->add("PNG to TIFF"); format_choice->add("PNG to TIFF");
format_choice->add("TIFF to JPG"); format_choice->add("TIFF to JPG");
format_choice->add("JPG to TIFF"); format_choice->add("JPG to TIFF");
format_choice->add("WEBP to JPG");
format_choice->add("WEBP to PNG");
format_choice->add("WEBP to BMP");
format_choice->add("WEBP to TIFF");
format_choice->add("JPG to WEBP");
format_choice->add("PNG to WEBP");
format_choice->add("BMP to WEBP");
format_choice->add("TIFF to WEBP");
format_choice->add("JPEG to PNG");
format_choice->add("JPEG to BMP");
format_choice->add("JPEG to TIFF");
format_choice->add("JPEG to WEBP");
format_choice->add("PNG to JPEG");
format_choice->add("BMP to JPEG");
format_choice->add("TIFF to JPEG");
format_choice->add("WEBP to JPEG");
format_choice->add("AVIF to PNG");
format_choice->add("AVIF to JPG");
format_choice->add("AVIF to JPEG");
format_choice->add("AVIF to BMP");
format_choice->add("AVIF to TIFF");
format_choice->add("AVIF to WEBP");
format_choice->add("PNG to AVIF");
format_choice->add("JPG to AVIF");
format_choice->add("JPEG to AVIF");
format_choice->add("BMP to AVIF");
format_choice->add("TIFF to AVIF");
format_choice->add("WEBP to AVIF");
format_choice->value(0); format_choice->value(0);
convert_btn = new Fl_Button(150, 150, 100, 30, "转换"); convert_btn = new Fl_Button(150, 150, 100, 30, "转换");
@@ -122,6 +178,90 @@ void MainWindow::convert_cb(Fl_Widget* w, void* data) {
case 11: // JPG to TIFF case 11: // JPG to TIFF
success = JpgToTiffConverter::convert(input, output); success = JpgToTiffConverter::convert(input, output);
break; break;
case 12: // WEBP to JPG
success = WebpToJpgConverter::convert(input, output);
break;
case 13: // WEBP to PNG
success = WebpToPngConverter::convert(input, output);
break;
case 14: // WEBP to BMP
success = WebpToBmpConverter::convert(input, output);
break;
case 15: // WEBP to TIFF
success = WebpToTiffConverter::convert(input, output);
break;
case 16: // JPG to WEBP
success = JpgToWebpConverter::convert(input, output);
break;
case 17: // PNG to WEBP
success = PngToWebpConverter::convert(input, output);
break;
case 18: // BMP to WEBP
success = BmpToWebpConverter::convert(input, output);
break;
case 19: // TIFF to WEBP
success = TiffToWebpConverter::convert(input, output);
break;
case 20: // JPEG to PNG
success = JpegToPngConverter::convert(input, output);
break;
case 21: // JPEG to BMP
success = JpegToBmpConverter::convert(input, output);
break;
case 22: // JPEG to TIFF
success = JpegToTiffConverter::convert(input, output);
break;
case 23: // JPEG to WEBP
success = JpegToWebpConverter::convert(input, output);
break;
case 24: // PNG to JPEG
success = PngToJpegConverter::convert(input, output);
break;
case 25: // BMP to JPEG
success = BmpToJpegConverter::convert(input, output);
break;
case 26: // TIFF to JPEG
success = TiffToJpegConverter::convert(input, output);
break;
case 27: // WEBP to JPEG
success = WebpToJpegConverter::convert(input, output);
break;
case 28: // AVIF to PNG
success = AvifToPngConverter().convert(input, output);
break;
case 29: // AVIF to JPG
success = AvifToJpgConverter().convert(input, output);
break;
case 30: // AVIF to JPEG
success = AvifToJpegConverter().convert(input, output);
break;
case 31: // AVIF to BMP
success = AvifToBmpConverter().convert(input, output);
break;
case 32: // AVIF to TIFF
success = AvifToTiffConverter().convert(input, output);
break;
case 33: // AVIF to WEBP
success = AvifToWebpConverter().convert(input, output);
break;
case 34: // PNG to AVIF
success = PngToAvifConverter().convert(input, output);
break;
case 35: // JPG to AVIF
success = JpgToAvifConverter().convert(input, output);
break;
case 36: // JPEG to AVIF
success = JpegToAvifConverter().convert(input, output);
break;
case 37: // BMP to AVIF
success = BmpToAvifConverter().convert(input, output);
break;
case 38: // TIFF to AVIF
success = TiffToAvifConverter().convert(input, output);
break;
case 39: // WEBP to AVIF
success = WebpToAvifConverter().convert(input, output);
break;
} }
if (!success) throw std::runtime_error("转换失败"); if (!success) throw std::runtime_error("转换失败");
} catch (const std::exception& e) { } catch (const std::exception& e) {

View File

@@ -2,13 +2,166 @@
#include "common.hpp" #include "common.hpp"
#include <stb/stb_image.h> #include <stb/stb_image.h>
#include <stb/stb_image_write.h> #include <stb/stb_image_write.h>
#include <webp/decode.h>
#include <avif/avif.h>
#include <fstream>
#include <stdexcept> #include <stdexcept>
#include <iostream> #include <iostream>
#include <cstring>
static bool is_avif_file(const std::string& path) {
std::ifstream file(path, std::ios::binary);
if (!file) return false;
char header[12];
if (!file.read(header, 12)) return false;
return !memcmp(header, "\0\0\0 ftypavif", 12) ||
!memcmp(header, "\0\0\0 ftypavis", 12);
}
static bool is_webp_file(const std::string& path) {
std::ifstream file(path, std::ios::binary);
if (!file) return false;
char header[12];
if (!file.read(header, 12)) return false;
return !memcmp(header, "RIFF", 4) && !memcmp(header + 8, "WEBP", 4);
}
static ImageData load_avif(const std::string& path) {
ImageData data;
avifDecoder* decoder = avifDecoderCreate();
if (!decoder) {
throw std::runtime_error("无法创建AVIF解码器");
}
avifResult result = avifDecoderSetIOFile(decoder, path.c_str());
if (result != AVIF_RESULT_OK) {
avifDecoderDestroy(decoder);
throw std::runtime_error("无法读取AVIF文件");
}
result = avifDecoderParse(decoder);
if (result != AVIF_RESULT_OK) {
avifDecoderDestroy(decoder);
throw std::runtime_error("无效的AVIF图像");
}
result = avifDecoderNextImage(decoder);
if (result != AVIF_RESULT_OK) {
avifDecoderDestroy(decoder);
throw std::runtime_error("无法解码AVIF图像");
}
data.width = decoder->image->width;
data.height = decoder->image->height;
data.channels = 4; // AVIF解码为RGBA
// 分配内存并转换图像数据
uint8_t* rgba_pixels = new uint8_t[data.width * data.height * 4];
avifRGBImage rgb;
avifRGBImageSetDefaults(&rgb, decoder->image);
rgb.format = AVIF_RGB_FORMAT_RGBA;
rgb.depth = 8;
rgb.pixels = rgba_pixels;
rgb.rowBytes = data.width * 4;
if (avifImageYUVToRGB(decoder->image, &rgb) != AVIF_RESULT_OK) {
delete[] rgba_pixels;
avifDecoderDestroy(decoder);
throw std::runtime_error("AVIF颜色空间转换失败");
}
avifDecoderDestroy(decoder);
// 验证图像数据
try {
ImageLoader::validate_image(rgba_pixels, data.width, data.height);
} catch (...) {
delete[] rgba_pixels;
throw;
}
data.pixels = std::unique_ptr<unsigned char, void(*)(void*)>(rgba_pixels, [](void* p) { delete[] static_cast<uint8_t*>(p); });
return data;
}
ImageData ImageLoader::load(const std::string& path) { ImageData ImageLoader::load(const std::string& path) {
ImageData data; ImageData data;
// 加载图像 // 检查是否为AVIF格式
if (is_avif_file(path)) {
return load_avif(path);
}
// 检查是否为WebP格式
else if (is_webp_file(path)) {
// 读取WebP文件数据
std::ifstream file(path, std::ios::binary | std::ios::ate);
if (!file) {
throw std::runtime_error("无法打开WebP文件");
}
size_t size = file.tellg();
file.seekg(0, std::ios::beg);
std::vector<uint8_t> webp_data(size);
if (!file.read(reinterpret_cast<char*>(webp_data.data()), size)) {
throw std::runtime_error("无法读取WebP文件");
}
// 解码WebP图像
WebPBitstreamFeatures features;
if (WebPGetFeatures(webp_data.data(), webp_data.size(), &features) != VP8_STATUS_OK) {
throw std::runtime_error("无效的WebP图像");
}
data.width = features.width;
data.height = features.height;
data.channels = features.has_alpha ? 4 : 3;
// 解码WebP图像为RGBA格式
uint8_t* rgba_pixels = WebPDecodeRGBA(webp_data.data(), webp_data.size(), &data.width, &data.height);
if (!rgba_pixels) {
throw std::runtime_error("无法解码WebP图像");
}
// 如果没有alpha通道转换为RGB格式
if (data.channels == 3) {
uint8_t* rgb_pixels = new uint8_t[data.width * data.height * 3];
for (int i = 0; i < data.width * data.height; ++i) {
rgb_pixels[i*3] = rgba_pixels[i*4];
rgb_pixels[i*3+1] = rgba_pixels[i*4+1];
rgb_pixels[i*3+2] = rgba_pixels[i*4+2];
}
WebPFree(rgba_pixels);
// 验证图像数据
try {
ImageLoader::validate_image(rgb_pixels, data.width, data.height);
} catch (...) {
delete[] rgb_pixels;
throw;
}
data.pixels = std::unique_ptr<unsigned char, void(*)(void*)>(rgb_pixels, [](void* p) { delete[] static_cast<uint8_t*>(p); });
} else {
// 验证图像数据
try {
ImageLoader::validate_image(rgba_pixels, data.width, data.height);
} catch (...) {
WebPFree(rgba_pixels);
throw;
}
data.pixels = std::unique_ptr<unsigned char, void(*)(void*)>(rgba_pixels, WebPFree);
}
} else {
// 使用STB加载其他格式图像
unsigned char* pixels = stbi_load(path.c_str(), unsigned char* pixels = stbi_load(path.c_str(),
&data.width, &data.width,
&data.height, &data.height,
@@ -20,14 +173,14 @@ ImageData ImageLoader::load(const std::string& path) {
// 验证图像数据 // 验证图像数据
try { try {
validate_image(pixels, data.width, data.height); ImageLoader::validate_image(pixels, data.width, data.height);
} catch (...) { } catch (...) {
stbi_image_free(pixels); stbi_image_free(pixels);
throw; throw;
} }
// 转移所有权到智能指针 data.pixels = std::unique_ptr<unsigned char, void(*)(void*)>(pixels, stbi_image_free);
data.pixels.reset(pixels); }
return data; return data;
} }

View File

@@ -0,0 +1,71 @@
#include "jpeg_to_avif.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <stb/stb_image.h>
bool JpegToAvifConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载JPEG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 创建AVIF编码器
avifEncoder* encoder = avifEncoderCreate();
encoder->maxThreads = 4;
encoder->minQuantizer = 20;
encoder->maxQuantizer = 20;
encoder->speed = 6;
// 创建AVIF图像
avifImage* image = avifImageCreate(data.width, data.height, 8, AVIF_PIXEL_FORMAT_YUV420);
avifRGBImage rgbImage;
avifRGBImageSetDefaults(&rgbImage, image);
rgbImage.pixels = data.pixels.get();
rgbImage.rowBytes = data.width * data.channels;
rgbImage.format = (data.channels == 3) ? AVIF_RGB_FORMAT_RGB : AVIF_RGB_FORMAT_RGBA;
// 转换RGB到YUV
if (avifImageRGBToYUV(image, &rgbImage) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 编码AVIF图像
avifRWData output = AVIF_DATA_EMPTY;
if (avifEncoderWrite(encoder, image, &output) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 保存AVIF文件
FILE* f = fopen(output_path.c_str(), "wb");
if (!f) {
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
fwrite(output.data, 1, output.size, f);
fclose(f);
// 清理资源
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return true;
}
bool JpegToAvifConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,33 @@
#include "jpeg_to_bmp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <stb/stb_image.h>
#include <stb/stb_image_write.h>
#include <string>
bool JpegToBmpConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载JPEG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 保存为BMP
return stbi_write_bmp(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get());
}
bool JpegToBmpConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 1 || data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,34 @@
#include "jpeg_to_png.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <stb/stb_image.h>
#include <stb/stb_image_write.h>
#include <string>
bool JpegToPngConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载JPEG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 保存为PNG
return stbi_write_png(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
data.width * data.channels);
}
bool JpegToPngConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 1 || data.channels == 3);
}

View File

@@ -0,0 +1,61 @@
#include "jpeg_to_tiff.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <tiffio.h>
#include <stb/stb_image.h>
#include <string>
bool JpegToTiffConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载JPEG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入
if (!validate_input(data)) {
return false;
}
// 创建TIFF文件
TIFF* tif = TIFFOpen(output_path.c_str(), "w");
if (!tif) {
return false;
}
// 设置TIFF标签
TIFFSetField(tif, TIFFTAG_IMAGEWIDTH, data.width);
TIFFSetField(tif, TIFFTAG_IMAGELENGTH, data.height);
TIFFSetField(tif, TIFFTAG_SAMPLESPERPIXEL, data.channels);
TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 8);
TIFFSetField(tif, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
TIFFSetField(tif, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
// 根据通道数设置PhotometricInterpretation
if (data.channels == 1) {
TIFFSetField(tif, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_MINISBLACK);
} else if (data.channels == 3 || data.channels == 4) {
TIFFSetField(tif, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_RGB);
} else {
TIFFClose(tif);
return false;
}
// 写入图像数据
tsize_t linebytes = data.width * data.channels;
unsigned char* buf = (unsigned char*)_TIFFmalloc(linebytes);
for (int y = 0; y < data.height; y++) {
memcpy(buf, &data.pixels.get()[y * linebytes], linebytes);
TIFFWriteScanline(tif, buf, y, 0);
}
_TIFFfree(buf);
TIFFClose(tif);
return true;
}
bool JpegToTiffConverter::validate_input(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 1 || data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,61 @@
#include "jpeg_to_webp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/encode.h>
#include <string>
bool JpegToWebpConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载JPEG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 编码为WebP
uint8_t* output = nullptr;
size_t output_size;
if (data.channels == 3) {
output_size = WebPEncodeRGB(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
} else {
output_size = WebPEncodeRGBA(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
}
if (output_size == 0) {
return false;
}
// 保存WebP文件
FILE* file = fopen(output_path.c_str(), "wb");
if (!file) {
WebPFree(output);
return false;
}
fwrite(output, 1, output_size, file);
fclose(file);
WebPFree(output);
return true;
}
bool JpegToWebpConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,71 @@
#include "jpg_to_avif.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <stb/stb_image.h>
bool JpgToAvifConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载JPG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 创建AVIF编码器
avifEncoder* encoder = avifEncoderCreate();
encoder->maxThreads = 4;
encoder->minQuantizer = 20;
encoder->maxQuantizer = 20;
encoder->speed = 6;
// 创建AVIF图像
avifImage* image = avifImageCreate(data.width, data.height, 8, AVIF_PIXEL_FORMAT_YUV420);
avifRGBImage rgbImage;
avifRGBImageSetDefaults(&rgbImage, image);
rgbImage.pixels = data.pixels.get();
rgbImage.rowBytes = data.width * data.channels;
rgbImage.format = (data.channels == 3) ? AVIF_RGB_FORMAT_RGB : AVIF_RGB_FORMAT_RGBA;
// 转换RGB到YUV
if (avifImageRGBToYUV(image, &rgbImage) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 编码AVIF图像
avifRWData output = AVIF_DATA_EMPTY;
if (avifEncoderWrite(encoder, image, &output) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 保存AVIF文件
FILE* f = fopen(output_path.c_str(), "wb");
if (!f) {
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
fwrite(output.data, 1, output.size, f);
fclose(f);
// 清理资源
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return true;
}
bool JpgToAvifConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,61 @@
#include "jpg_to_webp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/encode.h>
#include <string>
bool JpgToWebpConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载JPG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 编码为WebP
uint8_t* output = nullptr;
size_t output_size;
if (data.channels == 3) {
output_size = WebPEncodeRGB(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
} else {
output_size = WebPEncodeRGBA(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
}
if (output_size == 0) {
return false;
}
// 保存WebP文件
FILE* file = fopen(output_path.c_str(), "wb");
if (!file) {
WebPFree(output);
return false;
}
fwrite(output, 1, output_size, file);
fclose(file);
WebPFree(output);
return true;
}
bool JpgToWebpConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,71 @@
#include "png_to_avif.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <stb/stb_image.h>
bool PngToAvifConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载PNG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 创建AVIF编码器
avifEncoder* encoder = avifEncoderCreate();
encoder->maxThreads = 4;
encoder->minQuantizer = 20;
encoder->maxQuantizer = 20;
encoder->speed = 6;
// 创建AVIF图像
avifImage* image = avifImageCreate(data.width, data.height, 8, AVIF_PIXEL_FORMAT_YUV420);
avifRGBImage rgbImage;
avifRGBImageSetDefaults(&rgbImage, image);
rgbImage.pixels = data.pixels.get();
rgbImage.rowBytes = data.width * data.channels;
rgbImage.format = (data.channels == 3) ? AVIF_RGB_FORMAT_RGB : AVIF_RGB_FORMAT_RGBA;
// 转换RGB到YUV
if (avifImageRGBToYUV(image, &rgbImage) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 编码AVIF图像
avifRWData output = AVIF_DATA_EMPTY;
if (avifEncoderWrite(encoder, image, &output) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 保存AVIF文件
FILE* f = fopen(output_path.c_str(), "wb");
if (!f) {
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
fwrite(output.data, 1, output.size, f);
fclose(f);
// 清理资源
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return true;
}
bool PngToAvifConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,35 @@
#include "png_to_jpeg.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <stb/stb_image.h>
#include <stb/stb_image_write.h>
#include <string>
bool PngToJpegConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载PNG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 保存为JPEG
return stbi_write_jpg(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
quality);
}
bool PngToJpegConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,61 @@
#include "png_to_webp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/encode.h>
#include <string>
bool PngToWebpConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载PNG图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 编码为WebP
uint8_t* output = nullptr;
size_t output_size;
if (data.channels == 3) {
output_size = WebPEncodeRGB(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
} else {
output_size = WebPEncodeRGBA(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
}
if (output_size == 0) {
return false;
}
// 保存WebP文件
FILE* file = fopen(output_path.c_str(), "wb");
if (!file) {
WebPFree(output);
return false;
}
fwrite(output, 1, output_size, file);
fclose(file);
WebPFree(output);
return true;
}
bool PngToWebpConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,71 @@
#include "tiff_to_avif.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <tiffio.h>
bool TiffToAvifConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载TIFF图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 创建AVIF编码器
avifEncoder* encoder = avifEncoderCreate();
encoder->maxThreads = 4;
encoder->minQuantizer = 20;
encoder->maxQuantizer = 20;
encoder->speed = 6;
// 创建AVIF图像
avifImage* image = avifImageCreate(data.width, data.height, 8, AVIF_PIXEL_FORMAT_YUV420);
avifRGBImage rgbImage;
avifRGBImageSetDefaults(&rgbImage, image);
rgbImage.pixels = data.pixels.get();
rgbImage.rowBytes = data.width * data.channels;
rgbImage.format = (data.channels == 3) ? AVIF_RGB_FORMAT_RGB : AVIF_RGB_FORMAT_RGBA;
// 转换RGB到YUV
if (avifImageRGBToYUV(image, &rgbImage) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 编码AVIF图像
avifRWData output = AVIF_DATA_EMPTY;
if (avifEncoderWrite(encoder, image, &output) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 保存AVIF文件
FILE* f = fopen(output_path.c_str(), "wb");
if (!f) {
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
fwrite(output.data, 1, output.size, f);
fclose(f);
// 清理资源
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return true;
}
bool TiffToAvifConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 1 || data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,73 @@
#include "tiff_to_jpeg.hpp"
#include "common.hpp"
#include <tiffio.h>
#include <stb/stb_image_write.h>
#include <string>
bool TiffToJpegConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 使用libtiff加载TIFF图像
TIFF* tif = TIFFOpen(input_path.c_str(), "r");
if (!tif) {
return false;
}
// 获取图像信息
uint32 width, height;
TIFFGetField(tif, TIFFTAG_IMAGEWIDTH, &width);
TIFFGetField(tif, TIFFTAG_IMAGELENGTH, &height);
// 获取TIFF格式信息
uint16 samplesperpixel, bitspersample, photometric;
TIFFGetField(tif, TIFFTAG_SAMPLESPERPIXEL, &samplesperpixel);
TIFFGetField(tif, TIFFTAG_BITSPERSAMPLE, &bitspersample);
TIFFGetField(tif, TIFFTAG_PHOTOMETRIC, &photometric);
// 验证TIFF格式
if (bitspersample != 8) {
TIFFClose(tif);
return false;
}
// 设置输出通道数
ImageData data;
data.width = width;
data.height = height;
data.channels = samplesperpixel;
data.pixels.reset(new unsigned char[width * height * data.channels]);
// 读取图像数据
tdata_t buf = _TIFFmalloc(TIFFScanlineSize(tif));
for (uint32 row = 0; row < height; row++) {
if (TIFFReadScanline(tif, buf, row) == -1) {
_TIFFfree(buf);
TIFFClose(tif);
return false;
}
memcpy(&data.pixels.get()[row * width * data.channels],
buf,
width * data.channels);
}
_TIFFfree(buf);
TIFFClose(tif);
// 验证输入
if (!validate_input(data)) {
return false;
}
// 保存为JPEG
return stbi_write_jpg(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
quality);
}
bool TiffToJpegConverter::validate_input(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 1 || data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,61 @@
#include "tiff_to_webp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/encode.h>
#include <string>
bool TiffToWebpConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载TIFF图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 编码为WebP
uint8_t* output = nullptr;
size_t output_size;
if (data.channels == 3) {
output_size = WebPEncodeRGB(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
} else {
output_size = WebPEncodeRGBA(data.pixels.get(),
data.width,
data.height,
data.width * data.channels,
quality,
&output);
}
if (output_size == 0) {
return false;
}
// 保存WebP文件
FILE* file = fopen(output_path.c_str(), "wb");
if (!file) {
WebPFree(output);
return false;
}
fwrite(output, 1, output_size, file);
fclose(file);
WebPFree(output);
return true;
}
bool TiffToWebpConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -7,5 +7,7 @@ ImageFormat get_format_from_extension(const std::string& path) {
if (ext == "png") return ImageFormat::PNG; if (ext == "png") return ImageFormat::PNG;
if (ext == "jpg" || ext == "jpeg") return ImageFormat::JPG; if (ext == "jpg" || ext == "jpeg") return ImageFormat::JPG;
if (ext == "tif" || ext == "tiff") return ImageFormat::TIFF;
if (ext == "webp") return ImageFormat::WEBP;
return ImageFormat::UNKNOWN; return ImageFormat::UNKNOWN;
} }

View File

@@ -0,0 +1,71 @@
#include "webp_to_avif.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <avif/avif.h>
#include <webp/decode.h>
bool WebpToAvifConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载WEBP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate(data)) {
return false;
}
// 创建AVIF编码器
avifEncoder* encoder = avifEncoderCreate();
encoder->maxThreads = 4;
encoder->minQuantizer = 20;
encoder->maxQuantizer = 20;
encoder->speed = 6;
// 创建AVIF图像
avifImage* image = avifImageCreate(data.width, data.height, 8, AVIF_PIXEL_FORMAT_YUV420);
avifRGBImage rgbImage;
avifRGBImageSetDefaults(&rgbImage, image);
rgbImage.pixels = data.pixels.get();
rgbImage.rowBytes = data.width * data.channels;
rgbImage.format = (data.channels == 3) ? AVIF_RGB_FORMAT_RGB : AVIF_RGB_FORMAT_RGBA;
// 转换RGB到YUV
if (avifImageRGBToYUV(image, &rgbImage) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 编码AVIF图像
avifRWData output = AVIF_DATA_EMPTY;
if (avifEncoderWrite(encoder, image, &output) != AVIF_RESULT_OK) {
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
// 保存AVIF文件
FILE* f = fopen(output_path.c_str(), "wb");
if (!f) {
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return false;
}
fwrite(output.data, 1, output.size, f);
fclose(f);
// 清理资源
avifRWDataFree(&output);
avifEncoderDestroy(encoder);
avifImageDestroy(image);
return true;
}
bool WebpToAvifConverter::validate(const ImageData& data) {
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,33 @@
#include "webp_to_bmp.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/decode.h>
#include <stb/stb_image_write.h>
#include <string>
bool WebpToBmpConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载WebP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 保存为BMP
return stbi_write_bmp(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get());
}
bool WebpToBmpConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,35 @@
#include "webp_to_jpeg.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/decode.h>
#include <stb/stb_image_write.h>
#include <string>
bool WebpToJpegConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载WebP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 保存为JPEG
return stbi_write_jpg(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
quality);
}
bool WebpToJpegConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,35 @@
#include "webp_to_jpg.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/decode.h>
#include <stb/stb_image_write.h>
#include <string>
bool WebpToJpgConverter::convert(const std::string& input_path,
const std::string& output_path,
int quality) {
// 加载WebP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 保存为JPG
return stbi_write_jpg(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
quality);
}
bool WebpToJpgConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,34 @@
#include "webp_to_png.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/decode.h>
#include <stb/stb_image_write.h>
#include <string>
bool WebpToPngConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载WebP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 保存为PNG
return stbi_write_png(output_path.c_str(),
data.width,
data.height,
data.channels,
data.pixels.get(),
data.width * data.channels);
}
bool WebpToPngConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,78 @@
#include "webp_to_tiff.hpp"
#include "common.hpp"
#include "image_loader.hpp"
#include <webp/decode.h>
#include <tiffio.h>
#include <string>
bool WebpToTiffConverter::convert(const std::string& input_path,
const std::string& output_path) {
// 加载WebP图像
ImageData data = ImageLoader::load(input_path);
if (!data.pixels) {
return false;
}
// 验证输入格式
if (!validate_input(data)) {
return false;
}
// 创建TIFF文件
TIFF* tif = TIFFOpen(output_path.c_str(), "w");
if (!tif) {
return false;
}
// 设置TIFF文件参数
TIFFSetField(tif, TIFFTAG_IMAGEWIDTH, data.width);
TIFFSetField(tif, TIFFTAG_IMAGELENGTH, data.height);
TIFFSetField(tif, TIFFTAG_SAMPLESPERPIXEL, data.channels);
TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 8);
TIFFSetField(tif, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
TIFFSetField(tif, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
TIFFSetField(tif, TIFFTAG_PHOTOMETRIC,
data.channels == 3 ? PHOTOMETRIC_RGB : PHOTOMETRIC_MINISBLACK);
TIFFSetField(tif, TIFFTAG_COMPRESSION, COMPRESSION_LZW);
// 写入图像数据
tsize_t linebytes = data.width * data.channels;
unsigned char* buf = nullptr;
if (TIFFScanlineSize(tif) == linebytes) {
buf = (unsigned char*)data.pixels.get() +
(data.height - 1) * linebytes;
for (uint32 row = 0; row < data.height; row++) {
if (TIFFWriteScanline(tif, buf, row, 0) < 0) {
TIFFClose(tif);
return false;
}
buf -= linebytes;
}
} else {
buf = (unsigned char*)_TIFFmalloc(linebytes);
if (!buf) {
TIFFClose(tif);
return false;
}
unsigned char* src = (unsigned char*)data.pixels.get();
for (uint32 row = 0; row < data.height; row++) {
memcpy(buf, src, linebytes);
if (TIFFWriteScanline(tif, buf, row, 0) < 0) {
_TIFFfree(buf);
TIFFClose(tif);
return false;
}
src += linebytes;
}
_TIFFfree(buf);
}
TIFFClose(tif);
return true;
}
bool WebpToTiffConverter::validate_input(const ImageData& data) {
// 确保是有效的图像数据
return data.width > 0 && data.height > 0 &&
(data.channels == 3 || data.channels == 4);
}

View File

@@ -0,0 +1,13 @@
#ifndef SHA224CONVERTER_HPP
#define SHA224CONVERTER_HPP
#include "Converter.hpp"
#include <string>
class SHA224Converter : public Converter {
public:
std::string convert(const std::string& input) override;
std::string getName() const override { return "SHA224"; }
};
#endif // SHA224CONVERTER_HPP

View File

@@ -0,0 +1,13 @@
#ifndef SHA384CONVERTER_HPP
#define SHA384CONVERTER_HPP
#include "Converter.hpp"
#include <string>
class SHA384Converter : public Converter {
public:
std::string convert(const std::string& input) override;
std::string getName() const override { return "SHA384"; }
};
#endif // SHA384CONVERTER_HPP

View File

@@ -0,0 +1,13 @@
#ifndef SHA3CONVERTER_HPP
#define SHA3CONVERTER_HPP
#include "Converter.hpp"
#include <string>
class SHA3Converter : public Converter {
public:
std::string convert(const std::string& input) override;
std::string getName() const override { return "SHA3"; }
};
#endif // SHA3CONVERTER_HPP

View File

@@ -0,0 +1,13 @@
#ifndef SHA512CONVERTER_HPP
#define SHA512CONVERTER_HPP
#include "Converter.hpp"
#include <string>
class SHA512Converter : public Converter {
public:
std::string convert(const std::string& input) override;
std::string getName() const override { return "SHA512"; }
};
#endif // SHA512CONVERTER_HPP

View File

@@ -5,6 +5,18 @@
#include "Converter.hpp" #include "Converter.hpp"
#include "BinaryConverter.hpp" #include "BinaryConverter.hpp"
#include "HexConverter.hpp" #include "HexConverter.hpp"
#include "ROT13Converter.hpp"
#include "MD5Converter.hpp"
#include "SHA1Converter.hpp"
#include "SHA256Converter.hpp"
#include "SHA224Converter.hpp"
#include "SHA384Converter.hpp"
#include "SHA512Converter.hpp"
#include "SHA3Converter.hpp"
#include "Base64Converter.hpp"
#include "Base32Converter.hpp"
#include "Ascii85Converter.hpp"
#include "CRC32Converter.hpp"
class Utils { class Utils {
public: public:

View File

@@ -22,6 +22,10 @@ MainWindow::MainWindow(int w, int h, const char* title) : Fl_Window(w, h, title)
conversionType->add("MD5"); conversionType->add("MD5");
conversionType->add("SHA1"); conversionType->add("SHA1");
conversionType->add("SHA256"); conversionType->add("SHA256");
conversionType->add("SHA224");
conversionType->add("SHA384");
conversionType->add("SHA512");
conversionType->add("SHA3");
conversionType->add("Base64"); conversionType->add("Base64");
conversionType->add("Base32"); conversionType->add("Base32");
conversionType->add("Ascii85"); conversionType->add("Ascii85");
@@ -48,6 +52,7 @@ void MainWindow::ConvertCallback(Fl_Widget* widget, void* data) {
} }
void MainWindow::ConvertText() { void MainWindow::ConvertText() {
try {
const char* input = inputText->value(); const char* input = inputText->value();
if (!input || strlen(input) == 0) { if (!input || strlen(input) == 0) {
fl_alert("请输入要转换的文本"); fl_alert("请输入要转换的文本");
@@ -63,4 +68,9 @@ void MainWindow::ConvertText() {
std::string result = converter->convert(input); std::string result = converter->convert(input);
outputText->value(result.c_str()); outputText->value(result.c_str());
} catch (const std::exception& e) {
fl_alert(("转换失败: " + std::string(e.what())).c_str());
} catch (...) {
fl_alert("未知错误: 转换失败");
}
} }

View File

@@ -0,0 +1,45 @@
#include "../include/SHA224Converter.hpp"
#include <openssl/evp.h>
#include <sstream>
#include <iomanip>
#include <stdexcept>
std::string SHA224Converter::convert(const std::string& input) {
EVP_MD_CTX* mdctx = EVP_MD_CTX_new();
if (!mdctx) {
throw std::runtime_error("Failed to create EVP_MD_CTX");
}
const EVP_MD* md = EVP_sha224();
if (!md) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("SHA224 not supported by this OpenSSL version");
}
unsigned char digest[EVP_MAX_MD_SIZE];
unsigned int digest_len;
if (EVP_DigestInit_ex(mdctx, md, nullptr) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to initialize digest");
}
if (EVP_DigestUpdate(mdctx, input.c_str(), input.length()) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to update digest");
}
if (EVP_DigestFinal_ex(mdctx, digest, &digest_len) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to finalize digest");
}
EVP_MD_CTX_free(mdctx);
std::stringstream ss;
for(unsigned int i = 0; i < digest_len; i++) {
ss << std::hex << std::setw(2) << std::setfill('0') << (int)digest[i];
}
return ss.str();
}

View File

@@ -0,0 +1,45 @@
#include "../include/SHA384Converter.hpp"
#include <openssl/evp.h>
#include <sstream>
#include <iomanip>
#include <stdexcept>
std::string SHA384Converter::convert(const std::string& input) {
EVP_MD_CTX* mdctx = EVP_MD_CTX_new();
if (!mdctx) {
throw std::runtime_error("Failed to create EVP_MD_CTX");
}
const EVP_MD* md = EVP_sha384();
if (!md) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("SHA384 not supported by this OpenSSL version");
}
unsigned char digest[EVP_MAX_MD_SIZE];
unsigned int digest_len;
if (EVP_DigestInit_ex(mdctx, md, nullptr) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to initialize digest");
}
if (EVP_DigestUpdate(mdctx, input.c_str(), input.length()) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to update digest");
}
if (EVP_DigestFinal_ex(mdctx, digest, &digest_len) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to finalize digest");
}
EVP_MD_CTX_free(mdctx);
std::stringstream ss;
for(unsigned int i = 0; i < digest_len; i++) {
ss << std::hex << std::setw(2) << std::setfill('0') << (int)digest[i];
}
return ss.str();
}

View File

@@ -0,0 +1,45 @@
#include "../include/SHA3Converter.hpp"
#include <openssl/evp.h>
#include <sstream>
#include <iomanip>
#include <stdexcept>
std::string SHA3Converter::convert(const std::string& input) {
EVP_MD_CTX* mdctx = EVP_MD_CTX_new();
if (!mdctx) {
throw std::runtime_error("Failed to create EVP_MD_CTX");
}
const EVP_MD* md = EVP_sha3_256();
if (!md) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("SHA3-256 not supported by this OpenSSL version");
}
unsigned char digest[EVP_MAX_MD_SIZE];
unsigned int digest_len;
if (EVP_DigestInit_ex(mdctx, md, nullptr) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to initialize digest");
}
if (EVP_DigestUpdate(mdctx, input.c_str(), input.length()) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to update digest");
}
if (EVP_DigestFinal_ex(mdctx, digest, &digest_len) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to finalize digest");
}
EVP_MD_CTX_free(mdctx);
std::stringstream ss;
for(unsigned int i = 0; i < digest_len; i++) {
ss << std::hex << std::setw(2) << std::setfill('0') << (int)digest[i];
}
return ss.str();
}

View File

@@ -0,0 +1,45 @@
#include "../include/SHA512Converter.hpp"
#include <openssl/evp.h>
#include <sstream>
#include <iomanip>
#include <stdexcept>
std::string SHA512Converter::convert(const std::string& input) {
EVP_MD_CTX* mdctx = EVP_MD_CTX_new();
if (!mdctx) {
throw std::runtime_error("Failed to create EVP_MD_CTX");
}
const EVP_MD* md = EVP_sha512();
if (!md) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("SHA512 not supported by this OpenSSL version");
}
unsigned char digest[EVP_MAX_MD_SIZE];
unsigned int digest_len;
if (EVP_DigestInit_ex(mdctx, md, nullptr) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to initialize digest");
}
if (EVP_DigestUpdate(mdctx, input.c_str(), input.length()) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to update digest");
}
if (EVP_DigestFinal_ex(mdctx, digest, &digest_len) != 1) {
EVP_MD_CTX_free(mdctx);
throw std::runtime_error("Failed to finalize digest");
}
EVP_MD_CTX_free(mdctx);
std::stringstream ss;
for(unsigned int i = 0; i < digest_len; i++) {
ss << std::hex << std::setw(2) << std::setfill('0') << (int)digest[i];
}
return ss.str();
}

View File

@@ -3,6 +3,10 @@
#include "../include/MD5Converter.hpp" #include "../include/MD5Converter.hpp"
#include "../include/SHA1Converter.hpp" #include "../include/SHA1Converter.hpp"
#include "../include/SHA256Converter.hpp" #include "../include/SHA256Converter.hpp"
#include "../include/SHA224Converter.hpp"
#include "../include/SHA384Converter.hpp"
#include "../include/SHA512Converter.hpp"
#include "../include/SHA3Converter.hpp"
#include "../include/Base64Converter.hpp" #include "../include/Base64Converter.hpp"
#include "../include/Base32Converter.hpp" #include "../include/Base32Converter.hpp"
#include "../include/Ascii85Converter.hpp" #include "../include/Ascii85Converter.hpp"
@@ -20,6 +24,10 @@ std::unique_ptr<Converter> Utils::createConverter(int type) {
case 7: return std::unique_ptr<Converter>(new Base32Converter()); case 7: return std::unique_ptr<Converter>(new Base32Converter());
case 8: return std::unique_ptr<Converter>(new Ascii85Converter()); case 8: return std::unique_ptr<Converter>(new Ascii85Converter());
case 9: return std::unique_ptr<Converter>(new CRC32Converter()); case 9: return std::unique_ptr<Converter>(new CRC32Converter());
case 10: return std::unique_ptr<Converter>(new SHA224Converter());
case 11: return std::unique_ptr<Converter>(new SHA384Converter());
case 12: return std::unique_ptr<Converter>(new SHA512Converter());
case 13: return std::unique_ptr<Converter>(new SHA3Converter());
default: return nullptr; default: return nullptr;
} }
} }

View File

@@ -1,6 +1,6 @@
CC = gcc CC = gcc
CFLAGS = -Wall -Wextra -I./include CFLAGS = -Wall -Wextra -I./include
LDFLAGS = -Wl,-subsystem,windows -municode -luser32 -lgdi32 -lcomctl32 -lpsapi LDFLAGS = -static -Wl,-subsystem,windows -municode -luser32 -lgdi32 -lcomctl32 -lpsapi
SRC = src/main.c src/memory_ops.c src/ui.c src/init.c \ SRC = src/main.c src/memory_ops.c src/ui.c src/init.c \
src/utils.c src/config.c src/benchmark.c src/log.c \ src/utils.c src/config.c src/benchmark.c src/log.c \
src/error.c src/version.c src/error.c src/version.c

Binary file not shown.

Binary file not shown.

Binary file not shown.