1
0
mirror of synced 2025-01-26 07:53:47 +01:00

233 lines
7.7 KiB
C#
Raw Normal View History

2019-08-04 14:15:34 +01:00
using System;
using System.Collections.Generic;
using System.Windows.Forms;
using Toolbox.Library;
using Toolbox.Library.Forms;
using Toolbox.Library.IO;
2019-08-08 19:58:14 -04:00
using System.Linq;
using System.Runtime.InteropServices;
2019-08-04 14:15:34 +01:00
namespace FirstPlugin
2019-08-04 14:15:34 +01:00
{
class TXE : STGenericTexture, IFileFormat, ISingleTextureIconLoader
2019-08-04 14:15:34 +01:00
{
public STGenericTexture IconTexture { get { return this; } }
2019-08-04 14:15:34 +01:00
public FileType FileType { get; set; } = FileType.Image;
public bool CanSave { get; set; }
public string[] Description { get; set; } = new string[] { "Pikmin 1 Proprietary Texture" };
public string[] Extension { get; set; } = new string[] { "*.txe" };
public string FileName { get; set; }
public string FilePath { get; set; }
//Stores compression info from being opened (done automaitcally)
public IFileInfo IFileInfo { get; set; }
//Check how the file wil be opened
public bool Identify(System.IO.Stream stream)
{
return Utils.HasExtension(FileName, ".txe");
}
//A Type list for custom types
//With this you can add in classes with IFileMenuExtension to add menus for this format
public Type[] Types
{
get
{
List<Type> types = new List<Type>();
return types.ToArray();
}
}
public static Dictionary<ushort, TEX_FORMAT> FormatsTXE = new Dictionary<ushort, TEX_FORMAT>()
{
[0] = TEX_FORMAT.RGB5A3,
[1] = TEX_FORMAT.CMPR,
[2] = TEX_FORMAT.RGB565,
[3] = TEX_FORMAT.I4,
[4] = TEX_FORMAT.I8,
[5] = TEX_FORMAT.IA4,
[6] = TEX_FORMAT.IA8,
[7] = TEX_FORMAT.RGBA32,
};
2019-08-08 19:58:14 -04:00
public short Unknown;
public void Load(System.IO.Stream stream)
2019-08-04 14:15:34 +01:00
{
Text = FileName;
//Set this if you want to save the file format
2019-08-08 19:58:14 -04:00
CanSave = true;
CanReplace = true;
ImageKey = "Texture";
SelectedImageKey = "Texture";
2019-08-04 14:15:34 +01:00
//You can add a FileReader with Toolbox.Library.IO namespace
using (var reader = new FileReader(stream))
{
reader.SetByteOrder(true);
Width = reader.ReadUInt16();
Height = reader.ReadUInt16();
2019-08-08 19:58:14 -04:00
Unknown = reader.ReadInt16();
2019-08-04 14:15:34 +01:00
//Turn this format into a common format used by this tool
ushort texFormat = reader.ReadUInt16();
Format = FormatsTXE[texFormat];
int ImageSize = reader.ReadInt32(); //Unsure
2019-08-04 14:15:34 +01:00
//Lets set our method of decoding
PlatformSwizzle = PlatformSwizzle.Platform_Gamecube;
2019-08-04 14:15:34 +01:00
reader.Align(32);
2019-08-04 14:15:34 +01:00
//Calculate size automatically if ImageSize is 0
if (ImageSize == 0)
ImageSize = (int)reader.BaseStream.Length - (int)reader.Position;
2019-08-04 14:44:05 +01:00
ImageData = reader.ReadBytes(ImageSize);
2019-08-04 14:44:05 +01:00
Text = FileName;
}
2019-08-04 14:44:05 +01:00
}
public void Save(System.IO.Stream stream)
2019-08-04 14:15:34 +01:00
{
2019-08-08 19:58:14 -04:00
using (var writer = new FileWriter(stream))
{
writer.Write((ushort)Width);
writer.Write((ushort)Height);
writer.Write(Unknown);
ushort format = FormatsTXE.FirstOrDefault(x => x.Value == Format).Key;
writer.Write(format);
writer.Write(ImageData.Length);
writer.Align(32);
2019-08-08 19:58:14 -04:00
writer.Write(ImageData);
}
2019-08-04 14:15:34 +01:00
}
public void Unload()
{
}
public byte[] ImageData { get; set; }
2019-08-04 14:15:34 +01:00
//A list of supported formats
//This gets used in the re encode option
public override TEX_FORMAT[] SupportedFormats
{
get
2019-08-04 14:15:34 +01:00
{
return new TEX_FORMAT[]
2019-08-04 14:15:34 +01:00
{
TEX_FORMAT.I4,
TEX_FORMAT.I8,
2019-08-04 14:44:05 +01:00
TEX_FORMAT.IA4,
TEX_FORMAT.IA8,
2019-08-04 14:15:34 +01:00
TEX_FORMAT.RGB565,
TEX_FORMAT.RGB5A3,
TEX_FORMAT.RGBA32,
TEX_FORMAT.CMPR,
};
2019-08-04 14:15:34 +01:00
}
}
2019-08-04 14:15:34 +01:00
2019-08-08 18:04:15 -04:00
public Decode_Gamecube.TextureFormats DolphinTextureFormat
{
get { return Decode_Gamecube.FromGenericFormat(Format); }
}
2019-08-08 19:58:14 -04:00
public override bool CanEdit { get; set; } = true;
2019-08-04 14:15:34 +01:00
//This gets used in the image editor if the image gets edited
//This wll not be ran if "CanEdit" is set to false!
public override void SetImageData(System.Drawing.Bitmap bitmap, int ArrayLevel)
{
2019-08-08 19:58:14 -04:00
this.Width = (ushort)bitmap.Width;
this.Height = (ushort)bitmap.Height;
2019-08-04 14:15:34 +01:00
2019-08-08 19:58:14 -04:00
this.ImageData = Decode_Gamecube.EncodeFromBitmap(bitmap, DolphinTextureFormat).Item1;
}
//Gets the raw image data in bytes
//Gets decoded automatically
public override byte[] GetImageData(int ArrayLevel = 0, int MipLevel = 0)
{
2019-08-08 19:58:14 -04:00
return Decode_Gamecube.GetMipLevel(ImageData, Width, Height, MipCount, (uint)MipLevel, Format);
}
2019-08-04 14:15:34 +01:00
//This is an event for when the tree is clicked on
//Load our editor
public override void OnClick(TreeView treeView) {
UpdateEditor();
}
public void UpdateEditor()
{
//Here we check for an active editor and load a new one if not found
//This is used when a tree/object editor is used
ImageEditorBase editor = (ImageEditorBase)LibraryGUI.GetActiveContent(typeof(ImageEditorBase));
if (editor == null)
2019-08-04 14:15:34 +01:00
{
editor = new ImageEditorBase();
editor.Dock = DockStyle.Fill;
LibraryGUI.LoadEditor(editor);
2019-08-04 14:15:34 +01:00
}
//Load our image and any properties
//If you don't make a class for properties you can use a generic class provided in STGenericTexture
editor.LoadProperties(GenericProperties);
editor.LoadImage(this);
2019-08-04 14:15:34 +01:00
}
public override void Replace(string FileName)
{
GamecubeTextureImporterList importer = new GamecubeTextureImporterList(SupportedFormats);
GameCubeTextureImporterSettings settings = new GameCubeTextureImporterSettings();
2019-08-08 19:58:14 -04:00
importer.ForceMipCount = true;
importer.SelectedMipCount = 1;
if (Utils.GetExtension(FileName) == ".dds" ||
Utils.GetExtension(FileName) == ".dds2")
{
settings.LoadDDS(FileName);
importer.LoadSettings(new List<GameCubeTextureImporterSettings>() { settings, });
ApplySettings(settings);
UpdateEditor();
}
else
{
settings.LoadBitMap(FileName);
importer.LoadSettings(new List<GameCubeTextureImporterSettings>() { settings, });
if (importer.ShowDialog() == DialogResult.OK)
{
if (settings.GenerateMipmaps && !settings.IsFinishedCompressing)
settings.Compress();
ApplySettings(settings);
UpdateEditor();
}
}
}
private void ApplySettings(GameCubeTextureImporterSettings settings)
{
this.ImageData = settings.DataBlockOutput[0];
this.Width = settings.TexWidth;
this.Height = settings.TexHeight;
this.Format = settings.GenericFormat;
2019-08-08 19:58:14 -04:00
this.MipCount = 1; //Always 1
this.Depth = settings.Depth;
this.ArrayCount = (uint)settings.DataBlockOutput.Count;
}
2019-08-04 14:15:34 +01:00
}
}