fill in missing translations with Google Translate
This commit is contained in:
parent
84a0e13409
commit
186d8743b4
53
i18n.py
53
i18n.py
@ -13,19 +13,22 @@
|
|||||||
from __future__ import print_function
|
from __future__ import print_function
|
||||||
import os, fnmatch, re, shutil, errno
|
import os, fnmatch, re, shutil, errno
|
||||||
from sys import argv as _argv
|
from sys import argv as _argv
|
||||||
|
from sys import stderr as _stderr
|
||||||
|
|
||||||
# Running params
|
# Running params
|
||||||
params = {"recursive": False,
|
params = {"recursive": False,
|
||||||
"help": False,
|
"help": False,
|
||||||
"mods": False,
|
"mods": False,
|
||||||
"verbose": False,
|
"verbose": False,
|
||||||
"folders": []
|
"folders": [],
|
||||||
|
"no-old-file": False
|
||||||
}
|
}
|
||||||
# Available CLI options
|
# Available CLI options
|
||||||
options = {"recursive": ['--recursive', '-r'],
|
options = {"recursive": ['--recursive', '-r'],
|
||||||
"help": ['--help', '-h'],
|
"help": ['--help', '-h'],
|
||||||
"mods": ['--installed-mods'],
|
"mods": ['--installed-mods'],
|
||||||
"verbose": ['--verbose', '-v']
|
"verbose": ['--verbose', '-v'],
|
||||||
|
"no-old-file": ['--no-old-file']
|
||||||
}
|
}
|
||||||
|
|
||||||
# Strings longer than this will have extra space added between
|
# Strings longer than this will have extra space added between
|
||||||
@ -64,6 +67,8 @@ DESCRIPTION
|
|||||||
run on all subfolders of paths given
|
run on all subfolders of paths given
|
||||||
{', '.join(options["mods"])}
|
{', '.join(options["mods"])}
|
||||||
run on locally installed modules
|
run on locally installed modules
|
||||||
|
{', '.join(options["no-old-file"])}
|
||||||
|
do not create *.old files
|
||||||
{', '.join(options["verbose"])}
|
{', '.join(options["verbose"])}
|
||||||
add output information
|
add output information
|
||||||
''')
|
''')
|
||||||
@ -108,13 +113,15 @@ def main():
|
|||||||
|
|
||||||
#group 2 will be the string, groups 1 and 3 will be the delimiters (" or ')
|
#group 2 will be the string, groups 1 and 3 will be the delimiters (" or ')
|
||||||
#See https://stackoverflow.com/questions/46967465/regex-match-text-in-either-single-or-double-quote
|
#See https://stackoverflow.com/questions/46967465/regex-match-text-in-either-single-or-double-quote
|
||||||
pattern_lua = re.compile(r'[\.=^\t,{\(\s]N?S\(\s*(["\'])((?:\\\1|(?:(?!\1)).)*)(\1)[\s,\)]', re.DOTALL)
|
pattern_lua_s = re.compile(r'[\.=^\t,{\(\s]N?S\(\s*(["\'])((?:\\\1|(?:(?!\1)).)*)(\1)[\s,\)]', re.DOTALL)
|
||||||
pattern_lua_bracketed = re.compile(r'[\.=^\t,{\(\s]N?S\(\s*\[\[(.*?)\]\][\s,\)]', re.DOTALL)
|
pattern_lua_fs = re.compile(r'[\.=^\t,{\(\s]N?FS\(\s*(["\'])((?:\\\1|(?:(?!\1)).)*)(\1)[\s,\)]', re.DOTALL)
|
||||||
|
pattern_lua_bracketed_s = re.compile(r'[\.=^\t,{\(\s]N?S\(\s*\[\[(.*?)\]\][\s,\)]', re.DOTALL)
|
||||||
|
pattern_lua_bracketed_fs = re.compile(r'[\.=^\t,{\(\s]N?FS\(\s*\[\[(.*?)\]\][\s,\)]', re.DOTALL)
|
||||||
|
|
||||||
# Handles "concatenation" .. " of strings"
|
# Handles "concatenation" .. " of strings"
|
||||||
pattern_concat = re.compile(r'["\'][\s]*\.\.[\s]*["\']', re.DOTALL)
|
pattern_concat = re.compile(r'["\'][\s]*\.\.[\s]*["\']', re.DOTALL)
|
||||||
|
|
||||||
pattern_tr = re.compile(r'(.+?[^@])=(.*)')
|
pattern_tr = re.compile(r'(.*?[^@])=(.*)')
|
||||||
pattern_name = re.compile(r'^name[ ]*=[ ]*([^ \n]*)')
|
pattern_name = re.compile(r'^name[ ]*=[ ]*([^ \n]*)')
|
||||||
pattern_tr_filename = re.compile(r'\.tr$')
|
pattern_tr_filename = re.compile(r'\.tr$')
|
||||||
pattern_po_language_code = re.compile(r'(.*)\.po$')
|
pattern_po_language_code = re.compile(r'(.*)\.po$')
|
||||||
@ -205,8 +212,10 @@ def mkdir_p(path):
|
|||||||
# dKeyStrings is a dictionary of localized string to source file sets
|
# dKeyStrings is a dictionary of localized string to source file sets
|
||||||
# dOld is a dictionary of existing translations and comments from
|
# dOld is a dictionary of existing translations and comments from
|
||||||
# the previous version of this text
|
# the previous version of this text
|
||||||
def strings_to_text(dkeyStrings, dOld, mod_name):
|
def strings_to_text(dkeyStrings, dOld, mod_name, header_comments):
|
||||||
lOut = [f"# textdomain: {mod_name}\n"]
|
lOut = [f"# textdomain: {mod_name}\n"]
|
||||||
|
if header_comments is not None:
|
||||||
|
lOut.append(header_comments)
|
||||||
|
|
||||||
dGroupedBySource = {}
|
dGroupedBySource = {}
|
||||||
|
|
||||||
@ -266,7 +275,7 @@ def write_template(templ_file, dkeyStrings, mod_name):
|
|||||||
# read existing template file to preserve comments
|
# read existing template file to preserve comments
|
||||||
existing_template = import_tr_file(templ_file)
|
existing_template = import_tr_file(templ_file)
|
||||||
|
|
||||||
text = strings_to_text(dkeyStrings, existing_template[0], mod_name)
|
text = strings_to_text(dkeyStrings, existing_template[0], mod_name, existing_template[2])
|
||||||
mkdir_p(os.path.dirname(templ_file))
|
mkdir_p(os.path.dirname(templ_file))
|
||||||
with open(templ_file, "wt", encoding='utf-8') as template_file:
|
with open(templ_file, "wt", encoding='utf-8') as template_file:
|
||||||
template_file.write(text)
|
template_file.write(text)
|
||||||
@ -282,9 +291,13 @@ def read_lua_file_strings(lua_file):
|
|||||||
text = re.sub(pattern_concat, "", text)
|
text = re.sub(pattern_concat, "", text)
|
||||||
|
|
||||||
strings = []
|
strings = []
|
||||||
for s in pattern_lua.findall(text):
|
for s in pattern_lua_s.findall(text):
|
||||||
strings.append(s[1])
|
strings.append(s[1])
|
||||||
for s in pattern_lua_bracketed.findall(text):
|
for s in pattern_lua_bracketed_s.findall(text):
|
||||||
|
strings.append(s)
|
||||||
|
for s in pattern_lua_fs.findall(text):
|
||||||
|
strings.append(s[1])
|
||||||
|
for s in pattern_lua_bracketed_fs.findall(text):
|
||||||
strings.append(s)
|
strings.append(s)
|
||||||
|
|
||||||
for s in strings:
|
for s in strings:
|
||||||
@ -302,9 +315,11 @@ def read_lua_file_strings(lua_file):
|
|||||||
# returns both a dictionary of translations
|
# returns both a dictionary of translations
|
||||||
# and the full original source text so that the new text
|
# and the full original source text so that the new text
|
||||||
# can be compared to it for changes.
|
# can be compared to it for changes.
|
||||||
|
# Returns also header comments in the third return value.
|
||||||
def import_tr_file(tr_file):
|
def import_tr_file(tr_file):
|
||||||
dOut = {}
|
dOut = {}
|
||||||
text = None
|
text = None
|
||||||
|
header_comment = None
|
||||||
if os.path.exists(tr_file):
|
if os.path.exists(tr_file):
|
||||||
with open(tr_file, "r", encoding='utf-8') as existing_file :
|
with open(tr_file, "r", encoding='utf-8') as existing_file :
|
||||||
# save the full text to allow for comparison
|
# save the full text to allow for comparison
|
||||||
@ -318,6 +333,16 @@ def import_tr_file(tr_file):
|
|||||||
for line in existing_file.readlines():
|
for line in existing_file.readlines():
|
||||||
line = line.rstrip('\n')
|
line = line.rstrip('\n')
|
||||||
if line[:3] == "###":
|
if line[:3] == "###":
|
||||||
|
if header_comment is None:
|
||||||
|
# Save header comments
|
||||||
|
header_comment = latest_comment_block
|
||||||
|
# Stip textdomain line
|
||||||
|
tmp_h_c = ""
|
||||||
|
for l in header_comment.split('\n'):
|
||||||
|
if not l.startswith("# textdomain:"):
|
||||||
|
tmp_h_c += l + '\n'
|
||||||
|
header_comment = tmp_h_c
|
||||||
|
|
||||||
# Reset comment block if we hit a header
|
# Reset comment block if we hit a header
|
||||||
latest_comment_block = None
|
latest_comment_block = None
|
||||||
continue
|
continue
|
||||||
@ -338,7 +363,7 @@ def import_tr_file(tr_file):
|
|||||||
outval["comment"] = latest_comment_block
|
outval["comment"] = latest_comment_block
|
||||||
latest_comment_block = None
|
latest_comment_block = None
|
||||||
dOut[match.group(1)] = outval
|
dOut[match.group(1)] = outval
|
||||||
return (dOut, text)
|
return (dOut, text, header_comment)
|
||||||
|
|
||||||
# Walks all lua files in the mod folder, collects translatable strings,
|
# Walks all lua files in the mod folder, collects translatable strings,
|
||||||
# and writes it to a template.txt file
|
# and writes it to a template.txt file
|
||||||
@ -377,11 +402,12 @@ def update_tr_file(dNew, mod_name, tr_file):
|
|||||||
dOld = tr_import[0]
|
dOld = tr_import[0]
|
||||||
textOld = tr_import[1]
|
textOld = tr_import[1]
|
||||||
|
|
||||||
textNew = strings_to_text(dNew, dOld, mod_name)
|
textNew = strings_to_text(dNew, dOld, mod_name, tr_import[2])
|
||||||
|
|
||||||
if textOld and textOld != textNew:
|
if textOld and textOld != textNew:
|
||||||
print(f"{tr_file} has changed.")
|
print(f"{tr_file} has changed.")
|
||||||
shutil.copyfile(tr_file, f"{tr_file}.old")
|
if not params["no-old-file"]:
|
||||||
|
shutil.copyfile(tr_file, f"{tr_file}.old")
|
||||||
|
|
||||||
with open(tr_file, "w", encoding='utf-8') as new_tr_file:
|
with open(tr_file, "w", encoding='utf-8') as new_tr_file:
|
||||||
new_tr_file.write(textNew)
|
new_tr_file.write(textNew)
|
||||||
@ -399,7 +425,8 @@ def update_mod(folder):
|
|||||||
for tr_file in get_existing_tr_files(folder):
|
for tr_file in get_existing_tr_files(folder):
|
||||||
update_tr_file(data, modname, os.path.join(folder, "locale/", tr_file))
|
update_tr_file(data, modname, os.path.join(folder, "locale/", tr_file))
|
||||||
else:
|
else:
|
||||||
print("Unable to find modname in folder " + folder)
|
print(f"\033[31mUnable to find modname in folder {folder}.\033[0m", file=_stderr)
|
||||||
|
exit(1)
|
||||||
|
|
||||||
# Determines if the folder being pointed to is a mod or a mod pack
|
# Determines if the folder being pointed to is a mod or a mod pack
|
||||||
# and then runs update_mod accordingly
|
# and then runs update_mod accordingly
|
||||||
|
@ -1,12 +1,15 @@
|
|||||||
# textdomain: dynamic_liquid
|
# textdomain: dynamic_liquid
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### init.lua ###
|
### init.lua ###
|
||||||
|
|
||||||
A natural spring that generates an endless stream of water source blocks=
|
#WARNING: AUTOTRANSLATED BY GOOGLE TRANSLATE
|
||||||
|
A natural spring that generates an endless stream of water source blocks=Eine natürliche Quelle, die einen endlosen Strom von Wasserquellenblöcken erzeugt
|
||||||
|
|
||||||
Damp Clay=Feuchten Lehm
|
Damp Clay=Feuchten Lehm
|
||||||
|
|
||||||
Generates one source block of water directly on top of itself once per second, provided the space is clear. If this natural spring is dug out the flow stops and it is turned into ordinary cobble.=
|
#WARNING: AUTOTRANSLATED BY GOOGLE TRANSLATE
|
||||||
|
Generates one source block of water directly on top of itself once per second, provided the space is clear. If this natural spring is dug out the flow stops and it is turned into ordinary cobble.=Erzeugt einmal pro Sekunde einen Quellwasserblock direkt über sich, sofern der Raum frei ist. Wenn diese natürliche Quelle ausgegraben wird, stoppt der Fluss und es wird zu gewöhnlichem Kopfsteinpflaster.
|
||||||
|
|
||||||
Spring=Quelle
|
Spring=Quelle
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
# textdomain: dynamic_liquid
|
# textdomain: dynamic_liquid
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### init.lua ###
|
### init.lua ###
|
||||||
|
|
||||||
A natural spring that generates an endless stream of water source blocks=Un manantial natural que genera un flujo sin fín de bloques fuente de agua
|
A natural spring that generates an endless stream of water source blocks=Un manantial natural que genera un flujo sin fín de bloques fuente de agua
|
||||||
|
@ -1,12 +1,15 @@
|
|||||||
# textdomain: dynamic_liquid
|
# textdomain: dynamic_liquid
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### init.lua ###
|
### init.lua ###
|
||||||
|
|
||||||
A natural spring that generates an endless stream of water source blocks=
|
#WARNING: AUTOTRANSLATED BY GOOGLE TRANSLATE
|
||||||
|
A natural spring that generates an endless stream of water source blocks=Une source naturelle qui génère un flux sans fin de blocs de source d'eau
|
||||||
|
|
||||||
Damp Clay=Argile humide
|
Damp Clay=Argile humide
|
||||||
|
|
||||||
Generates one source block of water directly on top of itself once per second, provided the space is clear. If this natural spring is dug out the flow stops and it is turned into ordinary cobble.=
|
#WARNING: AUTOTRANSLATED BY GOOGLE TRANSLATE
|
||||||
|
Generates one source block of water directly on top of itself once per second, provided the space is clear. If this natural spring is dug out the flow stops and it is turned into ordinary cobble.=Génère un bloc source d'eau directement sur lui-même une fois par seconde, à condition que l'espace soit dégagé. Si cette source naturelle est creusée, le débit s'arrête et il se transforme en galets ordinaires.
|
||||||
|
|
||||||
Spring=Source
|
Spring=Source
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
# textdomain: dynamic_liquid
|
# textdomain: dynamic_liquid
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### init.lua ###
|
### init.lua ###
|
||||||
|
|
||||||
A natural spring that generates an endless stream of water source blocks=
|
A natural spring that generates an endless stream of water source blocks=
|
||||||
|
Loading…
x
Reference in New Issue
Block a user