mirror of
https://github.com/ShareX/ShareX.git
synced 2024-10-05 20:54:31 +13:00
238 lines
No EOL
6.7 KiB
C#
238 lines
No EOL
6.7 KiB
C#
#region License Information (GPL v3)
|
|
|
|
/*
|
|
ShareX - A program that allows you to take screenshots and share any file type
|
|
Copyright © 2007-2015 ShareX Developers
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
Optionally you can also view the license at <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#endregion License Information (GPL v3)
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Drawing;
|
|
using System.IO;
|
|
|
|
namespace ShareX.HelpersLib
|
|
{
|
|
public class GifClass
|
|
{
|
|
public enum GIFVersion
|
|
{
|
|
GIF87a,
|
|
GIF89a
|
|
}
|
|
|
|
public enum GIFBlockType
|
|
{
|
|
ImageDescriptor = 0x2C,
|
|
Extension = 0x21,
|
|
Trailer = 0x3B
|
|
}
|
|
|
|
public GIFVersion Version = GIFVersion.GIF87a;
|
|
public List<byte> GifSignature = new List<byte>();
|
|
public List<byte> ScreenDescriptor = new List<byte>();
|
|
public List<byte> ColorTable = new List<byte>();
|
|
public List<byte> ImageDescriptor = new List<byte>();
|
|
public List<byte> ImageData = new List<byte>();
|
|
|
|
public void LoadGifPicture(Image img, GIFQuality quality)
|
|
{
|
|
List<byte> dataList;
|
|
|
|
using (MemoryStream ms = new MemoryStream())
|
|
{
|
|
img.SaveGIF(ms, quality);
|
|
dataList = new List<byte>(ms.ToArray());
|
|
}
|
|
|
|
if (!AnalyzeGifSignature(dataList))
|
|
{
|
|
throw new Exception("File is not a gif!");
|
|
}
|
|
|
|
AnalyzeScreenDescriptor(dataList);
|
|
|
|
GIFBlockType blockType = GetTypeOfNextBlock(dataList);
|
|
|
|
while (blockType != GIFBlockType.Trailer)
|
|
{
|
|
switch (blockType)
|
|
{
|
|
case GIFBlockType.ImageDescriptor:
|
|
AnalyzeImageDescriptor(dataList);
|
|
break;
|
|
case GIFBlockType.Extension:
|
|
ThrowAwayExtensionBlock(dataList);
|
|
break;
|
|
}
|
|
|
|
blockType = GetTypeOfNextBlock(dataList);
|
|
}
|
|
}
|
|
|
|
private bool AnalyzeGifSignature(List<byte> gifData)
|
|
{
|
|
for (int i = 0; i < 6; i++)
|
|
{
|
|
GifSignature.Add(gifData[i]);
|
|
}
|
|
|
|
gifData.RemoveRange(0, 6);
|
|
|
|
List<char> chars = GifSignature.ConvertAll(ByteToChar);
|
|
|
|
string s = new string(chars.ToArray());
|
|
|
|
if (s == GIFVersion.GIF89a.ToString())
|
|
{
|
|
Version = GIFVersion.GIF89a;
|
|
}
|
|
else if (s == GIFVersion.GIF87a.ToString())
|
|
{
|
|
Version = GIFVersion.GIF87a;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private char ByteToChar(byte b)
|
|
{
|
|
return (char)b;
|
|
}
|
|
|
|
private void AnalyzeScreenDescriptor(List<byte> gifData)
|
|
{
|
|
for (int i = 0; i < 7; i++)
|
|
{
|
|
ScreenDescriptor.Add(gifData[i]);
|
|
}
|
|
|
|
gifData.RemoveRange(0, 7);
|
|
|
|
// if the first bit of the fifth byte is set the GlobelColorTable follows this block
|
|
|
|
bool globalColorTableFollows = (ScreenDescriptor[4] & 0x80) != 0;
|
|
|
|
if (globalColorTableFollows)
|
|
{
|
|
int pixel = ScreenDescriptor[4] & 0x07;
|
|
|
|
int lengthOfColorTableInByte = 3 * ((int)Math.Pow(2, pixel + 1));
|
|
|
|
for (int i = 0; i < lengthOfColorTableInByte; i++)
|
|
{
|
|
ColorTable.Add(gifData[i]);
|
|
}
|
|
|
|
gifData.RemoveRange(0, lengthOfColorTableInByte);
|
|
}
|
|
|
|
ScreenDescriptor[4] = (byte)(ScreenDescriptor[4] & 0x7F);
|
|
}
|
|
|
|
private GIFBlockType GetTypeOfNextBlock(List<byte> gifData)
|
|
{
|
|
GIFBlockType blockType = (GIFBlockType)gifData[0];
|
|
|
|
return blockType;
|
|
}
|
|
|
|
private void AnalyzeImageDescriptor(List<byte> gifData)
|
|
{
|
|
for (int i = 0; i < 10; i++)
|
|
{
|
|
ImageDescriptor.Add(gifData[i]);
|
|
}
|
|
|
|
gifData.RemoveRange(0, 10);
|
|
|
|
// get ColorTable if exists
|
|
|
|
bool localColorMapFollows = (ImageDescriptor[9] & 0x80) != 0;
|
|
|
|
if (localColorMapFollows)
|
|
{
|
|
int pixel = ImageDescriptor[9] & 0x07;
|
|
|
|
int lengthOfColorTableInByte = 3 * ((int)Math.Pow(2, pixel + 1));
|
|
|
|
ColorTable.Clear();
|
|
|
|
for (int i = 0; i < lengthOfColorTableInByte; i++)
|
|
{
|
|
ColorTable.Add(gifData[i]);
|
|
}
|
|
|
|
gifData.RemoveRange(0, lengthOfColorTableInByte);
|
|
}
|
|
else
|
|
{
|
|
int lastThreeBitsOfGlobalTableDescription = ScreenDescriptor[4] & 0x07;
|
|
|
|
ImageDescriptor[9] = (byte)(ImageDescriptor[9] & 0xF8);
|
|
|
|
ImageDescriptor[9] = (byte)(ImageDescriptor[9] | lastThreeBitsOfGlobalTableDescription);
|
|
}
|
|
|
|
ImageDescriptor[9] = (byte)(ImageDescriptor[9] | 0x80);
|
|
|
|
GetImageData(gifData);
|
|
}
|
|
|
|
private void GetImageData(List<byte> gifData)
|
|
{
|
|
ImageData.Add(gifData[0]);
|
|
|
|
gifData.RemoveAt(0);
|
|
|
|
while (gifData[0] != 0x00)
|
|
{
|
|
int countOfFollowingDataBytes = gifData[0];
|
|
|
|
for (int i = 0; i <= countOfFollowingDataBytes; i++)
|
|
{
|
|
ImageData.Add(gifData[i]);
|
|
}
|
|
|
|
gifData.RemoveRange(0, countOfFollowingDataBytes + 1);
|
|
}
|
|
|
|
ImageData.Add(gifData[0]);
|
|
|
|
gifData.RemoveAt(0);
|
|
}
|
|
|
|
private void ThrowAwayExtensionBlock(List<byte> gifData)
|
|
{
|
|
gifData.RemoveRange(0, 2); // Delete ExtensionBlockIndicator and ExtensionDetermination
|
|
|
|
while (gifData[0] != 0)
|
|
{
|
|
gifData.RemoveRange(0, gifData[0] + 1);
|
|
}
|
|
|
|
gifData.RemoveAt(0);
|
|
}
|
|
}
|
|
} |