-
Notifications
You must be signed in to change notification settings - Fork 460
Development
This page describes CopyQ source code and processes for developers.
Pull requests are welcome at github project page.
Try to keep the code style consistent with the existing code.
You can run automated tests if the application is built either in debug mode, with CMake flag -DWITH_TESTS=ON
or QMake flag CONFIG+=tests
(releases are usually build with tests).
Run the tests with following command.
copyq tests
While running tests there must be no keyboard and mouse interaction.
Preferably you can execute the tests in separate virtual environment.
On Linux you can run the tests on virtual X11 server with xvfb-run
.
xvfb-run sh -c 'openbox & sleep 1; copyq tests'
Test invocation examples:
- Run specific tests:
copyq tests commandHelp commandVersion
- Run specific tests for a plugin:
copyq tests 'PLUGINS:pinned' isPinned
- Run tests only for specific plugins:
copyq tests 'PLUGINS:pinned|tags'
- List tests:
copyq tests -functions
- List tests for a plugin:
copyq tests PLUGINS:tags -functions
- Less verbose tests:
copyq tests -silent
- Slower GUI tests:
COPYQ_TESTS_KEYS_WAIT=1000 COPYQ_TESTS_KEY_DELAY=50 copyq tests editItems
The application is written in C++11 and uses Qt framework.
Source code can be build either with CMake (preferred) or QMake.
mkdir build
cd build
cmake \
-DCMAKE_BUILD_TYPE=Debug \
-DCMAKE_INSTALL_PREFIX=/usr/local \
-DWITH_QT5=ON \
-DWITH_TESTS=ON \
..
Most icons in the application are taken from theme by default (which currently works only on Linux) with fallback to built-in icons provided by FontAwesome.
Application logo was created in Blender (scene source is here).
The logo is used for bigger application icon. Smaller icons were created in Inkscape (icon source is here).
There are these system processes:
- main GUI application,
- clipboard monitor (started from main application),
- multiple clients (run scripts in main application).
The main GUI application (or server) can be executed by running copyq
binary without attributes
(session name can be optionally specified on command line).
It creates local server allowing communication with clipboard monitor process and other client processes.
Each user can run multiple main application processes each with unique session name (default name is empty).
Clipboard monitoring happens in separate process because otherwise it would block GUI (in Qt clipboard needs to be accessed in main GUI thread). The process is allowed to crash or loop indefinitely due to bugs on some platforms.
Setting and retrieving clipboard can still happen in GUI thread (copying and pasting in various GUI widgets) but it's preferred to send and receive clipboard data using monitor process.
The monitor process is launched as soon as GUI application starts and is restarted whenever it doesn't respond to keep-alive requests.
Scripting language is Qt Script (mostly same syntax and functions as JavaScript).
API is described in src/scriptable/README.md.
A script can be started by passing arguments to copyq
.
This tells the server (main GUI application) to run the script.
After script finishes, the server sends back output of last command and exit code (non-zero if script crashes).
copyq eval 'read(0,1,2)' # prints first three items in list
copyq eval 'fail()' # exit code will be non-zero
While script is running, it can send print requests to client.
copyq eval 'print("Hello, "); print("World!\n")'
Scripts can ask for stdin from client.
copyq eval 'var client_stdin = input()'
The script run in current directory of client process.
copyq eval 'Dir().absolutePath()'
copyq eval 'execute("ls", "-l").stdout'
Single function call where all arguments are numbers or strings can be executed by passing function name and function arguments on command line. Following commands are equal.
copyq eval 'copy("Hello, World!")'
copyq copy "Hello, World!"
Getting application version or help mustn't require the server to be running.
copyq help
copyq version
Scripts run in separate thread and communicate with main thread by calling methods on an object of ScriptableProxy
class.
If called from non-main thread, these methods invoke a slot on an QObject
in main thread and pass it a function object which simply calls the method again.
bool ScriptableProxy::loadTab(const QString &tabName)
{
// This section is wrapped in an macro so to remove duplicate code.
if (!m_inMainThread) {
// Callable object just wraps the lambda so it's possible to send it to a slot.
auto callable = createCallable([&]{ return loadTab(tabName); });
m_inMainThread = true;
QMetaObject::invokeMethod(m_wnd, "invoke", Qt::BlockingQueuedConnection, Q_ARG(Callable*, &callable));
m_inMainThread = false;
return callable.result();
}
// Now it's possible to call method on an object in main thread.
return m_wnd->loadTab(tabName);
}
Code for various platforms is stored in src/platform.
This leverages amount of #if
s and similar preprocessor directives in common code.
Each supported platform implements PlatformNativeInterface
and createPlatformNativeInterface()
.
The implementations can contain:
- creating Qt application objects,
- clipboard handling (for clipboard monitor),
- focusing window and getting window titles,
- getting system paths,
- setting "autostart" option,
- handling global shortcuts (note: this part is in qxt/).
For unsupported platforms there is simple implementation to get started.
The application binaries and packages are built and tested on multiple CI servers.
-
- Builds packages for OS X.
- Builds and runs tests for Linux binaries with Qt 4.
-
- Builds and runs tests for Ubuntu 16.04 binaries with Qt 5.
- Screenshots are taken while GUI tests are running. These are available if a test fails.
-
- Builds installers and portable packages for Windows with Qt 5.
- Provides downloads for recent commits.
- Release build are based on gcc-compiled binaries (Visual Studio builds are also available).
-
- Builds release packages for various Linux distributions.
-
- Builds beta and unstable packages for various Linux distributions.
-
- Contains coverage report from tests run with Travis CI.
Translations can be done either via Weblate (preferred) or by using Qt utilities.
All GUI strings should be translatable.
This is indicated in code with tr("Some GUI text", "Hints for translators")
.
To add new language for the application follow these steps.
- Edit
copyq.pro
and add file name for new language (translations/copyq_<LANGUAGE>.ts
) toTRANSLATIONS
variable. - Create new language file with
lupdate copyq.pro
. - Add new language file to Git repository.
- Translate with Weblate service or locally with
linguist translations/copyq_<LANGUAGE>.ts
.