DeployQt4.cmake 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. # - Functions to help assemble a standalone Qt4 executable.
  2. # A collection of CMake utility functions useful for deploying
  3. # Qt4 executables.
  4. #
  5. # The following functions are provided by this module:
  6. # write_qt4_conf
  7. # resolve_qt4_paths
  8. # fixup_qt4_executable
  9. # install_qt4_plugin_path
  10. # install_qt4_plugin
  11. # install_qt4_executable
  12. # Requires CMake 2.6 or greater because it uses function and
  13. # PARENT_SCOPE. Also depends on BundleUtilities.cmake.
  14. #
  15. # WRITE_QT4_CONF(<qt_conf_dir> <qt_conf_contents>)
  16. # Writes a qt.conf file with the <qt_conf_contents> into <qt_conf_dir>.
  17. #
  18. # RESOLVE_QT4_PATHS(<paths_var> [<executable_path>])
  19. # Loop through <paths_var> list and if any don't exist resolve them
  20. # relative to the <executable_path> (if supplied) or the CMAKE_INSTALL_PREFIX.
  21. #
  22. # FIXUP_QT4_EXECUTABLE(<executable> [<qtplugins> <libs> <dirs> <plugins_dir> <request_qt_conf>])
  23. # Copies Qt plugins, writes a Qt configuration file (if needed) and fixes up a
  24. # Qt4 executable using BundleUtilities so it is standalone and can be
  25. # drag-and-drop copied to another machine as long as all of the system
  26. # libraries are compatible.
  27. #
  28. # <executable> should point to the executable to be fixed-up.
  29. #
  30. # <qtplugins> should contain a list of the names or paths of any Qt plugins
  31. # to be installed.
  32. #
  33. # <libs> will be passed to BundleUtilities and should be a list of any already
  34. # installed plugins, libraries or executables to also be fixed-up.
  35. #
  36. # <dirs> will be passed to BundleUtilities and should contain and directories
  37. # to be searched to find library dependencies.
  38. #
  39. # <plugins_dir> allows an custom plugins directory to be used.
  40. #
  41. # <request_qt_conf> will force a qt.conf file to be written even if not needed.
  42. #
  43. # INSTALL_QT4_PLUGIN_PATH(plugin executable copy installed_plugin_path_var <plugins_dir> <component> <configurations>)
  44. # Install (or copy) a resolved <plugin> to the default plugins directory
  45. # (or <plugins_dir>) relative to <executable> and store the result in
  46. # <installed_plugin_path_var>.
  47. #
  48. # If <copy> is set to TRUE then the plugins will be copied rather than
  49. # installed. This is to allow this module to be used at CMake time rather than
  50. # install time.
  51. #
  52. # If <component> is set then anything installed will use this COMPONENT.
  53. #
  54. # INSTALL_QT4_PLUGIN(plugin executable copy installed_plugin_path_var <plugins_dir> <component>)
  55. # Install (or copy) an unresolved <plugin> to the default plugins directory
  56. # (or <plugins_dir>) relative to <executable> and store the result in
  57. # <installed_plugin_path_var>. See documentation of INSTALL_QT4_PLUGIN_PATH.
  58. #
  59. # INSTALL_QT4_EXECUTABLE(<executable> [<qtplugins> <libs> <dirs> <plugins_dir> <request_qt_conf> <component>])
  60. # Installs Qt plugins, writes a Qt configuration file (if needed) and fixes up
  61. # a Qt4 executable using BundleUtilities so it is standalone and can be
  62. # drag-and-drop copied to another machine as long as all of the system
  63. # libraries are compatible. The executable will be fixed-up at install time.
  64. # <component> is the COMPONENT used for bundle fixup and plugin installation.
  65. # See documentation of FIXUP_QT4_BUNDLE.
  66. #=============================================================================
  67. # Copyright 2011 Mike McQuaid <[email protected]>
  68. #
  69. # Distributed under the OSI-approved BSD License (the "License");
  70. # see accompanying file Copyright.txt for details.
  71. #
  72. # This software is distributed WITHOUT ANY WARRANTY; without even the
  73. # implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  74. # See the License for more information.
  75. #=============================================================================
  76. # (To distribute this file outside of CMake, substitute the full
  77. # License text for the above reference.)
  78. # The functions defined in this file depend on the fixup_bundle function
  79. # (and others) found in BundleUtilities.cmake
  80. include(BundleUtilities)
  81. set(DeployQt4_cmake_dir "${CMAKE_CURRENT_LIST_DIR}")
  82. set(DeployQt4_apple_plugins_dir "PlugIns")
  83. function(write_qt4_conf qt_conf_dir qt_conf_contents)
  84. set(qt_conf_path "${qt_conf_dir}/qt.conf")
  85. message(STATUS "Writing ${qt_conf_path}")
  86. file(WRITE "${qt_conf_path}" "${qt_conf_contents}")
  87. endfunction()
  88. function(resolve_qt4_paths paths_var)
  89. set(executable_path ${ARGV1})
  90. set(paths_resolved)
  91. foreach(path ${${paths_var}})
  92. if(EXISTS "${path}")
  93. list(APPEND paths_resolved "${path}")
  94. else()
  95. if(${executable_path})
  96. list(APPEND paths_resolved "${executable_path}/${path}")
  97. else()
  98. list(APPEND paths_resolved "\${CMAKE_INSTALL_PREFIX}/${path}")
  99. endif()
  100. endif()
  101. endforeach()
  102. set(${paths_var} ${paths_resolved} PARENT_SCOPE)
  103. endfunction()
  104. function(fixup_qt4_executable executable)
  105. set(qtplugins ${ARGV1})
  106. set(libs ${ARGV2})
  107. set(dirs ${ARGV3})
  108. set(plugins_dir ${ARGV4})
  109. set(request_qt_conf ${ARGV5})
  110. message(STATUS "fixup_qt4_executable")
  111. message(STATUS " executable='${executable}'")
  112. message(STATUS " qtplugins='${qtplugins}'")
  113. message(STATUS " libs='${libs}'")
  114. message(STATUS " dirs='${dirs}'")
  115. message(STATUS " plugins_dir='${plugins_dir}'")
  116. message(STATUS " request_qt_conf='${request_qt_conf}'")
  117. if(QT_LIBRARY_DIR)
  118. list(APPEND dirs "${QT_LIBRARY_DIR}")
  119. endif()
  120. if(APPLE)
  121. set(qt_conf_dir "${executable}/Contents/Resources")
  122. set(executable_path "${executable}")
  123. set(write_qt_conf TRUE)
  124. if(NOT plugins_dir)
  125. set(plugins_dir "${DeployQt4_apple_plugins_dir}")
  126. endif()
  127. else()
  128. get_filename_component(executable_path "${executable}" PATH)
  129. if(NOT executable_path)
  130. set(executable_path ".")
  131. endif()
  132. set(qt_conf_dir "${executable_path}")
  133. set(write_qt_conf ${request_qt_conf})
  134. endif()
  135. foreach(plugin ${qtplugins})
  136. set(installed_plugin_path "")
  137. install_qt4_plugin("${plugin}" "${executable}" 1 installed_plugin_path)
  138. list(APPEND libs ${installed_plugin_path})
  139. endforeach()
  140. foreach(lib ${libs})
  141. if(NOT EXISTS "${lib}")
  142. message(FATAL_ERROR "Library does not exist: ${lib}")
  143. endif()
  144. endforeach()
  145. resolve_qt4_paths(libs "${executable_path}")
  146. if(write_qt_conf)
  147. set(qt_conf_contents "[Paths]\nPlugins = ${plugins_dir}")
  148. write_qt4_conf("${qt_conf_dir}" "${qt_conf_contents}")
  149. endif()
  150. fixup_bundle("${executable}" "${libs}" "${dirs}")
  151. endfunction()
  152. function(install_qt4_plugin_path plugin executable copy installed_plugin_path_var)
  153. set(plugins_dir ${ARGV4})
  154. set(component ${ARGV5})
  155. set(configurations ${ARGV6})
  156. if(EXISTS "${plugin}")
  157. if(APPLE)
  158. if(NOT plugins_dir)
  159. set(plugins_dir "${DeployQt4_apple_plugins_dir}")
  160. endif()
  161. set(plugins_path "${executable}/Contents/${plugins_dir}")
  162. else()
  163. get_filename_component(plugins_path "${executable}" PATH)
  164. if(NOT plugins_path)
  165. set(plugins_path ".")
  166. endif()
  167. if(plugins_dir)
  168. set(plugins_path "${plugins_path}/${plugins_dir}")
  169. endif()
  170. endif()
  171. set(plugin_group "")
  172. get_filename_component(plugin_path "${plugin}" PATH)
  173. get_filename_component(plugin_parent_path "${plugin_path}" PATH)
  174. get_filename_component(plugin_parent_dir_name "${plugin_parent_path}" NAME)
  175. get_filename_component(plugin_name "${plugin}" NAME)
  176. string(TOLOWER "${plugin_parent_dir_name}" plugin_parent_dir_name)
  177. if("${plugin_parent_dir_name}" STREQUAL "plugins")
  178. get_filename_component(plugin_group "${plugin_path}" NAME)
  179. set(${plugin_group_var} "${plugin_group}")
  180. endif()
  181. set(plugins_path "${plugins_path}/${plugin_group}")
  182. if(${copy})
  183. file(MAKE_DIRECTORY "${plugins_path}")
  184. file(COPY "${plugin}" DESTINATION "${plugins_path}")
  185. else()
  186. if(configurations AND (CMAKE_CONFIGURATION_TYPES OR CMAKE_BUILD_TYPE))
  187. set(configurations CONFIGURATIONS ${configurations})
  188. else()
  189. unset(configurations)
  190. endif()
  191. install(FILES "${plugin}" DESTINATION "${plugins_path}" ${configurations} ${component})
  192. endif()
  193. set(${installed_plugin_path_var} "${plugins_path}/${plugin_name}" PARENT_SCOPE)
  194. endif()
  195. endfunction()
  196. function(install_qt4_plugin plugin executable copy installed_plugin_path_var)
  197. set(plugins_dir ${ARGV4})
  198. set(component ${ARGV5})
  199. if(EXISTS "${plugin}")
  200. install_qt4_plugin_path("${plugin}" "${executable}" "${copy}" "${installed_plugin_path_var}" "${plugins_dir}" "${component}")
  201. else()
  202. if(QT_IS_STATIC)
  203. string(TOUPPER "QT_${plugin}_LIBRARY" plugin_var)
  204. else()
  205. string(TOUPPER "QT_${plugin}_PLUGIN" plugin_var)
  206. endif()
  207. set(plugin_release_var "${plugin_var}_RELEASE")
  208. set(plugin_debug_var "${plugin_var}_DEBUG")
  209. set(plugin_release "${${plugin_release_var}}")
  210. set(plugin_debug "${${plugin_debug_var}}")
  211. if(DEFINED "${plugin_release_var}" AND DEFINED "${plugin_debug_var}" AND NOT EXISTS "${plugin_release}" AND NOT EXISTS "${plugin_debug}")
  212. message(WARNING "Qt plugin \"${plugin}\" not recognized or found.")
  213. endif()
  214. install_qt4_plugin_path("${plugin_release}" "${executable}" "${copy}" "${installed_plugin_path_var}" "${plugins_dir}" "${component}" "Release|RelWithDebInfo|MinSizeRel")
  215. install_qt4_plugin_path("${plugin_debug}" "${executable}" "${copy}" "${installed_plugin_path_var}" "${plugins_dir}" "${component}" "Debug")
  216. endif()
  217. set(installed_plugin_paths ${${installed_plugin_path_var}} PARENT_SCOPE)
  218. endfunction()
  219. function(install_qt4_executable executable)
  220. set(qtplugins ${ARGV1})
  221. set(libs ${ARGV2})
  222. set(dirs ${ARGV3})
  223. set(plugins_dir ${ARGV4})
  224. set(request_qt_conf ${ARGV5})
  225. set(component ${ARGV6})
  226. if(QT_LIBRARY_DIR)
  227. list(APPEND dirs "${QT_LIBRARY_DIR}")
  228. endif()
  229. if(component)
  230. set(component COMPONENT ${component})
  231. else()
  232. unset(component)
  233. endif()
  234. get_filename_component(executable_absolute "${executable}" ABSOLUTE)
  235. if(EXISTS "${QT_QTCORE_LIBRARY_RELEASE}")
  236. gp_file_type("${executable_absolute}" "${QT_QTCORE_LIBRARY_RELEASE}" qtcore_type)
  237. elseif(EXISTS "${QT_QTCORE_LIBRARY_DEBUG}")
  238. gp_file_type("${executable_absolute}" "${QT_QTCORE_LIBRARY_DEBUG}" qtcore_type)
  239. endif()
  240. if(qtcore_type STREQUAL "system")
  241. set(qt_plugins_dir "")
  242. endif()
  243. foreach(plugin ${qtplugins})
  244. set(installed_plugin_paths "")
  245. install_qt4_plugin("${plugin}" "${executable}" 0 installed_plugin_paths "${plugins_dir}" "${component}")
  246. list(APPEND libs ${installed_plugin_paths})
  247. endforeach()
  248. resolve_qt4_paths(libs)
  249. install(CODE
  250. "INCLUDE(\"${DeployQt4_cmake_dir}/DeployQt4.cmake\")
  251. SET(BU_CHMOD_BUNDLE_ITEMS TRUE)
  252. FIXUP_QT4_EXECUTABLE(\"\${CMAKE_INSTALL_PREFIX}/${executable}\" \"\" \"${libs}\" \"${dirs}\" \"${plugins_dir}\" \"${request_qt_conf}\")"
  253. ${component}
  254. )
  255. endfunction()