-
-
Notifications
You must be signed in to change notification settings - Fork 1k
Coding Conventions
Everybody has their own preferred style for writing code, however, in the interest of making the codebase easier to maintain, we request that the following conventions be observed. Code that is lifted from other projects does not need to be modified to match these rules – no need to fix what isn't broken.
This convention page follows the requirements keywords of RFC 2119. Requirements are seen as CAPITAL letter words.
The majority of LMMS conventions follows the Qt Coding Style. Some conventions are repeated here, and if not noted here MAY fall under the Qt style.
Header guards MUST NOT begin with an underscore _
. Identifiers that begin with an underscore + CAPITAL letter are reserved identifiers in C++. If you edit an older file which contains an improper header guard, please fix it to comply with guidelines.
The first #include
in a cpp
file MUST be its own related header file. Afterwards group #include
s by type, where groups are separated by a newline, and ordered by name.
// MySourceFile.cpp
#include "MySourceFile.h"
#include <QMap>
#include <QString>
#include "DataFile.h"
#include "Engine.h"
#include "GuiApplication.h"
Types MUST begin with an uppercase letter.
class ResourcesDB;
enum MyEnum
{
...
};
typedef QList<AutomatableModel *> AutoModelList;
Variables MUST use camel case format, for example: nextValue
.
- Non-static member variables MUST be prefixed with
m_
- Static variables MUST be prefixed with
s_
The remaining variable name MUST follow the Variable Names rule.
float m_currentValue;
static int s_quantization;
Some older code prefixed function parameters with an underscore _
. Function parameters SHOULD NOT be prefixed with an underscore.
Every line of text SHOULD be at most 120 characters long.
Tabs MUST be used for indentation. Instructions for configuring QtCreator can be found here.
Flow control statements (if
, else if
, else
, for
, do
, while
, and switch
) MUST have a space between the keyword and the opening parenthesis.
You MUST use the true
/false
keywords instead of non-standard macros or integers.
b = TRUE; // BAD
b = true; // GOOD
b = 1; // BAD
You MUST use the nullptr
keyword instead of macros or integers.
p = NULL; // BAD
p = nullptr; // GOOD
p = 0; // BAD
The ternary operator ? :
SHOULD be used only when it makes the code more readable or streamlined.
- Spaces MUST NOT be added after an opening brace
- Spaces MUST NOT be added before a closing brace
- Flow control statements (
if
,else if
,else
,for
,do
,while
, andswitch
) MUST use explicit blocking - Block brackets SHOULD be on their own line
// Spaces before/after brackets
void doThis( int a ); // BAD
void doThis(int a); // GOOD
if ( m_sample > 0 ) // BAD
if (m_sample > 0) // GOOD
array[ offset ] // BAD
array[offset] // GOOD
// BAD - Example of not explicit blocking
if (m_sample > 0)
--m_sample;
// GOOD - Example of explicit blocking
if (m_sample > 0)
{
--m_sample;
}
// If the block can fit on one line, it is acceptable to format it as below
if (m_sample > 0) { --m_sample; }
Return statements MUST NOT have parenthesis around the returned value.
return (bar); // BAD
return bar; // GOOD
Spaces MUST NOT be added before semicolons.
return this ; // BAD
return this; // GOOD
Spaces MUST be before and after infix operators (=
, +
, -
, *
, /
, etc.).
sub_note_key_base=base_note_key+octave_cnt*NOTES_PER_OCTAVE; // BAD
sub_note_key_base = base_note_key + octave_cnt * NOTES_PER_OCTAVE; // GOOD
If long ternary expressions don't fit on one line, they MUST be formatted as one of the below, depending on the readability of the surrounding code. If the expressions are very long or convoluted, you SHOULD use if/else blocks instead.
a == condition
? value
: otherValue;
// OR, depending on surrounding code readability
a == condition
? value
: otherValue;
Doxygen comments SHOULD use //! ...
for inline comments about the following line, or //!< ...
for inline comments for the line they are in, or
/**
...
*/
for larger comments. Example:
/**
this
is a
larger
comment
about `class s`
*/
class s
{
//! this comment is above `f` and explains what `f` does
void f();
int m_i; //!< this comment is behind `m_i` and explains what `m_i` does
};
For referencing code, you can use backticks, like above, or e.g. @p
to reference a parameter name.
Comments in headers SHOULD use doxygen style whenever possible. For example, if you see code like
// This function does ...
void function();
then change it to
//! This function does ...
void function();
When submitting a PR, normal comments should be turned into doxygen comments. However, this must be done with care: In the following examples, blindly replacing //
by //!
would lead to bad doxygen comments:
// the next two functions do ...
void f();
void g();
// This class could need some cleanup
class c { /* ... */ };
Sometimes it's worth merging code even if there are improvements to be made. In these cases its helpful to leave comments highlighting what improvements should be made and where. For the sake of discoverability, please use a searchable keyword in these comments. Some keywords currently used in the codebase are TODO
, HACK
, Workaround
, and FIXME
. An example from PR #5427: // TODO: We should do this work outside the event thread
.
If a piece of code is only relevant for some versions of Qt, QT_VERSION
and QT_VERSION_CHECK(MAJ, MIN, PAT)
can be used to exclude it in other versions. This has the added benefit that when we update our minimum Qt version, a simple search for QT_VERSION
can be used to remove outdated code. A simplified example based on PR #5777:
#if QT_VERSION < QT_VERSION_CHECK(5, 12, 2)
// Your code here
#endif
Changes to the C++ standard can simplify or otherwise improve existing code. Since we lag behind the standards, sometimes we can know ahead of time that a particular section of code could be better written if a newer standard were used. In this case, a comment along the lines of // TODO C++##:
should be added. An example from PR #5589: // TODO C++17 and above: use std::optional instead