diff --git a/DatSplit/App.config b/DatSplit/App.config new file mode 100644 index 00000000..88fa4027 --- /dev/null +++ b/DatSplit/App.config @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/DatSplit/DatSplit.cs b/DatSplit/DatSplit.cs new file mode 100644 index 00000000..5c2fa110 --- /dev/null +++ b/DatSplit/DatSplit.cs @@ -0,0 +1,182 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Text; +using System.Xml; +using System.Xml.Linq; + +using SabreTools.Helper; + +namespace DatSplit +{ + class DatSplit + { + private static string extA; + private static string extB; + private static string filename; + + public static void Main(string[] args) + { + // If we don't have arguments, show help + if (args.Length == 0 && args.Length != 3) + { + Help(); + return; + } + + // Set needed strings + filename = args[0]; + extA = args[1]; + extB = args[2]; + + // Take the filename, and load it as an XML document + XmlDocument doc = new XmlDocument(); + try + { + doc.LoadXml(File.ReadAllText(filename)); + } + catch (XmlException ex) + { + doc.LoadXml(Converters.RomVaultToXML(File.ReadAllLines(filename)).ToString()); + } + + // We all start the same + XmlNode node = doc.FirstChild; + if (node != null && node.Name == "xml") + { + // Skip over everything that's not an element + while (node.NodeType != XmlNodeType.Element) + { + node = node.NextSibling; + } + } + + XmlDocument tempDoc = new XmlDocument(); + XmlNode outA = tempDoc.CreateNode(XmlNodeType.Element, node.Name, ""); + XmlNode outB = tempDoc.CreateNode(XmlNodeType.Element, node.Name, ""); + + // Once we find the main body, enter it + if (node != null && node.Name == "datafile") + { + node = node.FirstChild; + } + + // Now here's where it differs from import + while (node != null) + { + // If we're at a game node, add the parent node but not all the internals + if (node.NodeType == XmlNodeType.Element && (node.Name == "machine" || node.Name == "game")) + { + bool inA = false; + bool inB = false; + + // Get the roms from the machine + if (node.HasChildNodes) + { + // If this node has children, traverse the children + foreach (XmlNode child in node.ChildNodes) + { + // If we find a rom or disk, add it + if (child.NodeType == XmlNodeType.Element && (child.Name == "rom" || child.Name == "disk")) + { + // Take care of hex-sized files + long size = -1; + if (child.Attributes["size"] != null && child.Attributes["size"].Value.Contains("0x")) + { + size = Convert.ToInt64(child.Attributes["size"].Value, 16); + } + else if (child.Attributes["size"] != null) + { + size = Int64.Parse(child.Attributes["size"].Value); + } + + if (child.Attributes["name"].Value.EndsWith(extA)) + { + if (!inA) + { + //XmlNode temp = tempDoc.CreateNode(XmlNodeType.Element, node.Name, ""); + XmlNode temp = tempDoc.ImportNode(node, false); + outA.AppendChild(temp); + outA = outA.LastChild; + inA = true; + } + outA.AppendChild(tempDoc.ImportNode(child, true)); + } + else if (child.Attributes["name"].Value.EndsWith(extB)) + { + if (!inB) + { + //XmlNode temp = tempDoc.CreateNode(XmlNodeType.Element, node.Name, ""); + XmlNode temp = tempDoc.ImportNode(node, false); + outB.AppendChild(temp); + outB = outB.LastChild; + inB = true; + } + outB.AppendChild(tempDoc.ImportNode(child, true)); + } + else + { + outA.AppendChild(tempDoc.ImportNode(child, true)); + outB.AppendChild(tempDoc.ImportNode(child, true)); + } + } + } + + // Set the output node to the right one for both + if (inA) + { + outA = outA.ParentNode; + } + if (inB) + { + outB = outB.ParentNode; + } + } + } + else + { + XmlNode tempNode = tempDoc.ImportNode(node, true); + outA.AppendChild(tempNode); + tempNode = tempDoc.ImportNode(node, true); + outB.AppendChild(tempNode); + } + node = node.NextSibling; + } + + XmlDocument outDocA = new XmlDocument(); + outDocA.AppendChild(outDocA.CreateDocumentType("datafile", "-//Logiqx//DTD ROM Management Datafile//EN", "http://www.logiqx.com/Dats/datafile.dtd", null)); + outDocA.AppendChild(outDocA.ImportNode(outA, true)); + string outPathA = Path.GetFileNameWithoutExtension(filename) + extA + Path.GetExtension(filename); + File.WriteAllText(outPathA, Beautify(outDocA)); + + XmlDocument outDocB = new XmlDocument(); + outDocB.AppendChild(outDocB.CreateDocumentType("datafile", "-//Logiqx//DTD ROM Management Datafile//EN", "http://www.logiqx.com/Dats/datafile.dtd", null)); + outDocB.AppendChild(outDocB.ImportNode(outB, true)); + string outPathB = Path.GetFileNameWithoutExtension(filename) + extB + Path.GetExtension(filename); + File.WriteAllText(outPathB, Beautify(outDocB)); + } + + public static void Help() + { + Console.WriteLine("DatSplit.exe "); + } + + // http://stackoverflow.com/questions/203528/what-is-the-simplest-way-to-get-indented-xml-with-line-breaks-from-xmldocument + static public string Beautify(XmlDocument doc) + { + StringBuilder sb = new StringBuilder(); + XmlWriterSettings settings = new XmlWriterSettings + { + Indent = true, + IndentChars = "\t", + NewLineChars = "\r\n", + NewLineHandling = NewLineHandling.Replace + }; + using (XmlWriter writer = XmlWriter.Create(sb, settings)) + { + doc.Save(writer); + } + return sb.ToString(); + } + } +} diff --git a/DatSplit/DatSplit.csproj b/DatSplit/DatSplit.csproj new file mode 100644 index 00000000..3b024aa3 --- /dev/null +++ b/DatSplit/DatSplit.csproj @@ -0,0 +1,66 @@ + + + + + Debug + AnyCPU + {9EB4738D-CAE7-420D-8A26-78A53CEA5E82} + Exe + Properties + DatSplit + DatSplit + v4.5.2 + 512 + true + + + AnyCPU + true + full + false + ..\..\Builds\ + DEBUG;TRACE + prompt + 4 + + + AnyCPU + pdbonly + true + bin\Release\ + TRACE + prompt + 4 + + + + + + + + + + + + + + + + + + + + + {3b615702-1866-4d7b-8af1-7b43fd0cc1d0} + DATabase + + + + + \ No newline at end of file diff --git a/DatSplit/Properties/AssemblyInfo.cs b/DatSplit/Properties/AssemblyInfo.cs new file mode 100644 index 00000000..93cee71c --- /dev/null +++ b/DatSplit/Properties/AssemblyInfo.cs @@ -0,0 +1,36 @@ +using System.Reflection; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; + +// General Information about an assembly is controlled through the following +// set of attributes. Change these attribute values to modify the information +// associated with an assembly. +[assembly: AssemblyTitle("DatSplit")] +[assembly: AssemblyDescription("")] +[assembly: AssemblyConfiguration("")] +[assembly: AssemblyCompany("")] +[assembly: AssemblyProduct("DatSplit")] +[assembly: AssemblyCopyright("Copyright © 2016")] +[assembly: AssemblyTrademark("")] +[assembly: AssemblyCulture("")] + +// Setting ComVisible to false makes the types in this assembly not visible +// to COM components. If you need to access a type in this assembly from +// COM, set the ComVisible attribute to true on that type. +[assembly: ComVisible(false)] + +// The following GUID is for the ID of the typelib if this project is exposed to COM +[assembly: Guid("9eb4738d-cae7-420d-8a26-78a53cea5e82")] + +// Version information for an assembly consists of the following four values: +// +// Major Version +// Minor Version +// Build Number +// Revision +// +// You can specify all the values or you can default the Build and Revision Numbers +// by using the '*' as shown below: +// [assembly: AssemblyVersion("1.0.*")] +[assembly: AssemblyVersion("1.0.0.0")] +[assembly: AssemblyFileVersion("1.0.0.0")] diff --git a/SabreTools.sln b/SabreTools.sln index beb29ed7..6c8ab127 100644 --- a/SabreTools.sln +++ b/SabreTools.sln @@ -9,6 +9,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Headerer", "Deheader\Header EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SabreToolsUI", "SabreToolsUI\SabreToolsUI.csproj", "{7DC54E53-4A46-4323-97E1-062EEFB7E4BC}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "DatSplit", "DatSplit\DatSplit.csproj", "{9EB4738D-CAE7-420D-8A26-78A53CEA5E82}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -27,6 +29,10 @@ Global {7DC54E53-4A46-4323-97E1-062EEFB7E4BC}.Debug|Any CPU.Build.0 = Debug|Any CPU {7DC54E53-4A46-4323-97E1-062EEFB7E4BC}.Release|Any CPU.ActiveCfg = Release|Any CPU {7DC54E53-4A46-4323-97E1-062EEFB7E4BC}.Release|Any CPU.Build.0 = Release|Any CPU + {9EB4738D-CAE7-420D-8A26-78A53CEA5E82}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {9EB4738D-CAE7-420D-8A26-78A53CEA5E82}.Debug|Any CPU.Build.0 = Debug|Any CPU + {9EB4738D-CAE7-420D-8A26-78A53CEA5E82}.Release|Any CPU.ActiveCfg = Release|Any CPU + {9EB4738D-CAE7-420D-8A26-78A53CEA5E82}.Release|Any CPU.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE