diff options
author | Alexandru Croitor <alexandru.croitor@qt.io> | 2018-05-08 14:15:57 +0200 |
---|---|---|
committer | Alexandru Croitor <alexandru.croitor@qt.io> | 2018-05-16 09:11:43 +0000 |
commit | 15273fe0fe52569013dd4811bf9ed770ce7fb287 (patch) | |
tree | f58eab1b18998592f8f9dd541232b58253529aad /examples/scriptableapplication | |
parent | 9d9144b2b44677d2862e389b7a83900ee3e8e44c (diff) |
Add an example that demonstrates bindings to a custom C++ library
A CMake project is included that builds two shared libraries:
1) libuniverse - a hypothetical C++ library for which bindings
need to be created.
2) Universe - a Python module containing bindings to the above
library.
The example showcases the following concepts:
* primitive type bindings (bool, std::string)
* types with object and value semantics
(pass by pointer VS pass by copy)
* inheritance and overriding virtual methods
* ownership of heap-allocated C++ objects
* constructors with default parameters
* general structure of CMakeLists.txt file for generating bindings
Task-number: PYSIDE-597
Change-Id: I7b0f203e2844e815aa611af3de2b50a9aa9b5bfc
Reviewed-by: Friedemann Kleint <Friedemann.Kleint@qt.io>
Diffstat (limited to 'examples/scriptableapplication')
-rw-r--r-- | examples/scriptableapplication/CMakeLists.txt | 16 | ||||
-rw-r--r-- | examples/scriptableapplication/pyside2.pri | 14 | ||||
-rw-r--r-- | examples/scriptableapplication/pyside2_config.py | 285 |
3 files changed, 16 insertions, 299 deletions
diff --git a/examples/scriptableapplication/CMakeLists.txt b/examples/scriptableapplication/CMakeLists.txt index 7e57a291b..4119b6756 100644 --- a/examples/scriptableapplication/CMakeLists.txt +++ b/examples/scriptableapplication/CMakeLists.txt @@ -23,7 +23,7 @@ macro(pyside2_config option output_var) endif() execute_process( - COMMAND python "${CMAKE_SOURCE_DIR}/pyside2_config.py" ${option} + COMMAND python "${CMAKE_SOURCE_DIR}/../utils/pyside2_config.py" ${option} OUTPUT_VARIABLE ${output_var} OUTPUT_STRIP_TRAILING_WHITESPACE) @@ -35,19 +35,19 @@ macro(pyside2_config option output_var) endif() endmacro() -# Get relevant general paths, include paths and linker flags. +# Query for the PySide2 path, Python path, include paths and linker flags. pyside2_config(--pyside2 PYSIDE2_PATH) +pyside2_config(--python-include PYTHON_INCLUDE_DIR) +pyside2_config(--pyside2-include PYSIDE2_INCLUDE_DIR 1) +pyside2_config(--pyside2-shared-libraries-cmake PYSIDE2_SHARED_LIBRARIES 0) +pyside2_config(--python-link-cmake PYTHON_LINKING_DATA 0) + set(SHIBOKEN_PATH "${PYSIDE2_PATH}/shiboken2${CMAKE_EXECUTABLE_SUFFIX}") if(NOT EXISTS ${SHIBOKEN_PATH}) message(FATAL_ERROR "Shiboken executable not found at path: ${SHIBOKEN_PATH}") endif() -pyside2_config(--pyside2 PYSIDE2_DIR) -pyside2_config(--python-include PYTHON_INCLUDE_DIR) -pyside2_config(--pyside2-include PYSIDE2_INCLUDE_DIR 1) -pyside2_config(--pyside2-shared-libraries-cmake PYSIDE2_SHARED_LIBRARIES 0) -pyside2_config(--python-link-cmake PYTHON_LINKING_DATA 0) # Get all relevant Qt include dirs, to pass them on to shiboken. get_property(QT_CORE_INCLUDE_DIRS TARGET Qt5::Core PROPERTY INTERFACE_INCLUDE_DIRECTORIES) @@ -122,7 +122,7 @@ endforeach() # Enable rpaths so that the example can be executed from the build dir. set(CMAKE_SKIP_BUILD_RPATH FALSE) set(CMAKE_BUILD_WITH_INSTALL_RPATH TRUE) -set(CMAKE_INSTALL_RPATH ${PYSIDE2_DIR}) +set(CMAKE_INSTALL_RPATH ${PYSIDE2_PATH}) set(CMAKE_INSTALL_RPATH_USE_LINK_PATH TRUE) # ============================================================================================= # !!! End of dubious section. diff --git a/examples/scriptableapplication/pyside2.pri b/examples/scriptableapplication/pyside2.pri index 59f7fd983..17be4392f 100644 --- a/examples/scriptableapplication/pyside2.pri +++ b/examples/scriptableapplication/pyside2.pri @@ -1,19 +1,21 @@ -PYSIDE2 = $$system(python $$PWD/pyside2_config.py --pyside2) +PYSIDE_CONFIG = $$PWD/../utils/pyside2_config.py + +PYSIDE2 = $$system(python $$PYSIDE_CONFIG --pyside2) isEmpty(PYSIDE2): error(Unable to locate the PySide2 package location) -PYTHON_INCLUDE = $$system(python $$PWD/pyside2_config.py --python-include) +PYTHON_INCLUDE = $$system(python $$PYSIDE_CONFIG --python-include) isEmpty(PYTHON_INCLUDE): error(Unable to locate the Python include headers directory) -PYTHON_LFLAGS = $$system(python $$PWD/pyside2_config.py --python-link) +PYTHON_LFLAGS = $$system(python $$PYSIDE_CONFIG --python-link) isEmpty(PYTHON_LFLAGS): error(Unable to locate the Python library for linking) -PYSIDE2_INCLUDE = $$system(python $$PWD/pyside2_config.py --pyside2-include) +PYSIDE2_INCLUDE = $$system(python $$PYSIDE_CONFIG --pyside2-include) isEmpty(PYSIDE2_INCLUDE): error(Unable to locate the PySide2 include headers directory) -PYSIDE2_LFLAGS = $$system(python $$PWD/pyside2_config.py --pyside2-link) +PYSIDE2_LFLAGS = $$system(python $$PYSIDE_CONFIG --pyside2-link) isEmpty(PYSIDE2_LFLAGS): error(Unable to locate the PySide2 libraries for linking) -PYSIDE2_SHARED_LIBRARIES = $$system(python $$PWD/pyside2_config.py --pyside2-shared-libraries) +PYSIDE2_SHARED_LIBRARIES = $$system(python $$PYSIDE_CONFIG --pyside2-shared-libraries) isEmpty(PYSIDE2_SHARED_LIBRARIES): error(Unable to locate the used PySide2 shared libraries) INCLUDEPATH += "$$PYTHON_INCLUDE" $$PYSIDE2_INCLUDE diff --git a/examples/scriptableapplication/pyside2_config.py b/examples/scriptableapplication/pyside2_config.py deleted file mode 100644 index ce9c707c1..000000000 --- a/examples/scriptableapplication/pyside2_config.py +++ /dev/null @@ -1,285 +0,0 @@ -############################################################################# -## -## Copyright (C) 2018 The Qt Company Ltd. -## Contact: http://www.qt.io/licensing/ -## -## This file is part of the Qt for Python examples of the Qt Toolkit. -## -## $QT_BEGIN_LICENSE:BSD$ -## You may use this file under the terms of the BSD license as follows: -## -## "Redistribution and use in source and binary forms, with or without -## modification, are permitted provided that the following conditions are -## met: -## * Redistributions of source code must retain the above copyright -## notice, this list of conditions and the following disclaimer. -## * Redistributions in binary form must reproduce the above copyright -## notice, this list of conditions and the following disclaimer in -## the documentation and/or other materials provided with the -## distribution. -## * Neither the name of The Qt Company Ltd nor the names of its -## contributors may be used to endorse or promote products derived -## from this software without specific prior written permission. -## -## -## THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS -## "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT -## LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR -## A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT -## OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, -## SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT -## LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, -## DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY -## THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT -## (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE -## OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." -## -## $QT_END_LICENSE$ -## -############################################################################# - -import os, glob, re, sys, imp -from distutils import sysconfig - -usage = """ -Utility to determine include/link options of PySide2 and Python for qmake - -Usage: pyside2_config.py [option] -Options: - --python-include Print Python include path - --python-link Print Python link flags - --pyside2 Print PySide2 location - --pyside2-include Print PySide2 include paths - --pyside2-link Print PySide2 link flags - --pyside2-shared-libraries Print paths of PySide2 shared libraries (.so's, .dylib's, .dll's) - -a Print all - --help/-h Print this help -""" - -def cleanPath(path): - return path if sys.platform != 'win32' else path.replace('\\', '/') - -def sharedLibrarySuffix(): - if sys.platform == 'win32': - return 'lib' - elif sys.platform == 'darwin': - return 'dylib' - # Linux - else: - return 'so.*' - -def sharedLibraryGlobPattern(): - glob = '*.' + sharedLibrarySuffix() - return glob if sys.platform == 'win32' else 'lib' + glob - -def filterPySide2SharedLibraries(list): - def predicate(item): - basename = os.path.basename(item) - if 'shiboken' in basename or 'pyside2' in basename: - return True - return False - result = [item for item in list if predicate(item)] - return result - -# Return qmake link option for a library file name -def linkOption(lib): - # On Linux: - # Since we cannot include symlinks with wheel packages - # we are using an absolute path for the libpyside and libshiboken - # libraries when compiling the project - baseName = os.path.basename(lib) - link = ' -l' - if sys.platform in ['linux', 'linux2']: # Linux: 'libfoo.so' -> '/absolute/path/libfoo.so' - link = lib - elif sys.platform in ['darwin']: # Darwin: 'libfoo.so' -> '-lfoo' - link += os.path.splitext(baseName[3:])[0] - else: # Windows: 'libfoo.dll' -> 'libfoo.dll' - link += os.path.splitext(baseName)[0] - return link - -# Locate PySide2 via package path -def findPySide2(): - for p in sys.path: - if 'site-' in p: - pyside2 = os.path.join(p, 'PySide2') - if os.path.exists(pyside2): - return cleanPath(os.path.realpath(pyside2)) - return None - -# Return version as "3.5" -def pythonVersion(): - return str(sys.version_info[0]) + '.' + str(sys.version_info[1]) - -def pythonInclude(): - return sysconfig.get_python_inc() - -def pythonLinkQmake(): - flags = pythonLinkData() - if sys.platform == 'win32': - libdir = flags['libdir'] - # This will add the "~1" shortcut for directories that - # contain white spaces - # e.g.: "Program Files" to "Progra~1" - for d in libdir.split("\\"): - if " " in d: - libdir = libdir.replace(d, d.split(" ")[0][:-1]+"~1") - return '-L{} -l{}'.format(libdir, flags['lib']) - elif sys.platform == 'darwin': - return '-L{} -l{}'.format(flags['libdir'], flags['lib']) - - else: - # Linux and anything else - return '-L{} -l{}'.format(flags['libdir'], flags['lib']) - -def pythonLinkCmake(): - flags = pythonLinkData() - libdir = flags['libdir'] - lib = re.sub(r'.dll$', '.lib', flags['lib']) - return '{};{}'.format(libdir, lib) - -def pythonLinkData(): - # @TODO Fix to work with static builds of Python - libdir = sysconfig.get_config_var('LIBDIR') - if libdir is None: - libdir = os.path.abspath(os.path.join( - sysconfig.get_config_var('LIBDEST'), "..", "libs")) - version = pythonVersion() - version_no_dots = version.replace('.', '') - - flags = {} - flags['libdir'] = libdir - if sys.platform == 'win32': - suffix = '_d' if any([tup[0].endswith('_d.pyd') for tup in imp.get_suffixes()]) else '' - flags['lib'] = 'python{}{}'.format(version_no_dots, suffix) - - elif sys.platform == 'darwin': - flags['lib'] = 'python{}'.format(version) - - # Linux and anything else - else: - if sys.version_info[0] < 3: - suffix = '_d' if any([tup[0].endswith('_d.so') for tup in imp.get_suffixes()]) else '' - flags['lib'] = 'python{}{}'.format(version, suffix) - else: - flags['lib'] = 'python{}{}'.format(version, sys.abiflags) - - return flags - -def pyside2Include(): - pySide2 = findPySide2() - if pySide2 is None: - return None - return "{0}/include/PySide2 {0}/include/shiboken2".format(pySide2) - -def pyside2Link(): - pySide2 = findPySide2() - if pySide2 is None: - return None - link = "-L{}".format(pySide2) - glob_result = glob.glob(os.path.join(pySide2, sharedLibraryGlobPattern())) - for lib in filterPySide2SharedLibraries(glob_result): - link += ' ' - link += linkOption(lib) - return link - -def pyside2SharedLibrariesData(): - pySide2 = findPySide2() - if pySide2 is None: - return None - - glob_result = glob.glob(os.path.join(pySide2, sharedLibraryGlobPattern())) - filtered_libs = filterPySide2SharedLibraries(glob_result) - libs = [] - if sys.platform == 'win32': - for lib in filtered_libs: - libs.append(os.path.realpath(lib)) - else: - for lib in filtered_libs: - libs.append(lib) - return libs - -def pyside2SharedLibraries(): - libs = pyside2SharedLibrariesData() - if libs is None: - return None - - if sys.platform == 'win32': - if not libs: - return '' - dlls = '' - for lib in libs: - dll = os.path.splitext(lib)[0] + '.dll' - dlls += dll + ' ' - - return dlls - else: - libs_string = '' - for lib in libs: - libs_string += lib + ' ' - return libs_string - -def pyside2SharedLibrariesCmake(): - libs = pyside2SharedLibrariesData() - result = ';'.join(libs) - return result - -option = sys.argv[1] if len(sys.argv) == 2 else '-a' -if option == '-h' or option == '--help': - print(usage) - sys.exit(0) - -generic_error = (' Did you forget to activate your virtualenv? Or perhaps' - ' you forgot to build / install PySide2 into your currently active Python' - ' environment?') -pyside2_error = 'Unable to locate PySide2.' + generic_error -pyside2_libs_error = 'Unable to locate the PySide2 shared libraries.' + generic_error -python_link_error = 'Unable to locate the Python library for linking.' - -if option == '--pyside2' or option == '-a': - pySide2 = findPySide2() - if pySide2 is None: - sys.exit(pyside2_error) - print(pySide2) - -if option == '--pyside2-link' or option == '-a': - l = pyside2Link() - if l is None: - sys.exit(pyside2_error) - - print(l) - -if option == '--pyside2-include' or option == '-a': - i = pyside2Include() - if i is None: - sys.exit(pyside2_error) - print(i) - -if option == '--python-include' or option == '-a': - i = pythonInclude() - if i is None: - sys.exit('Unable to locate the Python include headers directory.') - print(i) - -if option == '--python-link' or option == '-a': - l = pythonLinkQmake() - if l is None: - sys.exit(python_link_error) - print(l) - -if option == '--python-link-cmake' or option == '-a': - l = pythonLinkCmake() - if l is None: - sys.exit(python_link_error) - print(l) - -if option == '--pyside2-shared-libraries' or option == '-a': - l = pyside2SharedLibraries() - if l is None: - sys.exit(pyside2_libs_error) - print(l) - -if option == '--pyside2-shared-libraries-cmake' or option == '-a': - l = pyside2SharedLibrariesCmake() - if l is None: - sys.exit(pyside2_libs_error) - print(l) |