Avito ProTools team Swift style guide
v 0.6.2
Use descriptive names with camel case for classes, methods, variables, etc. Type names (classes, structures, enumerations and protocols) should be capitalized, while method names and variables should start with a lower case letter. Avoid using "_" in names.
For functions and init methods, prefer named parameters for all arguments unless the context is very clear. Include external parameter names if it makes function calls more readable.
func date(fromString dateString: String) -> Date
func convertPointAt(column column: Int, row: Int) -> CGPoint
func timedAction(afterDelay delay: TimeInterval, perform action: Action) -> SKAction
For methods, follow the standard Apple convention of referring to the first parameter in the method name:
class Counter {
// MARK: - Methods -
func combine(with otherCounter: Counter, options: [String: Any]?) {
...
}
func increment(by amount: Int) {
...
}
}
Following Apple's API Design Guidelines, protocols names that describe what something is should be a noun. Examples: Collection
, WidgetFactory
. Protocols names that describe an ability should end in -ing, -able, or -ible. Examples: Equatable
, Resizing
.
Following Apple's API Design Guidelines for Swift 3, use lowerCamelCase for enumeration values.
enum Shape {
case rectangle
case square
case rightTriangle
case equilateralTriangle
}
Indent using 4 spaces
Method braces and other braces (if/else/switch/while etc.) always open on the same line as the statement but close on a new line.
if user.isReady {
...
} else {
...
}
There should be exactly one blank line between methods to aid in visual clarity and organization. Whitespace within methods should separate functionality, but having too many sections in a method often means you should refactor into several methods.
Blank line should be placed after braces if there are more than one code line:
if user.isReady {
user.run()
}
if user.isReady {
user.prepare()
user.run()
}
final class MyClass {
private(set) var property1: String?
private(set) var property2: String?
}
Same for methods, properties and initializers:
func doWork() {
user.doWork()
}
func doWork() {
user.prepare()
user.doWork()
}
Exceptions:
Single guard:
func doWork() {
guard let user = user else { return }
user.prepare()
user.doWork()
}
Calling super:
init(user: User) {
super.init()
self.user = user
}
override func viewDidLoad() {
super.viewDidLoad()
...
}
Colons always have no space on the left and one space on the right. Exceptions are the ternary operator ? :, empty dictionary [:] and #selector syntax for unnamed parameters (_:).
final class MyClass: NSObject {
var property1: [String: Any] = [:]
var property2: [String: Any] = ["A": 1.2, "B": 3.2]
}
For conciseness, avoid using self since Swift does not require it to access an object's properties or invoke its methods
For conciseness, if a computed property is read-only, omit the get clause. The get clause is required only when a set clause is provided.
var diameter: Double {
return radius * 2
}
var diameter: Double {
get {
radius * 2
}
set {
radius = newValue / 2
}
}
But not:
var diameter: Double {
get { radius * 2 }
set { radius = newValue / 2 }
}
Mark classes final when inheritance is not intended. Example:
final class MyClass {
....
}
Keep short function declarations on one line including the opening brace:
func update(with item: Item) -> Bool {
...
}
For functions with long signatures, add line breaks like:
func update(with item: Item,
info userInfo: [String: Any]? = nil,
options: [String: Any]? = nil,
user: User) -> Bool
{
...
}
Always use Swift's native types when available.
Constants are defined using the let keyword, and variables with the var keyword. Always use let instead of var if the value of the variable will not change.
Prefer compact code and let the compiler infer the type for local variables, but not for class/struct properties. In case of class/struct properties implicit type declaration only allowed for boolean, interger, string (not empty) and double constants plus direct constructor calls:
final class MyClass {
// MARK: - Properties -
var property1 = false
var property2: [String: Any] = SomeClass.calculateProperty2()
var property3: = "SomeString"
var property4: [String] = []
var property5 = 0
var property6 = SomeClass()
// MARK: - Mеthods -
func method1() {
var p1 = false
var p2: [String: Any] = SomeClass.calculateProperty2()
var p3 = ""
var p4 = [String]()
var p5 = 0
}
}
But not:
final class MyClass {
// MARK: - Properties -
var property1 = false
var property2 = SomeClass.calculateProperty2()
var property3 = ""
}
Marks syntax:
// MARK: - <Name> -
Required:
Mark protocols conformance methods:
final class MyClass: Cancelable {
// MARK: - Cancelable -
func cancel() {
...
}
}
Preferred:
Mark all class methods:
final class MyViewController: UIViewController {
// MARK: - Properties -
var model: Model? = nil
// MARK: - UIViewController -
override func viewDidLoad() {
super.viewDidLoad()
...
}
override func viewWillLayoutSubviews() {
super.viewWillLayoutSubviews()
...
}
// MARK: - Navigation -
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
...
}
}
final class MyClass {
// MARK: - Properties -
var a: A?
var b: B?
// MARK: - Init -
init(a: A?, b: B?) {
self.a = a
self.b = b
}
convenience init() {
self.init(a: nil, b: nil)
}
}
Use guard where possible:
func someFunc(a: A?) -> B? {
guard let a = a else { return nil }
...
}
for item in items {
guard let name = item.name else { continue }
...
}
Not:
for item in Items {
if let name = item.name {
...
}
}
Place return statement on same line if it's short like:
return value
Prefer split guard if multiple variables declared:
func run(a: A?, b: B?) -> C {
guard let a = a else { return nil }
guard let b = b where b.value > 0 else { return nil }
...
}
Use structs only for short data structures with no reference properties. Like:
struct Point {
var x: Int = 0
var y: Int = 0
}
final class NamedPoint {
var x: Int = 0
var y: Int = 0
var name: String?
}
In particular, when adding protocol conformance to a model, prefer adding it in class declaration rather than to a separate extension:
final class SomeViewController: UIViewController, UITableViewDataSource {
// MARK: - UITableViewDataSource -
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
...
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
...
}
}
Avoid using extensions in same file, use // MARK: - - instead.
Use functional programming where it fits well.
Use forEach when you need to iterate through all elements of collection:
collection.forEach {
$0.doSomething()
}
Use map or flatMap to transform one collection to another:
let b = a.map { ... }
or multiline:
let b = a.map {
...
...
}
Consider using map or flatMap to unwrap optionals:
let b = a.map { B($0) }
var a: SomeType?
a.map { doSomething(a: $0) }
a.map {
let c = SomeClass(a: a)
c.doSomething()
}
Aggregate protocols through typealias:
typealias CollectionViewProtocol = UICollectionViewDataSource & UICollectionViewDelegate
But not:
protocol CollectionViewProtocol: UICollectionViewDataSource, UICollectionViewDelegate {
}