news 2026/4/16 15:21:09

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/15 12:24:12

轻松搞定Apache Doris JDBC连接:Java应用集成实战指南

轻松搞定Apache Doris JDBC连接&#xff1a;Java应用集成实战指南 【免费下载链接】doris Apache Doris is an easy-to-use, high performance and unified analytics database. 项目地址: https://gitcode.com/gh_mirrors/dori/doris 还在为Java应用如何高效连接Apache…

作者头像 李华
网站建设 2026/4/16 12:51:50

Redis 零基础到进阶,Spring Boot 整合 Redis,笔记93-99

Redis 零基础到进阶&#xff0c;Spring Boot 整合 Redis&#xff0c;笔记93-99 一、参考资料 【尚硅谷Redis零基础到进阶&#xff0c;最强redis7教程&#xff0c;阳哥亲自带练&#xff08;附redis面试题&#xff09;】 https://www.bilibili.com/video/BV13R4y1v7sP/?p99&…

作者头像 李华
网站建设 2026/4/16 12:59:06

OkHttp终极网络请求库:10个高效开发技巧完全指南

OkHttp终极网络请求库&#xff1a;10个高效开发技巧完全指南 【免费下载链接】okhttp square/okhttp&#xff1a;这是一个基于Java的网络请求库&#xff0c;适合进行HTTP和HTTPS通信。特点包括高性能、易于使用、支持缓存和认证等。 项目地址: https://gitcode.com/gh_mirror…

作者头像 李华
网站建设 2026/4/16 13:07:43

Web动画新纪元:为什么说WebGL是未来交互的终极选择?

Web动画新纪元&#xff1a;为什么说WebGL是未来交互的终极选择&#xff1f; 【免费下载链接】lottie-web 项目地址: https://gitcode.com/gh_mirrors/lot/lottie-web 在Web技术快速迭代的今天&#xff0c;WebGL以其强大的图形渲染能力和硬件加速特性&#xff0c;正成为…

作者头像 李华
网站建设 2026/4/16 12:31:23

研究生必备:7款AI论文工具,开题报告到答辩全程助力!

如果你是正在为开题报告抓耳挠腮、被导师的“再改改”逼到深夜、查重一次花掉半周饭钱的研究生—— 请停下你疯狂敲击键盘的手&#xff0c;深呼吸3秒。你是否经历过这些绝望时刻&#xff1f; 开题报告写了3天&#xff0c;框架还被导师打回&#xff1a;“逻辑混乱&#xff0c;…

作者头像 李华
网站建设 2026/4/14 3:43:59

5大理由选择minimp3:轻量级MP3解码库的性能革命

5大理由选择minimp3&#xff1a;轻量级MP3解码库的性能革命 【免费下载链接】minimp3 Minimalistic MP3 decoder single header library 项目地址: https://gitcode.com/gh_mirrors/mi/minimp3 在音频应用开发领域&#xff0c;轻量级MP3解码库minimp3以其卓越的性能和极…

作者头像 李华