/CppCCQTemplate

A C++ Conan CMake Qt6 Project Template

Primary LanguageCMake

C++Conan CMake Qt6 Project Template

Windows Workflow macos Workflow Ubuntu Workflow codecov

This project uses C++20 and Qt 6.1.2.

The aim of the project is to provide examples of the following common tasks with C++ projects hosted on Github:

  • [x] Using Qt6 with CMake and Conan
  • [x] Using a custom static library
  • [x] Github actions to build on Windows, Mac and Ubuntu
  • [x] A unit test environment
  • [x] Code coverage
  • [x] Using Qt .ui files
  • [x] Using Qt Resource file

Each item is discussed below.

Details

Using Qt6 with CMake and Conan

The only way I could get Qt6 to work with Conan and CMake was by including the /CMake/conan.cmake file from: https://github.com/conan-io/cmake-conan However, in order to use that file I had to use CMake 3.20.1.

I also had to make sure to use the Qt6 located at qt/6.0.1@bincrafters/stable. Using the resource at qt/6.0.1 produced CMake configuration errors.

One problem I ran into was this error that I still don't understand:

ERROR: Conflict in cmake/3.19.1:
    'cmake/3.19.1' requires 'openssl/1.1.1i' while 'qt/6.0.1' requires 'openssl/1.1.1j'.
    To fix this conflict you need to override the package 'openssl' in your root package.

I saw this error about "Conflict in cmake/3.xx" in another project and simply had to override the library in question, which in this case is openssl. However I still do not understand why CMake has any requirement, much less one that is causing an error.

To fix it I had to specify the version of OpenSSL I wanted to use in the root CMakeLists.txt:

conan_cmake_run(REQUIRES
    qt/6.0.1@bincrafters/stable
    openssl/1.1.1k
    ...

Using a custom library

The library is called MyLib it exists in src/MyLib. The App project statically links to it.

Even though no linking is actually done when builing the library by itself, this line must still be included in the library's CMake file:

target_link_libraries(MyLib PRIVATE
    Qt6::Core
)

Without this the compiler complains that it cannot find QString on the line #include <QString>.

Unit Tests and Github Actions

macos

I had to make sure the latest version of CMake is installed, which was luckly a simple call to brew (even though this didn't work on my local Mac).

The Mac unit tests kept giving me the follow error when the CI system tried to invoke them:

dyld: Library not loaded: @rpath/libQt6Test_debug.6.dylib
  Referenced from: /bin/testMain
  Reason: image not found

This means that CTest could not execute the testMain binary because the testMain binary could not find the file libQt6Test_debug.6.dylib in the @rpath which is the Run Search Path. The RPath is a list of paths embedded in the application that tells the application which files to look in for dynamically linked libraries. When testMain loaded up the first thing it did was iterate through the list of its RPaths and start looking for the file libQt6Test_debug.dylib, which it could not find and gave us this error.

I put code in place to replace the @rpath with a direct path to the library, and once I got all the paths and filename correct, everything worked as expected.

Ubuntu

I could only get Qt6 to build on Github Actions if I used ubuntu-latest. I also hit a GLIB compatibility error that lead me to this Stack Overflow post.

I had forgotten to add the bincrafters repo, but the error I was getting said something like "could not find Qt/6.0.1@bincrafters/stable in repos", which made it sound like the Qt6 library wasn't in the repo. I would have instead expected something like "unknown repo".

And lastly, I had to make sure that the libgl1-mesa-dev library was installed on the build image.

Windows

First, I had to add the build type to the ctest command. Secondly for the tests to run I had to make sure that the Qt DLLs were in the test machine's path adding this in test/CMakeLists.txt:

if (WIN32)
    set_tests_properties(testMain 
        PROPERTIES
        ENVIRONMENT "PATH=${CONAN_BIN_DIRS_QT}\;$ENV{PATH}"
    )
endif()

Code Coverage

Code coverage is only supported on macos. I found useful reference projects here and here.

Instead of creating a new .yml workflow, I tacked this onto the existing macos.yml by adding another matrix variable of on/off with coverage and exluding a Release build with coverage turned on.

I did encounter this warning in the logs:

->  Issue detecting commit SHA. Please run actions/checkout with fetch-depth > 1 or set to 0
    project root: /Users/runner/work/CppCCQTemplate/CppCCQTemplate

Which I fixed by increasing the fetch-depth in my checkout step in macos.yml.

Qt UI File

Adding support for the UI file was simple. The only real thing of note with this was that changes to the .ui file alone will not trigger the auto UIC. Instead I had to touch the corresponsing header or source files.

Qt Resource File

The Qt Resoure file allows you to embed resources (images, data files, etc) directly into the executable that can be loaded and used during runtime. This example loads a resource image into a label object and displays that on the dialog:

auto image = new QImage(":/images/example.png");
auto imglbl = new QLabel();
imglbl->setPixmap(QPixmap::fromImage(*image));
imglbl->setAlignment(Qt::AlignCenter);

The catalog of resources is stored in the resource file which has a simple XML format:

<RCC>
    <qresource prefix="/">
        <file>images/example.png</file>
    </qresource>
</RCC>

The resource file is then linked into the executable like any other source file in CMake:

add_executable(App WIN32 MACOSX_BUNDLE
    res/MyApp.qrc
    ...
)

Building

First we clone the Github repository:

git clone https://github.com/zethon/CppCCQTemplate.git

Then create a build folder inside of the repository's folder and navigate into it:

cd CppCCQTemplate && mkdir build && cd build

At this point we're ready to run CMake, which in turn runs Conan. However, the bintray repository used to download Qt6 may not be configured on the local machine. It can be added to the Conan remote cache with the following command:

conan remote add bincrafters https://api.bintray.com/conan/bincrafters/public-conan

Once we've added the bincrafters repository, we can run Conan and CMake in one step:

cmake .. -DCMAKE_BUILD_TYPE=Debug

In this case we have configured our build system to compile debug versions of the binaries. Now we can build with the following command:

cmake --build .