news 2026/4/26 17:44:32

c#造个轮子--GIF录制工具

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
c#造个轮子--GIF录制工具

在以往几篇文章里面,大家都可以看到各种录制的GIF效果图,把gif放在文章开始,不仅可以减少很多冗余的解释白话文,更可以让读者一览无余看到文章大概要义。

以往都是使用“LicEcap”来录制的,那么我们是否能自己实现一个这样的工具呢?一方面国庆假期结束,练练代码手感,另一方面可以根据自己需求扩展需要的功能。

01介绍软件UI及操作

操作比较简单,以下是运行界面:

选择录制区域,绘制需要录制的ROI区域

点击开始录制

录制结束后,停止录制即可.弹出保存路径保存gif

image.png

02效果图

整个运行作业图

test.gif

实际录屏的ROI区域效果GIF

eee.gif

03源码介绍

image

private void InitializeComponents()

{

this.Text = "GIF录制工具";

this.Size = new Size(400, 200);

this.StartPosition = FormStartPosition.CenterScreen;

// 选择区域按钮

Button btnSelectArea = new Button();

btnSelectArea.Text = "选择录制区域";

btnSelectArea.Size = new Size(120, 30);

btnSelectArea.Location = new Point(20, 20);

btnSelectArea.Click += BtnSelectArea_Click;

this.Controls.Add(btnSelectArea);

// 开始录制按钮

Button btnStart = new Button();

btnStart.Text = "开始录制";

btnStart.Size = new Size(120, 30);

btnStart.Location = new Point(20, 60);

btnStart.Click += BtnStart_Click;

this.Controls.Add(btnStart);

// 停止录制按钮

Button btnStop = new Button();

btnStop.Text = "停止录制";

btnStop.Size = new Size(120, 30);

btnStop.Location = new Point(20, 100);

btnStop.Click += BtnStop_Click;

this.Controls.Add(btnStop);

// 帧率选择

Label lblFrameRate = new Label();

lblFrameRate.Text = "帧率:";

lblFrameRate.Location = new Point(160, 65);

lblFrameRate.Size = new Size(50, 20);

this.Controls.Add(lblFrameRate);

NumericUpDown numFrameRate = new NumericUpDown();

numFrameRate.Value = frameRate;

numFrameRate.Minimum = 1;

numFrameRate.Maximum = 30;

numFrameRate.Location = new Point(210, 65);

numFrameRate.Size = new Size(60, 20);

numFrameRate.ValueChanged += (s, e) => { frameRate = (int)numFrameRate.Value; };

this.Controls.Add(numFrameRate);

// 状态标签

Label lblStatus = new Label();

lblStatus.Text = "状态: 就绪";

lblStatus.Location = new Point(160, 25);

lblStatus.Size = new Size(200, 20);

lblStatus.Name = "lblStatus";

this.Controls.Add(lblStatus);

// 录制计时器

captureTimer = new System.Windows.Forms.Timer();

captureTimer.Tick += CaptureTimer_Tick;

}

选择ROI录屏区域

private void StartAreaSelection()

{

this.Hide();

Thread.Sleep(500); // 等待窗体隐藏

isSelectingArea = true;

Cursor = Cursors.Cross;

// 创建全屏透明窗体用于区域选择

Form selectionForm = new Form();

selectionForm.WindowState = FormWindowState.Maximized;

selectionForm.FormBorderStyle = FormBorderStyle.None;

selectionForm.BackColor = Color.Black;

selectionForm.Opacity = 0.3;

selectionForm.TopMost = true;

selectionForm.Cursor = Cursors.Cross;

Rectangle selectedArea = Rectangle.Empty;

bool isDragging = false;

Point dragStart = Point.Empty;

selectionForm.MouseDown += (s, e) =>

{

if (e.Button == MouseButtons.Left)

{

isDragging = true;

dragStart = e.Location;

}

};

selectionForm.MouseMove += (s, e) =>

{

if (isDragging)

{

int x = Math.Min(dragStart.X, e.X);

int y = Math.Min(dragStart.Y, e.Y);

int width = Math.Abs(e.X - dragStart.X);

int height = Math.Abs(e.Y - dragStart.Y);

selectedArea = new Rectangle(x, y, width, height);

selectionForm.Invalidate();

}

};

selectionForm.MouseUp += (s, e) =>

{

if (e.Button == MouseButtons.Left && isDragging)

{

isDragging = false;

if (selectedArea.Width > 10 && selectedArea.Height > 10)

{

recordingArea = selectedArea;

UpdateStatus($"已选择区域: {recordingArea}");

}

selectionForm.Close();

}

};

selectionForm.Paint += (s, e) =>

{

if (isDragging && !selectedArea.IsEmpty)

{

using (Pen pen = new Pen(Color.Red, 2))

{

e.Graphics.DrawRectangle(pen, selectedArea);

}

string sizeText = $"{selectedArea.Width} x {selectedArea.Height}";

using (Font font = new Font("Arial", 12))

using (Brush brush = new SolidBrush(Color.Red))

{

e.Graphics.DrawString(sizeText, font, brush, selectedArea.X, selectedArea.Y - 20);

}

}

};

selectionForm.KeyDown += (s, e) =>

{

if (e.KeyCode == Keys.Escape)

{

selectionForm.Close();

}

};

selectionForm.FormClosed += (s, e) =>

{

isSelectingArea = false;

Cursor = Cursors.Default;

this.Show();

this.BringToFront();

};

selectionForm.ShowDialog();

}

录制结束,保存GIF

private void SaveGif()

{

if (frames.Count == 0)

{

MessageBox.Show("没有可保存的帧!", "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);

return;

}

using (SaveFileDialog saveDialog = new SaveFileDialog())

{

saveDialog.Filter = "GIF 文件|*.gif";

saveDialog.Title = "保存GIF文件";

saveDialog.DefaultExt = "gif";

if (saveDialog.ShowDialog() == DialogResult.OK)

{

try

{

// 使用GifBitmapEncoder替代方案

SaveFramesAsGif(frames, saveDialog.FileName, frameRate);

MessageBox.Show($"GIF保存成功!\n文件: {saveDialog.FileName}\n帧数: {frames.Count}", "成功",

MessageBoxButtons.OK, MessageBoxIcon.Information);

}

catch (Exception ex)

{

MessageBox.Show($"保存GIF时出错: {ex.Message}", "错误", MessageBoxButtons.OK, MessageBoxIcon.Error);

}

}

}

// 清理资源

foreach (var frame in frames)

{

frame.Dispose();

}

frames.Clear();

}

private void SaveFramesAsGif(List<Bitmap> frames, string filePath, int frameRate)

{

using (var collection = new MagickImageCollection())

{

foreach (var frame in frames)

{

using (var memoryStream = new MemoryStream())

{

frame.Save(memoryStream, ImageFormat.Bmp);

memoryStream.Position = 0;

var image = new MagickImage(memoryStream);

image.AnimationDelay =Convert.ToUInt32( 100 / frameRate); // 设置帧延迟

collection.Add(image);

}

}

// 优化GIF

collection.Optimize();

collection.Write(filePath);

}

}

主要用到第三方nuget包

AnimatedGif

Magick.NET-Q16-AnyCPU

结束语

感谢各位耐心查阅! 如果您有更好的想法欢迎一起交流,有不懂的也可以微信公众号联系博主,作者公众号会经常发一些实用的小工具和demo源码,需要的可以去看看!另外,如果觉得本篇博文对您或者身边朋友有帮助的,麻烦点

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 11:52:57

人工智能训练效率革命:Modded-NanoGPT的技术突破与实践路径

在人工智能训练成本日益攀升的背景下&#xff0c;传统模型训练面临着能效比低、训练周期长的双重挑战。Modded-NanoGPT项目通过算法创新与系统优化&#xff0c;实现了GPT-2&#xff08;124M参数&#xff09;级别模型训练时间从45分钟压缩至2.86分钟的突破性进展&#xff0c;同时…

作者头像 李华
网站建设 2026/4/23 13:24:35

终极指南:用Python实现Xcode项目自动化管理

终极指南&#xff1a;用Python实现Xcode项目自动化管理 【免费下载链接】mod-pbxproj A python module to manipulate XCode projects 项目地址: https://gitcode.com/gh_mirrors/mo/mod-pbxproj 还在为手动配置Xcode项目而烦恼吗&#xff1f;pbxproj模块正是你需要的解…

作者头像 李华
网站建设 2026/4/23 7:55:53

PubMedBERT语义嵌入革命:生物医学AI的精准向量化突破

PubMedBERT语义嵌入革命&#xff1a;生物医学AI的精准向量化突破 【免费下载链接】pubmedbert-base-embeddings 项目地址: https://ai.gitcode.com/hf_mirrors/NeuML/pubmedbert-base-embeddings 当我们面对海量的生物医学文献时&#xff0c;传统的关键词搜索往往无法捕…

作者头像 李华
网站建设 2026/4/25 20:13:05

大模型开发者必读:Agentic RAG与传统RAG技术对比与选择,建议收藏

文章对比了传统RAG与Agentic RAG技术的核心差异。传统RAG采用简单线性架构&#xff0c;静态检索策略&#xff0c;响应快但准确性有限&#xff1b;Agentic RAG引入智能代理系统&#xff0c;支持动态检索、多步推理和信息验证&#xff0c;准确性高但成本较高。前者适合简单查询和…

作者头像 李华
网站建设 2026/4/25 9:19:41

【收藏级】AI智能体时代的记忆:构建下一代强人工智能的核心蓝图

这篇综述文章系统梳理了AI智能体记忆机制&#xff0c;从形式(符号级、参数化、潜在记忆)、功能(事实、经验、工作记忆)和动态机制(形成、演化、检索)三个维度构建统一分类体系。记忆是AI从"静态回答者"进化为"动态成长智能体"的关键&#xff0c;文章探讨了…

作者头像 李华
网站建设 2026/4/23 7:55:54

Continue:重新定义AI辅助编程的技术架构与设计哲学

Continue&#xff1a;重新定义AI辅助编程的技术架构与设计哲学 【免费下载链接】continue ⏩ Continue is an open-source autopilot for VS Code and JetBrains—the easiest way to code with any LLM 项目地址: https://gitcode.com/GitHub_Trending/co/continue 在A…

作者头像 李华