Some general rules to write code: Try to follow the same style/format of the file that you are editing (naming, indentation, etc.) or the style of the module. Some submodules, created by us, or by third-parties, have their own style.
There is a .clang-format file available for Aseprite and laf, and we are using it with Clang 19. You have to configure a pre-commit hook which will help you to do the formatting automatically before committing.
There is a .clang-tidy file used in the GitHub actions executed on each PR. These rules are adopted progressively on patches because are only executed in the diff, and if some rule is violated a comment by aseprite-bot is made. (Sometimes the bot will be wrong, so be careful.)
We use a column limit of 100. clang-format will break lines to avoid excessing more than 100 lines, but in some extreme cases it might not break this limit, as our PenaltyExcessCharacter is not the highest value.
Basic statements:
void function_with_long_args(const int argument1,
const int argument2,
const std::string& argument3,
const double argument4,
...)
{
}
void function_with_short_args(int arg1, const int arg2, const int arg3, ...)
{
const int constValue = 0;
int value;
// If a condition will return, we prefer the "return"
// statement in its own line to avoid missing the "return"
// keyword when we read code.
if (condition)
return;
// You can use braces {} if the condition has multiple lines
// or the if-body has multiple lines.
if (condition1 || condition2) {
...
return;
}
if (condition) {
...
}
for (int i = 0; i < 10; ++i) {
// Same case as in if-return.
if (condition)
break;
...
}
while (condition) {
...
}
do {
...
} while (condition);
switch (condition) {
case 1: ... break;
case 2: {
int varInsideCase;
// ...
break;
}
default: break;
}
}
Define namespaces with lower case:
namespace app {
...
} // namespace app
Define classes with CapitalCase
and member functions with camelCase
:
class ClassName {
public:
ClassName() : m_memberVarA(1), m_memberVarB(2), m_memberVarC(3) {}
ClassName(int a, int b, int c, int d)
: m_memberVarA(a)
, m_memberVarB(b)
, m_memberVarC(c)
, m_memberVarD(d)
{
// ...
}
virtual ~ClassName();
// We can return in the same line for getter-like functions
int memberVar() const { return m_memberVar; }
void setMemberVar();
protected:
virtual void onEvent1() {} // Do nothing functions can be defined as "{}"
virtual void onEvent2() = 0;
private:
int m_memberVarA;
int m_memberVarB;
int m_memberVarC;
int m_memberVarD = 4; // We can initialize variables here too
};
class Special : public ClassName {
public:
Special();
protected:
void onEvent2() override
{
// No need to repeat virtual in overridden methods
...
}
};
We use the const-west notation:
There is a problem with clang-tidy
that will make comments using
East const notation:
#4361, but
clang-format should fix the const
position anyway.
We are using C++17 standard. Some things cannot be used because we're targetting macOS 10.9, some notes are added about this:
- Use
nullptr
instead ofNULL
macro - Use
auto
/auto*
for complex types/pointers, iterators, or when the variable type is obvious (e.g.auto* s = new Sprite;
) - Use range-based for loops (
for (const auto& item : values) { ... }
) - Use template alias (
template<typename T> alias = orig<T>;
) - Use generic lambda functions
- Use
std::shared_ptr
,std::unique_ptr
, orbase::Ref
, but generally we'd prefer value semantics instead of smart pointers - Use
std::min
/std::max
/std::clamp
- Use
std::optional
but taking care of some limitations from macOS 10.9: - Use
std::variant
but taking care of some limitations from macOS 10.9: - Use
std::any
but taking care of some limitations from macOS 10.9:- Use
T* p = std::any_cast<T>(&value)
instead ofT v = std::any_cast<T>(value)
(example)
- Use
- Use
static constexpr T v = ...;
- You can use
<atomic>
,<thread>
,<mutex>
, and<condition_variable>
- Prefer
using T = ...;
instead oftypedef ... T
- Use
[[fallthrough]]
if needed - Use
= {}
only to specify a default argument value of an user-defined type in a function declaration, e.g.void func(const std::string& s = {}) { ... }
. In other cases (e.g. a member variable of an user-defined type) it's not required or we prefer to use the explicit value for built-in types (int m_var = 0;
). - We use gcc 9.2 or clang 9.0 on Linux, so check the features available in https://en.cppreference.com/w/cpp/compiler_support