ctkMacroWrapPythonQt.cmake 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. ###########################################################################
  2. #
  3. # Library: CTK
  4. #
  5. # Copyright (c) 2010 Kitware Inc.
  6. #
  7. # Licensed under the Apache License, Version 2.0 (the "License");
  8. # you may not use this file except in compliance with the License.
  9. # You may obtain a copy of the License at
  10. #
  11. # http://www.commontk.org/LICENSE
  12. #
  13. # Unless required by applicable law or agreed to in writing, software
  14. # distributed under the License is distributed on an "AS IS" BASIS,
  15. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  16. # See the License for the specific language governing permissions and
  17. # limitations under the License.
  18. #
  19. ###########################################################################
  20. #
  21. # ctkMacroWrapPythonQt
  22. #
  23. #
  24. # Depends on:
  25. # PythonQt
  26. # PythonQtGenerator (Only if IS_WRAP_FULL is TRUE)
  27. # PythonInterp (See function reSearchFile)
  28. #
  29. #
  30. # The different parameters are:
  31. #
  32. # WRAPPING_NAMESPACE: Namespace that should contain the library. For example: org.commontk
  33. #
  34. # TARGET ...........: Name of the wrapped library. For example: CTKWidget
  35. #
  36. # SRCS_LIST_NAME ...: Name of the variable that should contain the generated wrapper source.
  37. # For example: KIT_PYTHONQT_SRCS
  38. #
  39. # SOURCES ..........: List of source files that should be wrapped.
  40. #
  41. # IS_WRAP_FULL .....: Indicate if a Full wrapping if desired.
  42. #
  43. #
  44. # LOG FILE:
  45. # File ctkMacroWrapPythonQt_log.txt will be created in the current directory.
  46. # It will contain the list of file and the reason why a given class hasn't been wrapped.
  47. #
  48. set(verbose 0)
  49. #
  50. # Convenient function allowing to log the reason why a given class hasn't been wrapped
  51. # If verbose=1, it will also be displayed on the standard output
  52. #
  53. FUNCTION(ctkMacroWrapPythonQt_log msg)
  54. IF(verbose)
  55. MESSAGE(${msg})
  56. ENDIF()
  57. FILE(APPEND "${CMAKE_CURRENT_BINARY_DIR}/ctkMacroWrapPythonQt_log.txt" "${msg}\n")
  58. ENDFUNCTION()
  59. #
  60. # Convenient function allowing to invoke re.search(regex, string) using the given interpreter.
  61. # Note that is_matching will be set to True if there is a match
  62. #
  63. FUNCTION(ctkMacroWrapPythonQt_reSearchFile python_exe python_library_path regex file is_matching)
  64. set(python_cmd "import re
  65. f = open('${file}', 'r')
  66. res = re.search('${regex}', f.read(), re.MULTILINE)
  67. if res == None: print 'FALSE'
  68. else: print 'TRUE'
  69. ")
  70. #message("python_cmd: ${python_cmd}")
  71. ctkMacroSetPaths("${python_library_path}")
  72. EXECUTE_PROCESS(
  73. COMMAND ${python_exe} -c ${python_cmd}
  74. RESULT_VARIABLE result
  75. OUTPUT_VARIABLE output
  76. ERROR_VARIABLE error
  77. OUTPUT_STRIP_TRAILING_WHITESPACE
  78. )
  79. IF(result)
  80. MESSAGE(SEND_ERROR "reSearchFile - Problem with regex: ${regex}\n${error}")
  81. ENDIF()
  82. SET(is_matching ${output} PARENT_SCOPE)
  83. ENDFUNCTION()
  84. MACRO(ctkMacroWrapPythonQt WRAPPING_NAMESPACE TARGET SRCS_LIST_NAME SOURCES IS_WRAP_FULL)
  85. # Sanity check
  86. IF(IS_WRAP_FULL AND NOT EXISTS "${PYTHONQTGENERATOR_EXECUTABLE}")
  87. MESSAGE(SEND_ERROR "PYTHONQTGENERATOR_EXECUTABLE not specified or inexistent when calling ctkMacroWrapPythonQt")
  88. ENDIF()
  89. # TODO: this find package seems not to work when called form a superbuild, but the call is needed
  90. # in general to find the python interpreter. In CTK, the toplevel CMakeLists.txt does the find
  91. # package so this is a no-op. Other uses of this file may need to have this call so it is still enabled.
  92. find_package(PythonInterp)
  93. IF(NOT PYTHONINTERP_FOUND)
  94. MESSAGE(SEND_ERROR "PYTHON_EXECUTABLE not specified or inexistent when calling ctkMacroWrapPythonQt")
  95. ENDIF()
  96. # Extract python lib path
  97. get_filename_component(PYTHON_DIR_PATH ${PYTHON_EXECUTABLE} PATH)
  98. set(PYTHON_LIBRARY_PATH ${PYTHON_DIR_PATH}/../lib)
  99. IF(WIN32)
  100. set(PYTHON_LIBRARY_PATH ${PYTHON_DIR_PATH})
  101. ENDIF(WIN32)
  102. # Clear log file
  103. FILE(WRITE "${CMAKE_CURRENT_BINARY_DIR}/ctkMacroWrapPythonQt_log.txt" "")
  104. # Convert wrapping namespace to subdir
  105. STRING(REPLACE "." "_" WRAPPING_NAMESPACE_UNDERSCORE ${WRAPPING_NAMESPACE})
  106. SET(SOURCES_TO_WRAP)
  107. # For each class
  108. FOREACH(FILE ${SOURCES})
  109. SET(skip_wrapping FALSE)
  110. IF(NOT skip_wrapping)
  111. # Skip wrapping if file is NOT regular header
  112. IF(NOT ${FILE} MATCHES "^.*\\.[hH]$")
  113. SET(skip_wrapping TRUE)
  114. ctkMacroWrapPythonQt_log("${FILE}: skipping - Not a regular header")
  115. ENDIF()
  116. ENDIF()
  117. IF(NOT skip_wrapping)
  118. # Skip wrapping if file is a pimpl header
  119. IF(${FILE} MATCHES "^.*_[pP]\\.[hH]$")
  120. SET(skip_wrapping TRUE)
  121. ctkMacroWrapPythonQt_log("${FILE}: skipping - Pimpl header (*._p.h)")
  122. ENDIF()
  123. ENDIF()
  124. IF(NOT skip_wrapping)
  125. # Skip wrapping if file should excluded
  126. SET(skip_wrapping TRUE)
  127. GET_SOURCE_FILE_PROPERTY(TMP_WRAP_EXCLUDE ${FILE} WRAP_EXCLUDE)
  128. IF(NOT TMP_WRAP_EXCLUDE)
  129. SET(skip_wrapping FALSE)
  130. ENDIF()
  131. IF(skip_wrapping)
  132. ctkMacroWrapPythonQt_log("${FILE}: skipping - WRAP_EXCLUDE")
  133. ENDIF()
  134. ENDIF()
  135. # what is the filename without the extension
  136. GET_FILENAME_COMPONENT(TMP_FILENAME ${FILE} NAME_WE)
  137. # Extract classname - NOTE: We assume the filename matches the associated class
  138. SET(className ${TMP_FILENAME})
  139. IF(NOT skip_wrapping)
  140. # Skip wrapping if IS_WRAP_FULL=FALSE and if file do NOT contain Q_OBJECT
  141. IF(NOT IS_WRAP_FULL)
  142. file(READ ${CMAKE_CURRENT_SOURCE_DIR}/${FILE} file_content)
  143. IF(NOT "${file_content}" MATCHES "Q_OBJECT")
  144. SET(skip_wrapping TRUE)
  145. ctkMacroWrapPythonQt_log("${FILE}: skipping - No Q_OBJECT macro")
  146. ENDIF()
  147. ENDIF()
  148. ENDIF()
  149. IF(NOT skip_wrapping)
  150. # Skip wrapping if IS_WRAP_FULL=FALSE and if constructor doesn't match:
  151. # my_class()
  152. # my_class(QObject* newParent ...)
  153. # my_class(QWidget* newParent ...)
  154. IF(NOT IS_WRAP_FULL)
  155. # Constructor with either QWidget or QObject as first parameter
  156. SET(regex "[^~]${className}[\\s\\n]*\\([\\s\\n]*((QObject|QWidget)[\\s\\n]*\\*[\\s\\n]*\\w+[\\s\\n]*(\\=[\\s\\n]*(0|NULL)|,.*\\=.*\\)|\\)|\\)))")
  157. ctkMacroWrapPythonQt_reSearchFile(${PYTHON_EXECUTABLE} ${PYTHON_LIBRARY_PATH}
  158. ${regex} ${CMAKE_CURRENT_SOURCE_DIR}/${FILE} is_matching)
  159. IF(NOT is_matching)
  160. SET(skip_wrapping TRUE)
  161. ctkMacroWrapPythonQt_log("${FILE}: skipping - Missing expected constructor signature")
  162. ENDIF()
  163. ENDIF()
  164. ENDIF()
  165. IF(NOT skip_wrapping)
  166. # Skip wrapping if object has a virtual pure method
  167. SET(regex "virtual[\\w\\n\\s\\*\\(\\)]+\\=[\\s\\n]*(0|NULL)[\\s\\n]*\\x3b")
  168. ctkMacroWrapPythonQt_reSearchFile(${PYTHON_EXECUTABLE} ${PYTHON_LIBRARY_PATH}
  169. ${regex} ${CMAKE_CURRENT_SOURCE_DIR}/${FILE} is_matching)
  170. IF(is_matching)
  171. SET(skip_wrapping TRUE)
  172. ctkMacroWrapPythonQt_log("${FILE}: skipping - Contains a virtual pure method")
  173. ENDIF()
  174. ENDIF()
  175. # if we should wrap it
  176. IF (NOT skip_wrapping)
  177. # the input file might be full path so handle that
  178. GET_FILENAME_COMPONENT(TMP_FILEPATH ${FILE} PATH)
  179. # compute the input filename
  180. IF (TMP_FILEPATH)
  181. SET(TMP_INPUT ${TMP_FILEPATH}/${TMP_FILENAME}.h)
  182. ELSE (TMP_FILEPATH)
  183. SET(TMP_INPUT ${CMAKE_CURRENT_SOURCE_DIR}/${TMP_FILENAME}.h)
  184. ENDIF (TMP_FILEPATH)
  185. LIST(APPEND SOURCES_TO_WRAP ${TMP_INPUT})
  186. ENDIF()
  187. ENDFOREACH()
  188. # PythonQtGenerator expects a colon ':' separated list
  189. SET(INCLUDE_DIRS_TO_WRAP)
  190. FOREACH(include ${CTK_BASE_INCLUDE_DIRS})
  191. SET(INCLUDE_DIRS_TO_WRAP "${INCLUDE_DIRS_TO_WRAP}:${include}")
  192. ENDFOREACH()
  193. # Prepare custom_command argument
  194. SET(SOURCES_TO_WRAP_ARG)
  195. FOREACH(source ${SOURCES_TO_WRAP})
  196. SET(SOURCES_TO_WRAP_ARG "${SOURCES_TO_WRAP_ARG}^^${source}")
  197. ENDFOREACH()
  198. # Define wrap type and wrap intermediate directory
  199. SET(wrap_type "Light")
  200. SET(wrap_int_dir generated_cpp/${WRAPPING_NAMESPACE_UNDERSCORE}_${TARGET}/)
  201. SET(extra_files )
  202. IF(${IS_WRAP_FULL})
  203. SET(wrap_type "Full")
  204. SET(extra_files ${wrap_int_dir}ctkPythonQt_${TARGET}_masterinclude.h)
  205. ENDIF()
  206. #message("wrap_type:${wrap_type} - wrap_int_dir:${wrap_int_dir}")
  207. # Create intermediate output directory
  208. EXECUTE_PROCESS(COMMAND ${CMAKE_COMMAND} -E make_directory ${CMAKE_CURRENT_BINARY_DIR}/${wrap_int_dir})
  209. set(wrapper_init_cpp_filename ${WRAPPING_NAMESPACE_UNDERSCORE}_${TARGET}_init.cpp)
  210. set(wrapper_init_cpp_file ${CMAKE_CURRENT_BINARY_DIR}/${wrap_int_dir}${wrapper_init_cpp_filename})
  211. # Custom command allow to generate ${WRAPPING_NAMESPACE_UNDERSCORE}_${TARGET}_init.cpp and
  212. # associated wrappers ${WRAPPING_NAMESPACE_UNDERSCORE}_${TARGET}{0-N}.cpp
  213. ADD_CUSTOM_COMMAND(
  214. OUTPUT ${wrap_int_dir}${wrapper_init_cpp_filename} ${extra_files}
  215. DEPENDS ${pythonqtgenerator_executable_depends} ${SOURCES_TO_WRAP} ${CTK_CMAKE_DIR}/ctkScriptWrapPythonQt_${wrap_type}.cmake
  216. COMMAND ${CMAKE_COMMAND}
  217. -DPYTHONQTGENERATOR_EXECUTABLE:FILEPATH=${PYTHONQTGENERATOR_EXECUTABLE}
  218. -DPYTHON_EXECUTABLE:FILEPATH=${PYTHON_EXECUTABLE}
  219. -DPYTHON_LIBRARY_PATH:PATH=${PYTHON_LIBRARY_PATH}
  220. -DWRAPPING_NAMESPACE:STRING=${WRAPPING_NAMESPACE}
  221. -DTARGET:STRING=${TARGET}
  222. -DSOURCES:STRING=${SOURCES_TO_WRAP_ARG}
  223. -DINCLUDE_DIRS:STRING=${INCLUDE_DIRS_TO_WRAP}
  224. -DOUTPUT_DIR:PATH=${CMAKE_CURRENT_BINARY_DIR}
  225. -DWRAP_INT_DIR:STRING=${wrap_int_dir}
  226. -DQT_QMAKE_EXECUTABLE:FILEPATH=${QT_QMAKE_EXECUTABLE}
  227. -P ${CTK_CMAKE_DIR}/ctkScriptWrapPythonQt_${wrap_type}.cmake
  228. COMMENT "PythonQt ${wrap_type} Wrapping - Generating ${wrapper_init_cpp_filename}"
  229. VERBATIM
  230. )
  231. # Clear variable
  232. SET(moc_flags)
  233. # Grab moc flags
  234. QT4_GET_MOC_FLAGS(moc_flags)
  235. # Prepare custom_command argument
  236. SET(moc_flags_arg)
  237. FOREACH(flag ${moc_flags})
  238. SET(moc_flags_arg "${moc_flags_arg}^^${flag}")
  239. ENDFOREACH()
  240. SET(wrapper_master_moc_filename moc_${WRAPPING_NAMESPACE_UNDERSCORE}_${TARGET}_all.cpp)
  241. SET(wrapper_master_moc_file ${CMAKE_CURRENT_BINARY_DIR}/${wrap_int_dir}${wrapper_master_moc_filename})
  242. # Custom command allowing to call moc to process the wrapper headers
  243. ADD_CUSTOM_COMMAND(
  244. OUTPUT ${wrap_int_dir}${wrapper_master_moc_filename}
  245. DEPENDS ${wrap_int_dir}${wrapper_init_cpp_filename} ${extra_files} ${CTK_CMAKE_DIR}/ctkScriptMocPythonQtWrapper.cmake
  246. COMMAND ${CMAKE_COMMAND}
  247. -DWRAPPING_NAMESPACE:STRING=${WRAPPING_NAMESPACE}
  248. -DTARGET:STRING=${TARGET}
  249. -DMOC_FLAGS:STRING=${moc_flags_arg}
  250. -DWRAP_INT_DIR:STRING=${wrap_int_dir}
  251. -DWRAPPER_MASTER_MOC_FILE:STRING=${wrapper_master_moc_file}
  252. -DOUTPUT_DIR:PATH=${CMAKE_CURRENT_BINARY_DIR}
  253. -DQT_MOC_EXECUTABLE:FILEPATH=${QT_MOC_EXECUTABLE}
  254. -P ${CTK_CMAKE_DIR}/ctkScriptMocPythonQtWrapper.cmake
  255. COMMENT "PythonQt ${wrap_type} Wrapping - Moc'ing ${WRAPPING_NAMESPACE_UNDERSCORE}_${TARGET} wrapper headers"
  256. VERBATIM
  257. )
  258. #The following files are generated
  259. SET_SOURCE_FILES_PROPERTIES(
  260. ${wrap_int_dir}${wrapper_init_cpp_filename}
  261. ${wrap_int_dir}${wrapper_master_moc_filename}
  262. PROPERTIES GENERATED TRUE)
  263. # Create the Init File
  264. SET(${SRCS_LIST_NAME}
  265. ${${SRCS_LIST_NAME}}
  266. ${wrap_int_dir}${wrapper_init_cpp_filename}
  267. ${wrap_int_dir}${wrapper_master_moc_filename})
  268. #
  269. # Let's include the headers associated with PythonQt
  270. #
  271. FIND_PACKAGE(PythonQt)
  272. IF(NOT PYTHONQT_FOUND)
  273. MESSAGE(FATAL_ERROR "error: PythonQt package is required to build ${TARGET}PythonQt")
  274. ENDIF()
  275. INCLUDE_DIRECTORIES(${PYTHON_INCLUDE_DIRS} ${PYTHONQT_INCLUDE_DIR})
  276. ENDMACRO()