/* * Rufus: The Reliable USB Formatting Utility * Poedit <-> rufus.loc conversion utility * Copyright © 2018 Pete Batard * * 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 3 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, see . */ using System; using System.Collections.Generic; using System.ComponentModel; using System.IO; using System.Linq; using System.Net; using System.Reflection; using System.Text; using System.Text.RegularExpressions; using System.Threading; [assembly: AssemblyTitle("Pollock")] [assembly: AssemblyDescription("Poedit ↔ Rufus loc conversion utility")] [assembly: AssemblyCompany("Akeo Consulting")] [assembly: AssemblyProduct("Pollock")] [assembly: AssemblyCopyright("Copyright © 2018 Pete Batard ")] [assembly: AssemblyTrademark("GNU GPLv3")] [assembly: AssemblyVersion("1.0.*")] namespace pollock { public sealed class Message { public string id; public string str; public Message(string id, string str) { this.id = id; this.str = str; } } public sealed class Id { public string group; public string id; public Id(string group, string id) { this.group = group; this.id = id; } } public sealed class Language { public string id; public string name; public string version; public string lcid; public SortedDictionary> sections; public Dictionary comments; public Language() { sections = new SortedDictionary>(); comments = new Dictionary(); } } class Pollock { private static string app_name = System.Reflection.Assembly.GetExecutingAssembly().GetName().Name; private static string app_version = "v" + Assembly.GetEntryAssembly().GetName().Version.Major.ToString() + "." + Assembly.GetEntryAssembly().GetName().Version.Minor.ToString(); private static bool cancel_requested = false; private const string LANG_ID = "Language"; private const string LANG_NAME = "X-Rufus-LanguageName"; private const string LANG_VERSION = "Project-Id-Version"; private const string LANG_LCID = "X-Rufus-LCID"; private static Encoding encoding = new UTF8Encoding(false); private static List rtl_languages = new List { "ar-SA", "he-IL", "fa-IR" }; private static System.Diagnostics.Stopwatch sw = new System.Diagnostics.Stopwatch(); private static WebClient wc = new WebClient(); private static int download_status; private static bool in_progress = false; private static double speed = 0.0f; /// /// Wait for a key to be pressed. /// static void WaitForKey() { // Flush the input buffer while (Console.KeyAvailable) Console.ReadKey(true); Console.WriteLine(""); Console.WriteLine("Press any key to exit..."); Console.ReadKey(true); } /// /// Import languages from an existing rufus.loc /// /// The directy where the loc file is located. /// A list of Language elements. static List ParseLocFile(string path, string id = null) { var rufus_loc = path + @"\rufus.loc"; var rufus_pot = path + @"\rufus.pot"; var lines = File.ReadAllLines(rufus_loc); int line_nr = 0; string format = "D" + (int)(Math.Log10((double)lines.Count()) + 0.99999); string last_key = null; string section_name = null; string comment = null; List parts; List langs = new List(); Language lang = null; bool skip_line = false; sw.Start(); if (!File.Exists(rufus_loc)) { Console.Error.WriteLine($"Could not open {rufus_loc}"); return null; } Console.WriteLine($"Importing data from '{rufus_loc}':"); foreach (var line in lines) { if (cancel_requested) break; ++line_nr; Console.SetCursorPosition(0, Console.CursorTop); Console.Write($"[{line_nr.ToString(format)}/{lines.Count()}] "); var data = line.Trim(); int i = data.IndexOf("#"); if (i > 0) { comment = data.Substring(i + 1).Trim(); data = data.Substring(0, i).Trim(); } if (string.IsNullOrEmpty(data)) continue; if (skip_line && data[0] != 'l') continue; switch (data[0]) { case '#': comment += data.Substring(1).Trim() + "\n"; break; case 'l': comment = null; parts = Regex.Matches(data, @"[\""].+?[\""]|[^ ]+") .Cast() .Select(m => m.Value) .ToList(); if (parts.Count < 4) { Console.WriteLine("Error: Invalid 'l' command"); return null; } string lid = parts[1].Replace("\"", ""); if (id != null) { if ((!skip_line) && (id != lid) && (lid != "en-US")) skip_line = true; else if (skip_line && (id == lid)) skip_line = false; if (skip_line) break; } if (lang != null) langs.Add(lang); lang = new Language(); lang.id = parts[1].Replace("\"", ""); lang.name = parts[2].Replace("\"", ""); Console.WriteLine($"Found language {lang.id} '{lang.name}'"); lang.lcid = parts[3]; for (i = 4; i < parts.Count; i++) lang.lcid += " " + parts[i]; break; case 'a': // This attribue will be restored manually break; case 'g': comment = null; section_name = data.Substring(2).Trim(); lang.sections.Add(section_name, new List()); break; case 'v': lang.version = data.Substring(2).Trim(); break; case 't': if (data.StartsWith("t MSG") && section_name != "MSG") { section_name = "MSG"; lang.sections.Add(section_name, new List()); } if (data[1] != ' ') { Console.WriteLine("Error: Invalid 'l' command"); continue; } parts = Regex.Matches(data, @"(?() .Select(m => m.Value) .ToList(); if (parts.Count != 3) { Console.WriteLine("Error: Invalid 'l' command"); continue; } lang.sections[section_name].Add(new Message(parts[1], parts[2])); last_key = parts[1]; if (comment != null) { lang.comments[last_key] = comment.Trim(); comment = null; } break; case '"': if (String.IsNullOrEmpty(last_key)) { Console.WriteLine($"Error: No previous key for {data}"); continue; } lang.sections[section_name].Last().str += data; lang.sections[section_name].Last().str = lang.sections[section_name].Last().str.Replace("\"\"", ""); break; } } if (lang != null) langs.Add(lang); sw.Stop(); Console.WriteLine($"{(cancel_requested ? "CANCELLED after" : "DONE in")}" + $" {sw.ElapsedMilliseconds / 1000.0}s."); sw.Reset(); return langs; } /// /// Create .po/.pot files from a list of Language elements. /// /// The path where the .po/.pot files should be created. /// A lits of Languages elements /// true on success, false on error. static bool CreatePoFiles(string path, List langs, bool merge_pot = false) { if (langs == null) return false; var en_US = langs.Find(x => x.id == "en-US"); if (en_US == null) return false; var msg_to_ids = new Dictionary>(); // Build a dictionary of message string to List so that we can identify duplicates and remove them foreach (var section in en_US.sections) { foreach (var msg in section.Value) { if (msg_to_ids.ContainsKey(msg.str)) msg_to_ids[msg.str].Add(new Id(section.Key, msg.id)); else msg_to_ids.Add(msg.str, new List() { new Id(section.Key, msg.id) }); } } foreach (var lang in langs) { bool is_pot = (lang.id == "en-US"); var target = path + @"\" + (is_pot ? "rufus.pot" : lang.id + ".po"); Console.WriteLine($"Creating '{target}'"); using (var writer = new StreamWriter(target, false, encoding)) { writer.WriteLine(); writer.WriteLine("msgid \"\""); writer.WriteLine("msgstr \"\""); writer.WriteLine($"\"Project-Id-Version: {lang.version}\\n\""); writer.WriteLine($"\"Report-Msgid-Bugs-To: pete@akeo.ie\\n\""); writer.WriteLine($"\"POT-Creation-Date: {DateTime.Now.ToString("yyyy-MM-dd HH:mmzz00")}\\n\""); if (is_pot) writer.WriteLine($"\"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\\n\""); else writer.WriteLine($"\"PO-Revision-Date: {DateTime.Now.ToString("yyyy-MM-dd HH:mmzz00")}\\n\""); writer.WriteLine($"\"Last-Translator: FULL NAME \\n\""); writer.WriteLine($"\"Language-Team: LANGUAGE \\n\""); writer.WriteLine($"\"Language: {lang.id.Replace('-', '_')}\\n\""); writer.WriteLine($"\"MIME-Version: 1.0\\n\""); writer.WriteLine($"\"Content-Type: text/plain; charset=UTF-8\\n\""); writer.WriteLine($"\"Content-Transfer-Encoding: 8bit\\n\""); writer.WriteLine($"\"X-Poedit-SourceCharset: UTF-8\\n\""); writer.WriteLine($"\"X-Rufus-LanguageName: {lang.name}\\n\""); writer.WriteLine($"\"X-Rufus-LCID: {lang.lcid}\\n\""); var dupes = new List(); foreach (var section in lang.sections) { foreach (var msg in section.Value) { var en_str = en_US.sections[section.Key].Find(x => x.id == msg.id).str; // Handle duplicates if (dupes.Contains(en_str)) continue; writer.WriteLine(); foreach (var id in msg_to_ids[en_str]) { if (id.group == "MSG") writer.WriteLine($"#. • {id.id}"); else writer.WriteLine($"#. • {id.group} → {id.id}"); } if (msg_to_ids[en_str].Count > 1) dupes.Add(en_str); if (lang.comments.ContainsKey(msg.id)) { if (is_pot) writer.WriteLine("#."); foreach (var comment in lang.comments[msg.id].Split('\n')) if (comment.Trim() != "") writer.WriteLine((is_pot ? "#. " : "# ") + comment); } if (is_pot) { writer.WriteLine($"msgid {msg.str}"); writer.WriteLine("msgstr \"\""); } else { writer.WriteLine($"msgid {en_str}"); writer.WriteLine($"msgstr {msg.str}"); } } } } } Console.WriteLine("DONE."); return true; } /// /// Create a Language entry from a .po or .pot file. /// /// The name of the .po/.pot file. /// A Language element or null on error. static Language ParsePoFile(string file) { if (!File.Exists(file)) { Console.Error.WriteLine($"Could not open {file}"); return null; } Console.WriteLine($"Importing data from '{file}':"); bool is_pot = file.EndsWith(".pot"); var lines = File.ReadAllLines(file); string format = "D" + (int)(Math.Log10((double)lines.Count()) + 0.99999); int line_nr = 0; // msg_data[0] -> msgid, msg_data[1] -> msgstr string[] msg_data = new string[2] { null, null }; Language lang = new Language(); List ids = new List(); List comments = new List(); List codes = new List(); int msg_type = 0; sw.Start(); foreach (var line in lines) { if (cancel_requested) break; ++line_nr; Console.SetCursorPosition(0, Console.CursorTop); Console.Write($"[{line_nr.ToString(format)}/{lines.Count()}] "); var data = line.Trim(); if (!data.StartsWith("\"")) { var options = new Dictionary(); if ((msg_type == 1) && (string.IsNullOrEmpty(msg_data[0])) && (!string.IsNullOrEmpty(msg_data[1]))) { // Process the header string[] header = msg_data[1].Split(new string[] { "\\n" }, StringSplitOptions.None); foreach (string header_line in header) { if (string.IsNullOrEmpty(header_line)) continue; string[] opt = header_line.Split(new string[] { ": " }, StringSplitOptions.None); if (opt.Length != 2) { Console.WriteLine($"ERROR: Invalid header line '{header_line}'"); continue; } options.Add(opt[0], opt[1]); } lang.id = options[LANG_ID].Replace('_', '-'); lang.name = options[LANG_NAME]; lang.version = options[LANG_VERSION]; lang.lcid = options[LANG_LCID]; } } if (data.StartsWith("\"")) { if (data[data.Length - 1] != '"') { Console.WriteLine("ERROR: Unexpected quoted data"); continue; } msg_data[msg_type] += data.Substring(1, data.Length - 2); } else if (data.StartsWith("msgid ")) { if (data[6] != '"') { Console.WriteLine("ERROR: Unexpected data after 'msgid'"); continue; } msg_type = 0; msg_data[msg_type] = data.Substring(7, data.Length - 8); } else if (data.StartsWith("msgstr ")) { if (data[7] != '"') { Console.WriteLine("ERROR: Unexpected data after 'msgstr'"); continue; } msg_type = 1; msg_data[msg_type] = data.Substring(8, data.Length - 9); } else if (data.StartsWith("#. •")) { if (data.StartsWith("#. • MSG")) { ids.Add(new Id("MSG", data.Substring(5).Trim())); } else { string[] str = data.Substring(5).Split(new string[] { " → " }, StringSplitOptions.None); if (str.Length != 2) Console.WriteLine($"ERROR: Invalid ID {data}"); else ids.Add(new Id(str[0].Trim(), str[1].Trim())); } } else if (data.StartsWith("#. ")) { if (comments == null) comments = new List(); comments.Add(data.Substring(2).Trim()); } // Break or EOF => Process the previous section if (string.IsNullOrEmpty(data) || (line_nr == lines.Count())) { if ((!string.IsNullOrEmpty(msg_data[0])) && (ids.Count() != 0)) { foreach (var id in ids) { if (comments != null) { lang.comments.Add(id.id, ""); foreach (var comment in comments) lang.comments[id.id] += comment + "\n"; } // Ignore messages that have the same translation as en-US if (msg_data[0] == msg_data[1]) continue; if (!lang.sections.ContainsKey(id.group)) lang.sections.Add(id.group, new List()); lang.sections[id.group].Add(new Message(id.id, msg_data[is_pot ? 0 : 1])); } } ids = new List(); comments = null; } } // Sort the MSG section alphabetically lang.sections["MSG"] = lang.sections["MSG"].OrderBy(x => x.id).ToList(); sw.Stop(); Console.WriteLine($"{(cancel_requested ? "CANCELLED after" : "DONE in")}" + $" {sw.ElapsedMilliseconds / 1000.0}s."); sw.Reset(); return lang; } /// /// Write a loc language section. /// /// A streamwriter to the file to write to. /// The Language to write. static void WriteLoc(StreamWriter writer, Language lang) { bool is_pot = (lang.id == "en-US"); bool is_rtl = rtl_languages.Contains(lang.id); writer.WriteLine($"l \"{lang.id}\" \"{lang.name}\" {lang.lcid}"); writer.WriteLine($"v {lang.version}"); if (!is_pot) writer.WriteLine("b \"en-US\""); if (is_rtl) writer.WriteLine("a \"r\""); var sections = lang.sections.Keys.ToList(); foreach (var section in sections) { writer.WriteLine(); if (section != "MSG") writer.WriteLine($"g {section}"); foreach (var msg in lang.sections[section]) { if (lang.comments.ContainsKey(msg.id)) { foreach (var l in lang.comments[msg.id].Split('\n')) if (l.Trim() != "") writer.WriteLine($"# {l}"); } writer.WriteLine($"t {msg.id} \"{msg.str}\""); } } } /// /// Create a new rufus.loc from a list of Language elements. /// /// The path where the new 'rufus.loc' should be created. /// The list of Language elements. /// true on success, false on error. static bool UpdateLocFile(string path, Language lang) { if (lang == null) return false; var target = path + @"\rufus.loc"; var lines = File.ReadAllLines(target); using (var writer = new StreamWriter(target, false, encoding)) { bool skip = false; foreach (var line in lines) { if (line.StartsWith($"l \"{lang.id}\"")) { skip = true; WriteLoc(writer, lang); writer.WriteLine(); } else if (line.StartsWith("######")) { skip = false; } if (!skip) writer.WriteLine(line); } } return true; } /// /// Create a new rufus.loc from a list of Language elements. /// /// The path where the new 'rufus.loc' should be created. /// The list of Language elements. /// true on success, false on error. static bool SaveLocFile(string path, List list) { if ((list == null) || (list.Count == 0)) return false; var target = path + @"\rufus.loc"; sw.Start(); Console.WriteLine($"Creating '{target}':"); using (var writer = new StreamWriter(target, false, encoding)) { var notice = $"### Autogenerated by {app_name} {app_version} for use with Rufus - DO NOT EDIT!!! ###"; var sep = new String('#', notice.Length); writer.WriteLine(sep); writer.WriteLine(notice); writer.WriteLine(sep); writer.WriteLine(); writer.WriteLine("# List of all languages included in this file (with version)"); foreach (var lang in list) { writer.WriteLine($"# • v{lang.version} \"{lang.id}\" \"{lang.name}\""); } foreach (var lang in list) { if (cancel_requested) break; Console.WriteLine($"Adding {lang.id}"); writer.WriteLine(); writer.WriteLine(sep); WriteLoc(writer, lang); } } sw.Stop(); Console.WriteLine($"{(cancel_requested ? "CANCELLED after" : "DONE in")}" + $" {sw.ElapsedMilliseconds / 1000.0}s."); sw.Reset(); return true; } static bool DownloadFile(string url, string dest) { download_status = 0; in_progress = false; using (wc) { wc.DownloadFileCompleted += new AsyncCompletedEventHandler(DownloadCompleted); wc.DownloadProgressChanged += new DownloadProgressChangedEventHandler(DownloadProgress); Console.WriteLine($"Downloading {url}:"); sw.Start(); try { wc.DownloadFileAsync(new Uri(url), dest); } catch (Exception e) { Console.WriteLine("ERROR: " + e.Message); return false; } } while (download_status == 0) Thread.Sleep(100); Console.WriteLine(); if (download_status == 1) { Console.WriteLine("Download complete"); return true; } Console.WriteLine("Download has been canceled."); return false; } // The event that will fire whenever the progress of the WebClient is changed static void DownloadProgress(object sender, DownloadProgressChangedEventArgs e) { if (cancel_requested) { wc.CancelAsync(); return; } if (in_progress) return; // Prevent this call from being re-entrant in_progress = true; speed = (e.BytesReceived / 1024d / sw.Elapsed.TotalSeconds); Console.SetCursorPosition(0, Console.CursorTop); Console.Write($" {e.ProgressPercentage.ToString("000.0")} % ({speed.ToString("0.00")} KB/s)"); in_progress = false; } // The event that will trigger when the WebClient is completed static void DownloadCompleted(object sender, AsyncCompletedEventArgs e) { if (!e.Cancelled) { Console.SetCursorPosition(0, Console.CursorTop); Console.Write($" 100.0 % ({speed.ToString("0.00")} KB/s)"); } sw.Reset(); download_status = (e.Cancelled) ? 2 : 1; } static void Main(string[] args) { Console.OutputEncoding = System.Text.Encoding.UTF8; Console.CancelKeyPress += delegate (object sender, ConsoleCancelEventArgs e) { e.Cancel = true; cancel_requested = true; }; Console.WriteLine($"{app_name} {app_version} - Poedit to rufus.loc conversion utility"); var path = @"C:\pollock"; var loc = path + @"\download.loc"; // Download the loc file //var url = "https://github.com/pbatard/rufus/raw/master/res/localization/rufus.loc"; //if (!DownloadFile(url, loc)) // goto Exit; // Convert to CRLF and get all the language ids var lines = File.ReadAllLines(loc); string id = "", name = ""; var list = new List(); using (var writer = new StreamWriter(loc, false, encoding)) { foreach (var line in lines) { if (line.StartsWith("l ")) { var el = line.Split('\"'); id = el[1]; name = el[3].Split('(')[0].Trim(); } else if (line.StartsWith("v ")) { if (id != "en-US") list.Add(new string[] { name, id, line.Substring(2) }); } writer.WriteLine(line); } } Menu: Console.WriteLine(); Console.WriteLine("Please enter the number of the language you want to edit or 'q' to quit:"); Console.WriteLine(); int split = (list.Count + 1) / 2; for (int i = 0; i < split; i++) { name = $"{list[i][0]} ({list[i][1]})"; Console.Write($"[{(i+1).ToString("00")}] {name,-29} (v{list[i][2]})"); name = $"{list[i + split][0]} ({list[i + split][1]})"; Console.WriteLine($" | [{(i + 1 + split).ToString("00")}] {name,-29} (v{list[i + split][2]})"); } Console.WriteLine(); Retry: string input = Console.ReadLine(); if (input.StartsWith("q")) goto Exit; if (!Int32.TryParse(input, out int number) || (number <= 0) || (number > list.Count)) { if (input.StartsWith("m")) goto Menu; Console.WriteLine("Invalid selection (Type 'm' to display the menu again)"); goto Retry; } number--; Console.WriteLine($"{list[number][0]} was selected"); CreatePoFiles(path, ParseLocFile(path, list[number][1])); // NB: Can find PoEdit from Computer\HKEY_CURRENT_USER\Software\Classes\Local Settings\Software\Microsoft\Windows\Shell\MuiCache //CreatePoFiles(path, ParseLocFile(@"C:\rufus\res\localization")); //var en_US = ParsePoFile(path + @"\rufus.pot"); //var fr_FR = ParsePoFile(path + @"\fr-FR.po"); //var ar_SA = ParsePoFile(path + @"\ar-SA.po"); //var vi_VN = ParsePoFile(path + @"\vi-VN.po"); //List list = new List(); //list.Add(en_US); //list.Add(ar_SA); //list.Add(fr_FR); //list.Add(vi_VN); //SaveLocFile(path, list); // UpdateLocFile(path + @"\test", fr_FR); Exit: WaitForKey(); } } }