Skip to content

Commit d31dc8f

Browse files
committed
Merge bitcoin/bitcoin#33200: cmake: Introduce translate.cmake script for translate target
05255d5 cmake: Drop dependency on sed for translate target (Daniel Pfeifer) d5054be cmake: Introduce translate.cmake script for translate target (Daniel Pfeifer) Pull request description: Using `file(GLOB)` in the generates step is discouraged because the globbing result may be out of date when the target is built. Performing the globbing in a script that is executed as the build target means the result is always reproducable and the overhead of globbing is only paid when used. As a follow up, the dependency on `sed` may be removed by performing the replacement with cmake. Also, the logic from extract_strings_qt.py can be migrated to cmake. ACKs for top commit: hebasto: ACK 05255d5. Tree-SHA512: ae55d9199e6294109b37e5e18f21f2b0e582c1f9903421cf22a237cfdbd215cc431706563b3caa03068cdba79f936b019526638fe3a1f83b4f01a72817e39be1
2 parents 57e8f34 + 05255d5 commit d31dc8f

File tree

2 files changed

+90
-58
lines changed

2 files changed

+90
-58
lines changed

share/qt/translate.cmake

Lines changed: 82 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
# Copyright (c) 2025 The Bitcoin Core developers
2+
# Distributed under the MIT software license, see the accompanying
3+
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
4+
5+
set(input_variables
6+
PROJECT_SOURCE_DIR
7+
COPYRIGHT_HOLDERS
8+
LCONVERT_EXECUTABLE
9+
LUPDATE_EXECUTABLE
10+
PYTHON_EXECUTABLE
11+
XGETTEXT_EXECUTABLE
12+
)
13+
14+
foreach(var IN LISTS input_variables)
15+
if(NOT DEFINED ${var})
16+
message(FATAL_ERROR "Variable '${var}' is not defined!")
17+
endif()
18+
endforeach()
19+
20+
file(GLOB_RECURSE translatable_sources
21+
"${PROJECT_SOURCE_DIR}/src/*.h"
22+
"${PROJECT_SOURCE_DIR}/src/*.cpp"
23+
"${PROJECT_SOURCE_DIR}/src/*.mm"
24+
)
25+
26+
file(GLOB_RECURSE qt_translatable_sources
27+
"${PROJECT_SOURCE_DIR}/src/qt/*.h"
28+
"${PROJECT_SOURCE_DIR}/src/qt/*.cpp"
29+
"${PROJECT_SOURCE_DIR}/src/qt/*.mm"
30+
)
31+
32+
file(GLOB ui_files
33+
"${PROJECT_SOURCE_DIR}/src/qt/forms/*.ui"
34+
)
35+
36+
set(subtrees crc32c crypto/ctaes leveldb minisketch secp256k1)
37+
set(exclude_dirs bench compat crypto support test univalue)
38+
foreach(directory IN LISTS subtrees exclude_dirs)
39+
list(FILTER translatable_sources
40+
EXCLUDE REGEX "${PROJECT_SOURCE_DIR}/src/${directory}/.*"
41+
)
42+
endforeach()
43+
44+
execute_process(
45+
COMMAND ${CMAKE_COMMAND} -E env
46+
XGETTEXT=${XGETTEXT_EXECUTABLE}
47+
COPYRIGHT_HOLDERS=${COPYRIGHT_HOLDERS}
48+
${PYTHON_EXECUTABLE}
49+
${CMAKE_CURRENT_LIST_DIR}/extract_strings_qt.py
50+
${translatable_sources}
51+
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}/src
52+
COMMAND_ERROR_IS_FATAL ANY
53+
)
54+
55+
execute_process(
56+
COMMAND ${LUPDATE_EXECUTABLE}
57+
-no-obsolete
58+
-I ${PROJECT_SOURCE_DIR}/src
59+
-locations relative
60+
${ui_files}
61+
${qt_translatable_sources}
62+
${PROJECT_SOURCE_DIR}/src/qt/bitcoinstrings.cpp
63+
-ts ${PROJECT_SOURCE_DIR}/src/qt/locale/bitcoin_en.ts
64+
COMMAND_ERROR_IS_FATAL ANY
65+
)
66+
67+
execute_process(
68+
COMMAND ${LCONVERT_EXECUTABLE}
69+
-drop-translations
70+
-o ${PROJECT_SOURCE_DIR}/src/qt/locale/bitcoin_en.xlf
71+
-i ${PROJECT_SOURCE_DIR}/src/qt/locale/bitcoin_en.ts
72+
COMMAND_ERROR_IS_FATAL ANY
73+
)
74+
75+
file(READ "${PROJECT_SOURCE_DIR}/src/qt/locale/bitcoin_en.xlf" bitcoin_en)
76+
string(REPLACE "source-language=\"en\" target-language=\"en\""
77+
"source-language=\"en\"" bitcoin_en "${bitcoin_en}"
78+
)
79+
string(REGEX REPLACE " *<target xml:space=\"preserve\"></target>\n"
80+
"" bitcoin_en "${bitcoin_en}"
81+
)
82+
file(WRITE "${PROJECT_SOURCE_DIR}/src/qt/locale/bitcoin_en.xlf" "${bitcoin_en}")

src/qt/CMakeLists.txt

Lines changed: 8 additions & 58 deletions
Original file line numberDiff line numberDiff line change
@@ -289,69 +289,19 @@ if(BUILD_GUI_TESTS)
289289
add_subdirectory(test)
290290
endif()
291291

292-
293-
# Gets sources to be parsed to gather translatable strings.
294-
function(get_translatable_sources var)
295-
set(result)
296-
set(targets)
297-
foreach(dir IN ITEMS ${ARGN})
298-
get_directory_property(dir_targets DIRECTORY ${PROJECT_SOURCE_DIR}/${dir} BUILDSYSTEM_TARGETS)
299-
list(APPEND targets ${dir_targets})
300-
endforeach()
301-
foreach(target IN LISTS targets)
302-
get_target_property(target_sources ${target} SOURCES)
303-
if(target_sources)
304-
foreach(source IN LISTS target_sources)
305-
# Get an expression from the generator expression, if any.
306-
if(source MATCHES ":([^>]+)>$")
307-
set(source ${CMAKE_MATCH_1})
308-
endif()
309-
cmake_path(GET source EXTENSION LAST_ONLY ext)
310-
if(ext STREQUAL ".qrc")
311-
continue()
312-
endif()
313-
if(NOT IS_ABSOLUTE source)
314-
get_target_property(target_source_dir ${target} SOURCE_DIR)
315-
cmake_path(APPEND target_source_dir ${source} OUTPUT_VARIABLE source)
316-
endif()
317-
get_property(is_generated
318-
SOURCE ${source} TARGET_DIRECTORY ${target}
319-
PROPERTY GENERATED
320-
)
321-
if(NOT is_generated)
322-
list(APPEND result ${source})
323-
endif()
324-
endforeach()
325-
endif()
326-
endforeach()
327-
set(${var} ${result} PARENT_SCOPE)
328-
endfunction()
329-
330292
find_program(XGETTEXT_EXECUTABLE xgettext)
331-
find_program(SED_EXECUTABLE sed)
332293
if(NOT XGETTEXT_EXECUTABLE)
333294
add_custom_target(translate
334295
COMMAND ${CMAKE_COMMAND} -E echo "Error: GNU gettext-tools not found"
335296
)
336-
elseif(NOT SED_EXECUTABLE)
337-
add_custom_target(translate
338-
COMMAND ${CMAKE_COMMAND} -E echo "Error: GNU sed not found"
339-
)
340297
else()
341-
set(translatable_sources_directories src src/qt src/util)
342-
if(ENABLE_WALLET)
343-
list(APPEND translatable_sources_directories src/wallet)
344-
endif()
345-
get_translatable_sources(translatable_sources ${translatable_sources_directories})
346-
get_translatable_sources(qt_translatable_sources src/qt)
347-
file(GLOB ui_files ${CMAKE_CURRENT_SOURCE_DIR}/forms/*.ui)
348-
add_custom_target(translate
349-
COMMAND ${CMAKE_COMMAND} -E env XGETTEXT=${XGETTEXT_EXECUTABLE} COPYRIGHT_HOLDERS=${COPYRIGHT_HOLDERS} $<TARGET_FILE:Python3::Interpreter> ${PROJECT_SOURCE_DIR}/share/qt/extract_strings_qt.py ${translatable_sources}
350-
COMMAND Qt6::lupdate -no-obsolete -I ${PROJECT_SOURCE_DIR}/src -locations relative ${CMAKE_CURRENT_SOURCE_DIR}/bitcoinstrings.cpp ${ui_files} ${qt_translatable_sources} -ts ${CMAKE_CURRENT_SOURCE_DIR}/locale/bitcoin_en.ts
351-
COMMAND Qt6::lconvert -drop-translations -o ${CMAKE_CURRENT_SOURCE_DIR}/locale/bitcoin_en.xlf -i ${CMAKE_CURRENT_SOURCE_DIR}/locale/bitcoin_en.ts
352-
COMMAND ${SED_EXECUTABLE} -i.old -e "s|source-language=\"en\" target-language=\"en\"|source-language=\"en\"|" -e "/<target xml:space=\"preserve\"><\\/target>/d" ${CMAKE_CURRENT_SOURCE_DIR}/locale/bitcoin_en.xlf
353-
COMMAND ${CMAKE_COMMAND} -E rm ${CMAKE_CURRENT_SOURCE_DIR}/locale/bitcoin_en.xlf.old
354-
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}/src
355-
VERBATIM
298+
add_custom_target(translate COMMAND ${CMAKE_COMMAND}
299+
-D "PROJECT_SOURCE_DIR=${PROJECT_SOURCE_DIR}"
300+
-D "COPYRIGHT_HOLDERS=${COPYRIGHT_HOLDERS}"
301+
-D "LCONVERT_EXECUTABLE=$<TARGET_FILE:Qt6::lconvert>"
302+
-D "LUPDATE_EXECUTABLE=$<TARGET_FILE:Qt6::lupdate>"
303+
-D "PYTHON_EXECUTABLE=$<TARGET_FILE:Python3::Interpreter>"
304+
-D "XGETTEXT_EXECUTABLE=${XGETTEXT_EXECUTABLE}"
305+
-P ${PROJECT_SOURCE_DIR}/share/qt/translate.cmake
356306
)
357307
endif()

0 commit comments

Comments
 (0)