自动翻译.po文件?
曾经有一些服务使用谷歌翻译API V1自动翻译.po
文件。自动翻译.po文件?
谷歌已经停止了他们的V1 API,而用V2他们收取了20美元的1000,000字。
我已经搜索,但无法找到任何提供翻译V2版本的工具。你会期望有人更新他们的工具,并收取20,000美元的2美元的话费,并获得很好的利润。
是否有任何付费或免费的工具可以自动翻译.po
文件?
见我的免费PHP工具Potrans,其中使用谷歌翻译API从这里回购
下载翻译PO文件:https://github.com/OzzyCzech/potrans
太棒了!有%s和%d个变量存在一些缺陷,但总而言之,这很好。谢谢 – JohnWolf 2015-02-14 09:48:39
http://sourceforge.net/projects/po-auto-tran/?source=dlp在这里你走我只是下载和使用它,它的伟大工程!我不知道这是否回答你的问题,但它对我的工作不起作用
以上的重复 – Sam 2014-08-18 05:55:02
随意编辑这篇文章进行更正或更好的错误处理。我不是.po
文件格式的专家,但我认为这将适用于我的angular-gettext需求。我认为我使用的唯一外部库是Newtonsoft的Json.NET。我不属于Frengly。它出现在谷歌上。如果您需要每3秒钟翻译一次以上,我建议您使用其他翻译API。
输入文件:template.pot
msgid ""
msgstr ""
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
#: Comment
msgid "You do not have permission to view this application"
msgstr ""
msgid "You have permission to view this application"
msgstr ""
输入命令:> program.exe template.pot en es_VE fr_FR
输出文件1:en.cache.json 这是创建所以,无论翻译uti你使用的礼物不必一遍又一遍地被击中。
{
"es_VE": {
"You do not have permission to view this application": "Tu no la habana permiso que vista este aplicación",
"You have permission to view this application": "Tu tienes permiso que vista este aplicación"
},
"fr_FR": {
"You do not have permission to view this application": "Vous le faites pas as autorisation a vue cette une demande",
"You have permission to view this application": "Tuas autorisation a vue cette une demande"
}
}
输出文件2:es_VE.po
msgid ""
msgstr ""
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
#: Comment
msgid "You do not have permission to view this application"
msgstr "Tu no la habana permiso que vista este aplicación"
msgid "You have permission to view this application"
msgstr "Tu tienes permiso que vista este aplicación"
输出文件3:fr_FR目录。PO
msgid ""
msgstr ""
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
#: Comment
msgid "You do not have permission to view this application"
msgstr "Vous le faites pas as autorisation a vue cette une demande"
msgid "You have permission to view this application"
msgstr "Tuas autorisation a vue cette une demande"
来源
public interface ICache
{
void Add(string language, IEntry entry);
IEntry Get(string language, string id);
string GetSerialized();
}
public class JsonCache : ICache
{
private Dictionary<string, Dictionary<string, string>> _cache;
public JsonCache(string json)
{
this._cache =
json == null ?
new Dictionary<string, Dictionary<string, string>>() :
JsonConvert.DeserializeObject<Dictionary<string, Dictionary<string, string>>>(json);
}
public void Add(string language, IEntry entry)
{
if (!this._cache.ContainsKey(language))
{
this._cache.Add(language, new Dictionary<string, string>());
}
var languageCache = this._cache[language];
languageCache.Add(entry.Id, entry.Value);
}
public IEntry Get(string language, string id)
{
if (!this._cache.ContainsKey(language))
{
return null;
}
var languageCache = this._cache[language];
Entry result = null;
if (languageCache.ContainsKey(id))
{
result = new Entry();
result.Id = id;
result.Value = languageCache[id];
}
return result;
}
public string GetSerialized()
{
return JsonConvert.SerializeObject(this._cache, Formatting.Indented);
}
}
public interface IReader : IDisposable
{
IEntry Read();
}
public class PoReader : IReader
{
private StreamReader _reader;
public PoReader(string fileName)
{
this._reader = new StreamReader(fileName);
}
public void Dispose()
{
if (this._reader != null)
{
this._reader.Dispose();
}
}
public IEntry Read()
{
var entry = new Entry();
while (entry.Id == null || entry.Value == null)
{
var line = this._reader.ReadLine();
if (line == null)
{
return null;
}
if (line.StartsWith(Constants.StartComment))
{
entry.Comment = line.Substring(Constants.StartComment.Length);
}
else if (line.StartsWith(Constants.StartId))
{
entry.Id = line.Substring(Constants.StartId.Length);
// Remove the double quotes.
entry.Id = entry.Id.Substring(1, entry.Id.Length - 2);
}
else if (line.StartsWith(Constants.StartValue))
{
entry.Value = line.Substring(Constants.StartValue.Length);
// Remove the double quotes.
entry.Value = entry.Value.Substring(1, entry.Value.Length - 2);
}
}
// Skip the first entry
if (entry.Id.Length == 0)
{
return this.Read();
}
return entry;
}
}
public class CachedTranslator : ITranslator
{
private ITranslator _translator;
private ICache _cache;
public CachedTranslator(ICache cache, ITranslator translator)
{
this._translator = translator;
this._cache = cache;
}
public IEntry Translate(string language, IEntry entry)
{
var result = this._cache.Get(language, entry.Id);
if (result == null)
{
result = this._translator.Translate(language, entry);
this._cache.Add(language, result);
}
else
{
// We don't want to use the cached comment.
var clone = new Entry();
clone.Comment = entry.Comment;
clone.Value = result.Value;
clone.Id = result.Id;
result = clone;
}
return result;
}
}
public class FrenglyTranslator : ITranslator
{
private string _password;
private string _email;
private string _inLanguage;
public FrenglyTranslator(string email, string password, string inLanguage)
{
this._email = email;
this._password = password;
this._inLanguage = inLanguage;
}
public IEntry Translate(string language, IEntry entry)
{
var url = string.Format("http://syslang.com?src={4}&dest={0}&text={1}&email={2}&password={3}&outformat=json",
language.Substring(0, 2),
entry.Id,
this._email,
this._password,
this._inLanguage);
var result = new Entry();
result.Id = entry.Id;
result.Comment = entry.Comment;
using (var client = new HttpClient())
{
var clientResult = client.GetStringAsync(url).Result;
var jo = (JObject)JsonConvert.DeserializeObject(clientResult);
result.Value = jo.Property("translation").Value.Value<string>();
}
// Must wait 3 seconds between calls.
Thread.Sleep(3001);
return result;
}
}
public interface ITranslator
{
IEntry Translate(string language, IEntry entry);
}
public interface IWriter : IDisposable
{
void Write(IEntry entry);
}
public class PoWriter : IWriter
{
private StreamWriter _writer;
public PoWriter(string fileName)
{
this._writer = new StreamWriter(fileName);
var header = @"msgid """"
msgstr """"
""Content-Type: text/plain; charset=UTF-8\n""
""Content-Transfer-Encoding: 8bit\n""";
this._writer.WriteLine(header);
}
public void Write(IEntry entry)
{
this._writer.WriteLine();
if (entry.Comment != null && entry.Comment.Length > 0)
{
this._writer.WriteLine(Constants.StartComment + entry.Comment);
}
this._writer.WriteLine(string.Format("{0}\"{1}\"", Constants.StartId, entry.Id));
this._writer.WriteLine(string.Format("{0}\"{1}\"", Constants.StartValue, entry.Value));
}
public void Dispose()
{
if (this._writer != null)
{
this._writer.Dispose();
}
}
}
public static class Constants
{
public const string StartComment = "#: ";
public const string StartId = "msgid ";
public const string StartValue = "msgstr ";
}
public class Entry : IEntry
{
public string Comment { get; set; }
public string Value { get; set; }
public string Id { get; set; }
}
public interface IEntry
{
string Comment { get; }
string Value { get; }
string Id { get; }
}
class Program
{
private const string cacheFileNameSuffix = ".cache.json";
private const string frenglyEmail = "[email protected]";
private const string frenglyPassword = "YourPassword";
static void Main(string[] args)
{
//
// INITIALIZE
//
var inFileName = args[0];
var inLanguage = args[1];
var outLanguages = args.Skip(2);
// ICache
var cacheFileName = inLanguage + cacheFileNameSuffix;
var json = File.Exists(cacheFileName) ? File.ReadAllText(cacheFileName) : null;
ICache cache = new JsonCache(json);
// ITranslator
ITranslator translator = new FrenglyTranslator(frenglyEmail, frenglyPassword, inLanguage);
ITranslator cachedTranslator = new CachedTranslator(cache, translator);
// IWriters
var writers = new Dictionary<string, IWriter>();
foreach (var language in outLanguages)
{
writers.Add(language, new PoWriter(language + ".po"));
}
try
{
using (IReader reader = new PoReader(inFileName))
{
//
// RUN
//
IEntry entry = null;
while (true)
{
entry = reader.Read();
if (entry == null)
{
break;
}
foreach (var kv in writers)
{
var translated = cachedTranslator.Translate(kv.Key, entry);
kv.Value.Write(translated);
}
}
}
}
finally
{
// Store the cache.
File.WriteAllText(cacheFileName, cache.GetSerialized());
//
// CLEANUP
//
// Dispose of the writers.
foreach (var writer in writers.Values)
{
if (writer != null)
{
writer.Dispose();
}
}
}
}
}
谷歌不允许你重新出售其支付的翻译,根据其服务条款,所以没有人能够做什么你建议... – daveagp 2012-08-31 00:21:21