using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Shapes;
using System.IO;
using System.Windows.Navigation;
using System.Diagnostics;
using Ookii.Dialogs.Wpf;
using System.Threading;
using System.Windows.Threading;
using LibCPK;
namespace CriPakGUI
{
///
/// CpkPatcher.xaml 的交互逻辑
///
public partial class CpkPatcher : Window
{
public CpkPatcher(double x, double y)
{
InitializeComponent();
this.WindowStartupLocation = WindowStartupLocation.Manual;
this.Top = x;
this.Left = y;
}
private void button_selPatchPath_Click(object sender, RoutedEventArgs e)
{
VistaFolderBrowserDialog saveFilesDialog = new VistaFolderBrowserDialog();
saveFilesDialog.SelectedPath = myPackage.basePath + "/";
if (saveFilesDialog.ShowDialog().Value)
{
Debug.Print(saveFilesDialog.SelectedPath);
textbox_patchDir.Text = saveFilesDialog.SelectedPath;
}
}
private void button_selDstCPK_Click(object sender, RoutedEventArgs e)
{
VistaSaveFileDialog saveDialog = new VistaSaveFileDialog();
saveDialog.InitialDirectory = myPackage.basePath;
saveDialog.RestoreDirectory = true;
saveDialog.Filter = "CPK File(*.cpk)|*.cpk";
if (saveDialog.ShowDialog() == true)
{
string saveFileName = saveDialog.FileName;
textbox_cpkDir.Text = saveFileName;
}
}
private delegate void textblockDelegate(string text);
private void updateTextblock(string text)
{
textblock0.Text += string.Format("Updating ... {0}\n", text);
scrollview0.ScrollToEnd();
}
private delegate void progressbarDelegate(float no);
private void updateprogressbar(float no)
{
progressbar1.Value = no;
}
public class actionCPK
{
public string cpkDir { get; set; }
public string patchDir { get; set; }
public bool bForceCompress { get; set; }
public Dictionary batch_file_list { get; set; }
}
private void button_PatchCPK_Click(object sender, RoutedEventArgs e)
{
string cpkDir = textbox_cpkDir.Text;
string patchDir = textbox_patchDir.Text;
Dictionary batch_file_list = new Dictionary();
List ls = new List();
if ((myPackage.cpk != null) && (Directory.Exists(patchDir)))
{
GetFilesFromPath(patchDir, ref ls);
Debug.Print(string.Format("GOT {0} Files.", ls.Count));
foreach (string s in ls)
{
string name = s.Remove(0, patchDir.Length + 1);
name = name.Replace("\\" , @"/");
if (!name.Contains(@"/"))
{
name = @"/" + name;
}
batch_file_list.Add(name, s);
}
actionCPK t = new actionCPK();
t.cpkDir = cpkDir;
t.patchDir = patchDir;
if (checkbox_donotcompress.IsChecked == true)
{
t.bForceCompress = false;
}
else
{
t.bForceCompress = true;
}
t.batch_file_list = batch_file_list;
ThreadPool.QueueUserWorkItem(new WaitCallback(PatchCPK), t);
}
else
{
MessageBox.Show("Error, cpkdata or patchdata not found.");
}
}
private void PatchCPK(object t)
{
string msg;
string cpkDir = ((actionCPK)t).cpkDir;
string patchDir = ((actionCPK)t).patchDir;
bool bForceCompress = ((actionCPK)t).bForceCompress;
Dictionary batch_file_list = ((actionCPK)t).batch_file_list;
CPK cpk = myPackage.cpk;
BinaryReader oldFile = new BinaryReader(File.OpenRead(myPackage.cpk_name));
string outputName = cpkDir;
BinaryWriter newCPK = new BinaryWriter(File.OpenWrite(outputName));
List entries = cpk.FileTable.OrderBy(x => x.FileOffset).ToList();
Tools tool = new Tools();
int id;
bool bFileRepeated = Tools.CheckListRedundant(entries);
for (int i = 0; i < entries.Count; i++)
{
this.UI_SetProgess((float)i / (float)entries.Count * 100f);
if (entries[i].FileType != "CONTENT")
{
if (entries[i].FileType == "FILE")
{
// I'm too lazy to figure out how to update the ContextOffset position so this works :)
if ((ulong)newCPK.BaseStream.Position < cpk.ContentOffset)
{
ulong padLength = cpk.ContentOffset - (ulong)newCPK.BaseStream.Position;
for (ulong z = 0; z < padLength; z++)
{
newCPK.Write((byte)0);
}
}
}
id = Convert.ToInt32(entries[i].ID);
string currentName;
if (id > 0 && bFileRepeated)
{
currentName = (((entries[i].DirName != null) ?
entries[i].DirName + "/" : "") + string.Format("[{0}]", id.ToString()) + entries[i].FileName);
}
else
{
currentName = ((entries[i].DirName != null) ? entries[i].DirName + "/" : "") + entries[i].FileName;
}
if (!currentName.Contains("/"))
{
currentName = "/" + currentName;
}
Debug.Print("Got File:" + currentName.ToString());
if (!batch_file_list.Keys.Contains(currentName.ToString()))
//如果不在表中,复制原始数据
{
oldFile.BaseStream.Seek((long)entries[i].FileOffset, SeekOrigin.Begin);
entries[i].FileOffset = (ulong)newCPK.BaseStream.Position;
if (entries[i].FileName.ToString() == "ETOC_HDR")
{
cpk.EtocOffset = entries[i].FileOffset;
Debug.Print("Fix ETOC_OFFSET to {0:x8}", cpk.EtocOffset);
}
cpk.UpdateFileEntry(entries[i]);
byte[] chunk = oldFile.ReadBytes(Int32.Parse(entries[i].FileSize.ToString()));
newCPK.Write(chunk);
if ((newCPK.BaseStream.Position % 0x800) > 0 && i < entries.Count - 1)
{
long cur_pos = newCPK.BaseStream.Position;
for (int j = 0; j < (0x800 - (cur_pos % 0x800)); j++)
{
newCPK.Write((byte)0);
}
}
}
else
{
string replace_with = batch_file_list[currentName.ToString()];
//Got patch file name
msg = string.Format("Patching: {0}", currentName.ToString());
this.UI_SetTextBlock(msg);
Debug.Print(msg);
byte[] newbie = File.ReadAllBytes(replace_with);
entries[i].FileOffset = (ulong)newCPK.BaseStream.Position;
int o_ext_size = Int32.Parse((entries[i].ExtractSize).ToString());
int o_com_size = Int32.Parse((entries[i].FileSize).ToString());
if ((o_com_size < o_ext_size) && entries[i].FileType == "FILE" && bForceCompress == true)
{
// is compressed
msg = string.Format("Compressing data:{0:x8}", newbie.Length);
this.UI_SetTextBlock(msg);
Console.Write(msg);
byte[] dest_comp = cpk.CompressCRILAYLA(newbie);
entries[i].FileSize = Convert.ChangeType(dest_comp.Length, entries[i].FileSizeType);
entries[i].ExtractSize = Convert.ChangeType(newbie.Length, entries[i].FileSizeType);
cpk.UpdateFileEntry(entries[i]);
newCPK.Write(dest_comp);
msg = string.Format(">> {0:x8}\r\n", dest_comp.Length);
this.UI_SetTextBlock(msg);
Console.Write(msg);
}
else
{
msg = string.Format("Storing data:{0:x8}\r\n", newbie.Length);
this.UI_SetTextBlock(msg);
Console.Write(msg);
entries[i].FileSize = Convert.ChangeType(newbie.Length, entries[i].FileSizeType);
entries[i].ExtractSize = Convert.ChangeType(newbie.Length, entries[i].FileSizeType);
cpk.UpdateFileEntry(entries[i]);
newCPK.Write(newbie);
}
if ((newCPK.BaseStream.Position % 0x800) > 0 && i < entries.Count - 1)
{
long cur_pos = newCPK.BaseStream.Position;
for (int j = 0; j < (0x800 - (cur_pos % 0x800)); j++)
{
newCPK.Write((byte)0);
}
}
}
}
else
{
// Content is special.... just update the position
cpk.UpdateFileEntry(entries[i]);
}
}
cpk.WriteCPK(newCPK);
msg = string.Format("Writing TOC....");
this.UI_SetTextBlock(msg);
Console.WriteLine(msg);
cpk.WriteITOC(newCPK);
cpk.WriteTOC(newCPK);
cpk.WriteETOC(newCPK, cpk.EtocOffset);
cpk.WriteGTOC(newCPK);
newCPK.Close();
oldFile.Close();
msg = string.Format("Saving CPK to {0}....", outputName);
this.UI_SetTextBlock(msg);
Console.WriteLine(msg);
MessageBox.Show("CPK Patched.");
this.UI_SetProgess(0f);
}
public void UI_SetProgess(float value)
{
this.Dispatcher.Invoke(new progressbarDelegate(updateprogressbar), new object[] { (float)value });
}
public void UI_SetTextBlock(string msg)
{
this.Dispatcher.Invoke(new textblockDelegate(updateTextblock), new object[] { msg });
}
private void GetFilesFromPath(string directoryname , ref List ls)
{
FileInfo[] fi = new DirectoryInfo(directoryname).GetFiles();
DirectoryInfo[] di = new DirectoryInfo(directoryname).GetDirectories();
if (fi.Length != 0)
{
foreach (FileInfo v in fi)
{
ls.Add(v.FullName);
}
}
if (di.Length != 0)
{
foreach (DirectoryInfo v in di)
{
GetFilesFromPath(v.FullName , ref ls);
}
}
}
}
}