hgt.py 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. import os
  2. import sys
  3. import re
  4. import warnings
  5. from typing import List, Tuple, Dict
  6. FileList = List[str]
  7. FlatList = Dict[str, Tuple[str]]
  8. class RunError(Exception):
  9. def __init__(self, message="RunError: run as module is not allowed."):
  10. self.message = message
  11. class ArgError(Exception):
  12. def __init__(self, message="ArgError: just 6 arg is allowed."):
  13. self.message = message
  14. if __name__ != '__main__':
  15. raise RunError()
  16. argv = sys.argv
  17. if len(argv) < 7:
  18. warnings.warn(f"Too few argument [{len(argv)}]")
  19. raise ArgError()
  20. output_dir = argv[1]
  21. export_h = argv[2]
  22. export = argv[3]
  23. translation = argv[4]
  24. base_name = argv[5]
  25. input_dir = argv[6:]
  26. translation_output = os.path.join(output_dir, "tr")
  27. base_tr_file = os.path.join(translation, "base.py")
  28. try:
  29. os.makedirs(output_dir) # 生成输出目录
  30. except OSError:
  31. ...
  32. try:
  33. os.makedirs(translation_output) # 生成tr目录
  34. except OSError:
  35. ...
  36. # 执行base.py代码, 并保存结果
  37. base_tr = {}
  38. if os.path.exists(base_tr_file):
  39. with open(base_tr_file, "r", encoding="utf-8") as f:
  40. code = f.read()
  41. exec(code, base_tr, base_tr)
  42. def getFileFromPath(paths: List[str], file_type_: str) -> FileList:
  43. """
  44. 函数名: 获取目录列表中所有指定后缀的文件
  45. :param paths: 目录列表
  46. :param file_type_: 指定后在
  47. :return: 文件列表
  48. """
  49. tmp: FileList = []
  50. for path in paths:
  51. for file_path, dir_names, file_names in os.walk(path):
  52. for names in file_names:
  53. file_type = os.path.splitext(names)[-1]
  54. if file_type == file_type_:
  55. tmp.append(os.path.join(file_path, names))
  56. return tmp
  57. # 获取宏 HT_aFunGetText的列表并计算
  58. file_list: FileList = getFileFromPath(input_dir, '.c')
  59. pattern = re.compile(f'HT_{base_name}GetText' + r'\(([\S]+),[\s]*(\"[^\"]*\")\)') # 宏的定义
  60. flat_list: FlatList = {}
  61. for file in file_list:
  62. with open(file, "r", encoding="utf-8") as f:
  63. data = f.readline()
  64. while data:
  65. result: List[str] = pattern.findall(data)
  66. for i in result:
  67. tmp = i[1]
  68. if tmp == "\"<base-tr>\"":
  69. tmp = f"\"{base_tr.get(i[0], None)}\""
  70. default_ = tmp.replace('\n', '\\n').replace('\r', '')
  71. if i[0] in flat_list:
  72. if flat_list[i[0]][0] != default_ and default_ != '""' and flat_list[i[0]][0] != '""':
  73. # 若果是空字串则可以覆盖
  74. warnings.warn(f"Double define text: {i[0]}")
  75. continue
  76. elif default_ == '""':
  77. default_ = flat_list[i[0]][0]
  78. flat_list[i[0]] = (default_,) # 生成一个数组
  79. data = f.readline()
  80. # 生成对应文件
  81. with open(os.path.join(output_dir, f"{base_name}_ht.py"), "w", encoding="utf-8") as fpy:
  82. with open(os.path.join(output_dir, f"{base_name}_ht.c"), "w", encoding="utf-8") as fc:
  83. with open(os.path.join(output_dir, f"{base_name}_ht.h"), "w", encoding="utf-8") as fh:
  84. head = f'''/*
  85. * File: {base_name}_ht.c/{base_name}_ht.c
  86. * The file is automatically generated by Huan-GetText
  87. * Encoding: utf-8
  88. */'''
  89. fc.write(head + '\n\n')
  90. fc.write(f"#include \"{base_name}_ht.h\"\n")
  91. fc.write(f"#undef HT_{base_name}GetText\n")
  92. fh.write(head + '\n\n')
  93. fh.write("#ifndef HT_GETTEXT_H\n")
  94. fh.write("#define HT_GETTEXT_H\n")
  95. fh.write(f"#ifdef HT_{base_name}GetText\n")
  96. fh.write(f"#error \"Double define HT_{base_name}GetText\"\n")
  97. fh.write("#endif\n")
  98. fh.write(f"#include \"{export_h}.h\"\n")
  99. fh.write(f"#define HT_{base_name}GetText(name, ...) ((char *)(HT_TEXT_{base_name}_ ## name))\n")
  100. fh.write(f"{export} int HT_init{base_name}GetText(char *lang);\n")
  101. fpy.write("# Example for tr file" + '\n\n')
  102. for i in flat_list:
  103. fc.write(f"{export} const char *HT_TEXT_{base_name}_{i} = {flat_list[i][0]};\n")
  104. fh.write(f"{export} const char *HT_TEXT_{base_name}_{i};\n")
  105. fpy.write(f"# {i}: str = \"\" # {flat_list[i][0]}\n")
  106. fc.write(f'''\n
  107. /* define init{base_name}GetText */
  108. /* need dlfcn or dlfcn-win32 */
  109. #include "dlfcn.h"
  110. #include "stdlib.h"
  111. static void *handle = NULL;
  112. int HT_init{base_name}GetText(char *lang) {{
  113. if (lang == NULL || handle != NULL)
  114. return 2;
  115. handle = dlopen(lang, RTLD_NOW);
  116. if (handle == NULL)
  117. return 1;
  118. char **tmp;\n\n''')
  119. for i in flat_list:
  120. fc.write(f''' tmp = dlsym(handle, "HT__TEXT_{base_name}_{i}");\n''')
  121. fc.write(f''' if (tmp != NULL) HT_TEXT_{base_name}_{i} = *tmp;\n\n''')
  122. fc.write(' return 0;\n}\n')
  123. fh.write("#endif\n")
  124. for i in flat_list:
  125. print(f"TEXT: {i}")
  126. translation_list: FileList = getFileFromPath([translation], '.py')
  127. for t in translation_list:
  128. name = os.path.splitext(os.path.split(t)[-1])[0]
  129. if name == 'base':
  130. continue
  131. print(f"tr: {name}")
  132. with open(t, "r", encoding="utf-8") as f:
  133. code = f.read()
  134. var = {}
  135. for i in flat_list:
  136. var[i] = flat_list[i][0][1:-2] # [1:-2] 去除引号
  137. exec(code, var, var)
  138. with open(os.path.join(translation_output, f"{name}.c"), "w", encoding="utf-8") as fc:
  139. fc.write(f"#include \"{export_h}.h\"\n") # 需要导出
  140. for i in flat_list: # 根据 flat_list 生成变量
  141. var[i].strip() # 去除首尾不需要的符号
  142. res = var[i].replace('\n', '\\n').replace('\r', '')
  143. fc.write(f"{export} const char *const HT__TEXT_{base_name}_{i} = \"{res}\";\n")