Skip to content

Latest commit

 

History

History
92 lines (60 loc) · 6.82 KB

README.md

File metadata and controls

92 lines (60 loc) · 6.82 KB

learning--qt-file-explorer

It is a small project in which I learn C++ and Qt by creating a simple file explorer.

It covers topics like widgets and models, signals and slots, smart pointers, and a bit of concurrency (in image_preview_widget.cc).

Development setup

macOS

  • I was developing this project on macOS Sonoma 14.4.1, on a MacBook with Apple M1 Max CPU.
  • I installed Qt6 with an official installer of the open source version
    • during an installation process I picked Qt > Qt 6.7.1 > Desktop and deselected everything else
    • in my case valid CMAKE_PREFIX_PATH (used below) was: /Users/<username>/Qt6/6.7.1/macos/lib/cmake"
  • I was developing it in CLion 2024.1.1.
    • I had to add -DCMAKE_PREFIX_PATH="<path_to_my_qt_cmake>" in "CMake options" section of my CLion CMake settings
  • I had CMake 3.29.3 installed with brew install cmake.

The set of commands to build and run this project in Debug mode (on macOS):

./cleanup.sh
CMAKE_PREFIX_PATH="<path_to_your_qt_cmake>" ./run_debug.sh

For Release mode please use ./run_release.sh.

Windows

I do not have any Windows machine right know which I could use for setting up Qt development env.

To build this project on Windows, please take a look at macOS section above and try to replicate those steps since the steps should be similar (e.g. to have Qt 6 installed, to have cmake available, etc.)

Usage

app overview with info about what is what

Shortcuts

action macOS Windows
Go to: Home Ctrl+H Win+H
Go to: Downloads Ctrl+D Win+D
Back Cmd+[ Alt+Left
Forward Cmd+] Alt+Right
Switch to icons/list Ctrl+V Win+V
Show/hide preview Ctrl+P Win+P
Reset layout Ctrl+R Win+R
Select and preview next/prev dir in the picker up/down, then Return up/down, then Return
Select next/prev file/dir on the list up/down up/down

File preview and performance concerns

At some point I experienced performance issues when quickly changing the currently selected file (with a constantly pressed up/down arrow).

There are basically 3 types of a preview implemented in this file explorer:

  • ImagePreviewWidget for images in formats supported by Qt (QImageReader::supportedImageFormats()), based on QLabel and QPixmap(pathToImage)
  • MarkdownPreviewWidget, JsonPreviewWidget, TextPreviewWidget – three similar preview widgets based on QTextEdit/QPlainTextEdit
  • FileNamePreviewWidget as a fallback one, which simply prints a file's or dir's name inside a QLabel

First, I noticed GUI freezes when navigating over a 3 MB .txt/.json files. I was trying to put some work into threads, but apparently the performance issue was in QLabel::setText(hugeText) itself, which is supposed to be run on the main thread. Luckily, after some googling I learnt there is a dedicated QPlainTextEdit widget for such type of content. I tried it an realized the performance issues are gone. Apparently, according to what I google / read in docs, QPlainTextEdit loads and renders only as much text as needed for a given visible area (which is a tiny fraction of an entire file).

Second, I tackled another GUI freeze, a less impactful one, but still apparent – for a 76 MB PNG image file. After some debugging I learned that the freeze time is mostly spent on QPixmap(pathToImage). I watched several YouTube videos on how to put work into threads in Qt and tried some of them. In the end I cleaned up the implementation to just a simple combination of QFutureWatcher<QPixmap> and setting it to observe a new QFuture<QPixmap> every time an image is selected to be previewed. The performance issue was gone.

For images, I also had to deal with a race condition: since my images were loading in a separate thread, it was easy to start preview of a huge image, then switch to preview of a tiny image, then the huge image replaces the tiny one, because if finished loading. I addressed it by both a) setting a new QFuture on the QFutureWatcher every time a loading is initiated and b) calling .cancel() on the QFutureWatcher to avoid unnecessary work in a background.

Utils

Since I do not have a strong know-how about profiles in C++ ecosystem, I wrote a simple JavaScript tool count_destructors.js which analyzes the app's output and let me know if C++ constructors (indicated by my custom log line) have their corresponding deconstructors called.

Of course, this approach is not ideal, but it was easy to implement and already helped me to find some memory management issues.

You can run it with:

node utils/count_destructors.js < logs/<log_filename>

Where:

  • node is a Node.js executable (the version I use is v18.18.2)
  • <log_filename> is a filename of a log file generated by the file explorer app when run in DEBUG mode (e.g. through ./run_debug.sh). You can check related code in message_handler_with_file_output.h

Other notes

  • I took a look at official Google C++ Style Guide and applied some of those on this codebase here.
  • Code formatting was performed by CLion IDE (with some configuration modifications applied by me to match aforementioned Google's guide a bit). It might result in some strange indentation, especially for long constructor signatures (mostly because the impose line length limit is 80 chars).
  • I was thinking about adding a language switcher between English and Polish. This is why I prepared the GUI labels with wrapping them with tr(…) calls. In the end, I decided setting up the whole Qt Linguist workflow would be a low gain for a high cost of adding more dependencies and more tooling to be run in a way I cannot test cross platform (I have no decent Windows machine right now).
  • Personally I am not a fan of inheriting from QWidget and its subclasses to create my own classes, because it makes them expose way much broader API than expected to be used in my app. But, while learning Qt I got under an impression that it is idiomatic to create new widgets as subclasses of Qt classes and I decided to not mess up with that.