blob: e421082796c5061ccd9865b5036465432c751c53 [file] [log] [blame]
//===- unittest/Format/FormatTestObjC.cpp - Formatting unit tests----------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "clang/Format/Format.h"
#include "../Tooling/ReplacementTest.h"
#include "FormatTestUtils.h"
#include "llvm/Support/Debug.h"
#include "llvm/Support/MemoryBuffer.h"
#include "gtest/gtest.h"
#define DEBUG_TYPE "format-test"
using testing::ScopedTrace;
namespace clang {
namespace format {
namespace {
class FormatTestObjC : public ::testing::Test {
protected:
FormatTestObjC() {
Style = getLLVMStyle();
Style.Language = FormatStyle::LK_ObjC;
}
enum StatusCheck { SC_ExpectComplete, SC_ExpectIncomplete, SC_DoNotCheck };
std::string format(llvm::StringRef Code,
StatusCheck CheckComplete = SC_ExpectComplete) {
LLVM_DEBUG(llvm::errs() << "---\n");
LLVM_DEBUG(llvm::errs() << Code << "\n\n");
std::vector<tooling::Range> Ranges(1, tooling::Range(0, Code.size()));
FormattingAttemptStatus Status;
tooling::Replacements Replaces =
reformat(Style, Code, Ranges, "<stdin>", &Status);
if (CheckComplete != SC_DoNotCheck) {
bool ExpectedCompleteFormat = CheckComplete == SC_ExpectComplete;
EXPECT_EQ(ExpectedCompleteFormat, Status.FormatComplete)
<< Code << "\n\n";
}
auto Result = applyAllReplacements(Code, Replaces);
EXPECT_TRUE(static_cast<bool>(Result));
LLVM_DEBUG(llvm::errs() << "\n" << *Result << "\n\n");
return *Result;
}
void _verifyFormat(const char *File, int Line, StringRef Code) {
ScopedTrace t(File, Line, ::testing::Message() << Code.str());
EXPECT_EQ(Code.str(), format(Code)) << "Expected code is not stable";
EXPECT_EQ(Code.str(), format(test::messUp(Code)));
}
void _verifyIncompleteFormat(const char *File, int Line, StringRef Code) {
ScopedTrace t(File, Line, ::testing::Message() << Code.str());
EXPECT_EQ(Code.str(), format(test::messUp(Code), SC_ExpectIncomplete));
}
FormatStyle Style;
};
#define verifyIncompleteFormat(...) \
_verifyIncompleteFormat(__FILE__, __LINE__, __VA_ARGS__)
#define verifyFormat(...) _verifyFormat(__FILE__, __LINE__, __VA_ARGS__)
TEST(FormatTestObjCStyle, DetectsObjCInHeaders) {
auto Style = getStyle("LLVM", "a.h", "none",
"@interface\n"
"- (id)init;");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("LLVM", "a.h", "none",
"@interface\n"
"+ (id)init;");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("LLVM", "a.h", "none",
"@interface\n"
"@end\n"
"//comment");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("LLVM", "a.h", "none",
"@interface\n"
"@end //comment");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
// No recognizable ObjC.
Style = getStyle("LLVM", "a.h", "none", "void f() {}");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_Cpp, Style->Language);
Style = getStyle("{}", "a.h", "none", "@interface Foo\n@end\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("{}", "a.h", "none",
"const int interface = 1;\nconst int end = 2;\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_Cpp, Style->Language);
Style = getStyle("{}", "a.h", "none", "@protocol Foo\n@end\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("{}", "a.h", "none",
"const int protocol = 1;\nconst int end = 2;\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_Cpp, Style->Language);
Style = getStyle("{}", "a.h", "none", "typedef NS_ENUM(int, Foo) {};\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style =
getStyle("{}", "a.h", "none", "typedef NS_CLOSED_ENUM(int, Foo) {};\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("{}", "a.h", "none", "enum Foo {};");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_Cpp, Style->Language);
Style =
getStyle("{}", "a.h", "none", "inline void Foo() { Log(@\"Foo\"); }\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style =
getStyle("{}", "a.h", "none", "inline void Foo() { Log(\"Foo\"); }\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_Cpp, Style->Language);
Style =
getStyle("{}", "a.h", "none", "inline void Foo() { id = @[1, 2, 3]; }\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("{}", "a.h", "none",
"inline void Foo() { id foo = @{1: 2, 3: 4, 5: 6}; }\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("{}", "a.h", "none",
"inline void Foo() { int foo[] = {1, 2, 3}; }\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_Cpp, Style->Language);
// ObjC characteristic types.
Style = getStyle("{}", "a.h", "none", "extern NSString *kFoo;\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("{}", "a.h", "none", "extern NSInteger Foo();\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("{}", "a.h", "none", "NSObject *Foo();\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
Style = getStyle("{}", "a.h", "none", "NSSet *Foo();\n");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_ObjC, Style->Language);
}
TEST(FormatTestObjCStyle, AvoidDetectingDesignatedInitializersAsObjCInHeaders) {
auto Style = getStyle("LLVM", "a.h", "none",
"static const char *names[] = {[0] = \"foo\",\n"
"[kBar] = \"bar\"};");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_Cpp, Style->Language);
Style = getStyle("LLVM", "a.h", "none",
"static const char *names[] = {[0] EQ \"foo\",\n"
"[kBar] EQ \"bar\"};");
ASSERT_TRUE((bool)Style);
EXPECT_EQ(FormatStyle::LK_Cpp, Style->Language);
}
TEST_F(FormatTestObjC, FormatObjCTryCatch) {
verifyFormat("@try {\n"
" f();\n"
"} @catch (NSException e) {\n"
" @throw;\n"
"} @finally {\n"
" exit(42);\n"
"}");
verifyFormat("DEBUG({\n"
" @try {\n"
" } @finally {\n"
" }\n"
"});\n");
}
TEST_F(FormatTestObjC, FormatObjCAutoreleasepool) {
verifyFormat("@autoreleasepool {\n"
" f();\n"
"}\n"
"@autoreleasepool {\n"
" f();\n"
"}\n");
Style.BreakBeforeBraces = FormatStyle::BS_Custom;
Style.BraceWrapping.AfterControlStatement = FormatStyle::BWACS_Always;
verifyFormat("@autoreleasepool\n"
"{\n"
" f();\n"
"}\n"
"@autoreleasepool\n"
"{\n"
" f();\n"
"}\n");
}
TEST_F(FormatTestObjC, FormatObjCGenerics) {
Style.ColumnLimit = 40;
verifyFormat("int aaaaaaaaaaaaaaaa(\n"
" NSArray<aaaaaaaaaaaaaaaaaa *>\n"
" aaaaaaaaaaaaaaaaa);\n");
verifyFormat("int aaaaaaaaaaaaaaaa(\n"
" NSArray<aaaaaaaaaaaaaaaaaaa<\n"
" aaaaaaaaaaaaaaaa *> *>\n"
" aaaaaaaaaaaaaaaaa);\n");
}
TEST_F(FormatTestObjC, FormatObjCSynchronized) {
verifyFormat("@synchronized(self) {\n"
" f();\n"
"}\n"
"@synchronized(self) {\n"
" f();\n"
"}\n");
Style.BreakBeforeBraces = FormatStyle::BS_Custom;
Style.BraceWrapping.AfterControlStatement = FormatStyle::BWACS_Always;
verifyFormat("@synchronized(self)\n"
"{\n"
" f();\n"
"}\n"
"@synchronized(self)\n"
"{\n"
" f();\n"
"}\n");
}
TEST_F(FormatTestObjC, FormatObjCInterface) {
verifyFormat("@interface Foo : NSObject <NSSomeDelegate> {\n"
"@public\n"
" int field1;\n"
"@protected\n"
" int field2;\n"
"@private\n"
" int field3;\n"
"@package\n"
" int field4;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface /* wait for it */ Foo\n"
"+ (id)init;\n"
"// Look, a comment!\n"
"- (int)answerWith:(int)i;\n"
"@end");
verifyFormat("@interface Foo\n"
"@end\n"
"@interface Bar\n"
"@end");
verifyFormat("@interface Foo : Bar\n"
"@property(assign, readwrite) NSInteger bar;\n"
"+ (id)init;\n"
"@end");
verifyFormat("FOUNDATION_EXPORT NS_AVAILABLE_IOS(10.0) @interface Foo : Bar\n"
"@property(assign, readwrite) NSInteger bar;\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo : /**/ Bar /**/ <Baz, /**/ Quux>\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo (HackStuff)\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo ()\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo (HackStuff) <MyProtocol>\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo : Bar {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo : Bar <Baz, Quux> {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo<Baz : Blech> : Bar <Baz, Quux> {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo<Bar : Baz <Blech>> : Xyzzy <Corge> {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo<Bar : Baz <Blech>> : Xyzzy <Corge> <Quux> {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo : Bar <Baz> <Blech>\n"
"@end");
verifyFormat("@interface Foo : Bar <Baz> <Blech, Xyzzy, Corge>\n"
"@end");
verifyFormat("@interface Foo (HackStuff) {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo () {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo (HackStuff) <MyProtocol> {\n"
" int _i;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo\n"
"- (void)foo {\n"
"}\n"
"@end\n"
"@implementation Bar\n"
"- (void)bar {\n"
"}\n"
"@end");
Style.ColumnLimit = 40;
verifyFormat("@interface ccccccccccccc () <\n"
" ccccccccccccc, ccccccccccccc,\n"
" ccccccccccccc, ccccccccccccc> {\n"
"}");
verifyFormat("@interface ccccccccccccc (ccccccccccc) <\n"
" ccccccccccccc> {\n"
"}");
Style.ObjCBinPackProtocolList = FormatStyle::BPS_Never;
verifyFormat("@interface ddddddddddddd () <\n"
" ddddddddddddd,\n"
" ddddddddddddd,\n"
" ddddddddddddd,\n"
" ddddddddddddd> {\n"
"}");
Style.BinPackParameters = false;
Style.ObjCBinPackProtocolList = FormatStyle::BPS_Auto;
verifyFormat("@interface eeeeeeeeeeeee () <\n"
" eeeeeeeeeeeee,\n"
" eeeeeeeeeeeee,\n"
" eeeeeeeeeeeee,\n"
" eeeeeeeeeeeee> {\n"
"}");
Style.ObjCBinPackProtocolList = FormatStyle::BPS_Always;
verifyFormat("@interface fffffffffffff () <\n"
" fffffffffffff, fffffffffffff,\n"
" fffffffffffff, fffffffffffff> {\n"
"}");
Style = getGoogleStyle(FormatStyle::LK_ObjC);
verifyFormat("@interface Foo : NSObject <NSSomeDelegate> {\n"
" @public\n"
" int field1;\n"
" @protected\n"
" int field2;\n"
" @private\n"
" int field3;\n"
" @package\n"
" int field4;\n"
"}\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo : Bar <Baz, Quux>\n"
"+ (id)init;\n"
"@end");
verifyFormat("@interface Foo (HackStuff) <MyProtocol>\n"
"+ (id)init;\n"
"@end");
Style.ColumnLimit = 40;
// BinPackParameters should be true by default.
verifyFormat("void eeeeeeee(int eeeee, int eeeee,\n"
" int eeeee, int eeeee);\n");
// ObjCBinPackProtocolList should be BPS_Never by default.
verifyFormat("@interface fffffffffffff () <\n"
" fffffffffffff,\n"
" fffffffffffff,\n"
" fffffffffffff,\n"
" fffffffffffff> {\n"
"}");
verifyFormat("@interface ggggggggggggg\n"
" : ggggggggggggg <ggggggggggggg>\n"
" <ggggggggggggg>\n"
"@end");
}
TEST_F(FormatTestObjC, FormatObjCImplementation) {
verifyFormat("@implementation Foo : NSObject {\n"
"@public\n"
" int field1;\n"
"@protected\n"
" int field2;\n"
"@private\n"
" int field3;\n"
"@package\n"
" int field4;\n"
"}\n"
"+ (id)init {\n}\n"
"@end");
verifyFormat("@implementation Foo\n"
"+ (id)init {\n"
" if (true)\n"
" return nil;\n"
"}\n"
"// Look, a comment!\n"
"- (int)answerWith:(int)i {\n"
" return i;\n"
"}\n"
"+ (int)answerWith:(int)i {\n"
" return i;\n"
"}\n"
"@end");
verifyFormat("@implementation Foo\n"
"@end\n"
"@implementation Bar\n"
"@end");
EXPECT_EQ("@implementation Foo : Bar\n"
"+ (id)init {\n}\n"
"- (void)foo {\n}\n"
"@end",
format("@implementation Foo : Bar\n"
"+(id)init{}\n"
"-(void)foo{}\n"
"@end"));
verifyFormat("@implementation Foo {\n"
" int _i;\n"
"}\n"
"+ (id)init {\n}\n"
"@end");
verifyFormat("@implementation Foo : Bar {\n"
" int _i;\n"
"}\n"
"+ (id)init {\n}\n"
"@end");
verifyFormat("@implementation Foo (HackStuff)\n"
"+ (id)init {\n}\n"
"@end");
verifyFormat("@implementation ObjcClass\n"
"- (void)method;\n"
"{}\n"
"@end");
Style = getGoogleStyle(FormatStyle::LK_ObjC);
verifyFormat("@implementation Foo : NSObject {\n"
" @public\n"
" int field1;\n"
" @protected\n"
" int field2;\n"
" @private\n"
" int field3;\n"
" @package\n"
" int field4;\n"
"}\n"
"+ (id)init {\n}\n"
"@end");
}
TEST_F(FormatTestObjC, FormatObjCProtocol) {
verifyFormat("@protocol Foo\n"
"@property(weak) id delegate;\n"
"- (NSUInteger)numberOfThings;\n"
"@end");
verifyFormat("@protocol MyProtocol <NSObject>\n"
"- (NSUInteger)numberOfThings;\n"
"@end");
verifyFormat("@protocol Foo;\n"
"@protocol Bar;\n");
verifyFormat("@protocol Foo\n"
"@end\n"
"@protocol Bar\n"
"@end");
verifyFormat("FOUNDATION_EXPORT NS_AVAILABLE_IOS(10.0) @protocol Foo\n"
"@property(assign, readwrite) NSInteger bar;\n"
"@end");
verifyFormat("@protocol myProtocol\n"
"- (void)mandatoryWithInt:(int)i;\n"
"@optional\n"
"- (void)optional;\n"
"@required\n"
"- (void)required;\n"
"@optional\n"
"@property(assign) int madProp;\n"
"@end\n");
verifyFormat("@property(nonatomic, assign, readonly)\n"
" int *looooooooooooooooooooooooooooongNumber;\n"
"@property(nonatomic, assign, readonly)\n"
" NSString *looooooooooooooooooooooooooooongName;");
verifyFormat("@implementation PR18406\n"
"}\n"
"@end");
Style = getGoogleStyle(FormatStyle::LK_ObjC);
verifyFormat("@protocol MyProtocol <NSObject>\n"
"- (NSUInteger)numberOfThings;\n"
"@end");
}
TEST_F(FormatTestObjC, FormatObjCMethodDeclarations) {
verifyFormat("- (void)doSomethingWith:(GTMFoo *)theFoo\n"
" rect:(NSRect)theRect\n"
" interval:(float)theInterval {\n"
"}");
verifyFormat("- (void)shortf:(GTMFoo *)theFoo\n"
" longKeyword:(NSRect)theRect\n"
" longerKeyword:(float)theInterval\n"
" error:(NSError **)theError {\n"
"}");
verifyFormat("- (void)shortf:(GTMFoo *)theFoo\n"
" longKeyword:(NSRect)theRect\n"
" evenLongerKeyword:(float)theInterval\n"
" error:(NSError **)theError {\n"
"}");
verifyFormat("+ (instancetype)new;\n");
Style.ColumnLimit = 60;
verifyFormat("- (instancetype)initXxxxxx:(id<x>)x\n"
" y:(id<yyyyyyyyyyyyyyyyyyyy>)y\n"
" NS_DESIGNATED_INITIALIZER;");
verifyFormat("- (void)drawRectOn:(id)surface\n"
" ofSize:(size_t)height\n"
" :(size_t)width;");
Style.ColumnLimit = 40;
// Make sure selectors with 0, 1, or more arguments are indented when wrapped.
verifyFormat("- (aaaaaaaaaaaaaaaaaaaaaaaaaaaaa)\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaaa;\n");
verifyFormat("- (aaaaaaaaaaaaaaaaaaaaaaaaaaaaa)\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaaa:(int)a;\n");
verifyFormat("- (aaaaaaaaaaaaaaaaaaaaaaaaaaaaa)\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaaa:(int)a\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaaa:(int)a;\n");
verifyFormat("- (aaaaaaaaaaaaaaaaaaaaaaaaaaaaa)\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaa:(int)a\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaaa:(int)a;\n");
verifyFormat("- (aaaaaaaaaaaaaaaaaaaaaaaaaaaaa)\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaaa:(int)a\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaa:(int)a;\n");
// Continuation indent width should win over aligning colons if the function
// name is long.
Style = getGoogleStyle(FormatStyle::LK_ObjC);
Style.ColumnLimit = 40;
verifyFormat("- (void)shortf:(GTMFoo *)theFoo\n"
" dontAlignNamef:(NSRect)theRect {\n"
"}");
// Make sure we don't break aligning for short parameter names.
verifyFormat("- (void)shortf:(GTMFoo *)theFoo\n"
" aShortf:(NSRect)theRect {\n"
"}");
// Format pairs correctly.
Style.ColumnLimit = 80;
verifyFormat("- (void)drawRectOn:(id)surface\n"
" ofSize:(aaaaaaaa)height\n"
" :(size_t)width\n"
" atOrigin:(size_t)x\n"
" :(size_t)y\n"
" aaaaa:(a)yyy\n"
" bbb:(d)cccc;");
verifyFormat("- (void)drawRectOn:(id)surface ofSize:(aaa)height:(bbb)width;");
// BraceWrapping AfterFunction is respected for ObjC methods
Style = getGoogleStyle(FormatStyle::LK_ObjC);
Style.BreakBeforeBraces = FormatStyle::BS_Custom;
Style.BraceWrapping.AfterFunction = true;
verifyFormat("@implementation Foo\n"
"- (void)foo:(id)bar\n"
"{\n"
"}\n"
"@end\n");
}
TEST_F(FormatTestObjC, FormatObjCMethodExpr) {
verifyFormat("[foo bar:baz];");
verifyFormat("[foo bar]->baz;");
verifyFormat("return [foo bar:baz];");
verifyFormat("return (a)[foo bar:baz];");
verifyFormat("f([foo bar:baz]);");
verifyFormat("f(2, [foo bar:baz]);");
verifyFormat("f(2, a ? b : c);");
verifyFormat("[[self initWithInt:4] bar:[baz quux:arrrr]];");
// Unary operators.
verifyFormat("int a = +[foo bar:baz];");
verifyFormat("int a = -[foo bar:baz];");
verifyFormat("int a = ![foo bar:baz];");
verifyFormat("int a = ~[foo bar:baz];");
verifyFormat("int a = ++[foo bar:baz];");
verifyFormat("int a = --[foo bar:baz];");
verifyFormat("int a = sizeof [foo bar:baz];");
verifyFormat("int a = alignof [foo bar:baz];");
verifyFormat("int a = &[foo bar:baz];");
verifyFormat("int a = *[foo bar:baz];");
// FIXME: Make casts work, without breaking f()[4].
// verifyFormat("int a = (int)[foo bar:baz];");
// verifyFormat("return (int)[foo bar:baz];");
// verifyFormat("(void)[foo bar:baz];");
verifyFormat("return (MyType *)[self.tableView cellForRowAtIndexPath:cell];");
// Binary operators.
verifyFormat("[foo bar:baz], [foo bar:baz];");
verifyFormat("[foo bar:baz] = [foo bar:baz];");
verifyFormat("[foo bar:baz] *= [foo bar:baz];");
verifyFormat("[foo bar:baz] /= [foo bar:baz];");
verifyFormat("[foo bar:baz] %= [foo bar:baz];");
verifyFormat("[foo bar:baz] += [foo bar:baz];");
verifyFormat("[foo bar:baz] -= [foo bar:baz];");
verifyFormat("[foo bar:baz] <<= [foo bar:baz];");
verifyFormat("[foo bar:baz] >>= [foo bar:baz];");
verifyFormat("[foo bar:baz] &= [foo bar:baz];");
verifyFormat("[foo bar:baz] ^= [foo bar:baz];");
verifyFormat("[foo bar:baz] |= [foo bar:baz];");
verifyFormat("[foo bar:baz] ? [foo bar:baz] : [foo bar:baz];");
verifyFormat("[foo bar:baz] || [foo bar:baz];");
verifyFormat("[foo bar:baz] && [foo bar:baz];");
verifyFormat("[foo bar:baz] | [foo bar:baz];");
verifyFormat("[foo bar:baz] ^ [foo bar:baz];");
verifyFormat("[foo bar:baz] & [foo bar:baz];");
verifyFormat("[foo bar:baz] == [foo bar:baz];");
verifyFormat("[foo bar:baz] != [foo bar:baz];");
verifyFormat("[foo bar:baz] >= [foo bar:baz];");
verifyFormat("[foo bar:baz] <= [foo bar:baz];");
verifyFormat("[foo bar:baz] > [foo bar:baz];");
verifyFormat("[foo bar:baz] < [foo bar:baz];");
verifyFormat("[foo bar:baz] >> [foo bar:baz];");
verifyFormat("[foo bar:baz] << [foo bar:baz];");
verifyFormat("[foo bar:baz] - [foo bar:baz];");
verifyFormat("[foo bar:baz] + [foo bar:baz];");
verifyFormat("[foo bar:baz] * [foo bar:baz];");
verifyFormat("[foo bar:baz] / [foo bar:baz];");
verifyFormat("[foo bar:baz] % [foo bar:baz];");
// Whew!
verifyFormat("return in[42];");
verifyFormat("for (auto v : in[1]) {\n}");
verifyFormat("for (int i = 0; i < in[a]; ++i) {\n}");
verifyFormat("for (int i = 0; in[a] < i; ++i) {\n}");
verifyFormat("for (int i = 0; i < n; ++i, ++in[a]) {\n}");
verifyFormat("for (int i = 0; i < n; ++i, in[a]++) {\n}");
verifyFormat("for (int i = 0; i < f(in[a]); ++i, in[a]++) {\n}");
verifyFormat("for (id foo in [self getStuffFor:bla]) {\n"
"}");
verifyFormat("[self aaaaa:MACRO(a, b:, c:)];");
verifyFormat("[self aaaaa:MACRO(a, b:c:, d:e:)];");
verifyFormat("[self aaaaa:MACRO(a, b:c:d:, e:f:g:)];");
verifyFormat("int XYMyFoo(int a, int b) NS_SWIFT_NAME(foo(self:scale:));");
verifyFormat("[self aaaaa:(1 + 2) bbbbb:3];");
verifyFormat("[self aaaaa:(Type)a bbbbb:3];");
verifyFormat("[self stuffWithInt:(4 + 2) float:4.5];");
verifyFormat("[self stuffWithInt:a ? b : c float:4.5];");
verifyFormat("[self stuffWithInt:a ? [self foo:bar] : c];");
verifyFormat("[self stuffWithInt:a ? (e ? f : g) : c];");
verifyFormat("[cond ? obj1 : obj2 methodWithParam:param]");
verifyFormat("[button setAction:@selector(zoomOut:)];");
verifyFormat("[color getRed:&r green:&g blue:&b alpha:&a];");
verifyFormat("arr[[self indexForFoo:a]];");
verifyFormat("throw [self errorFor:a];");
verifyFormat("@throw [self errorFor:a];");
verifyFormat("[(id)foo bar:(id)baz quux:(id)snorf];");
verifyFormat("[(id)foo bar:(id) ? baz : quux];");
verifyFormat("4 > 4 ? (id)a : (id)baz;");
unsigned PreviousColumnLimit = Style.ColumnLimit;
Style.ColumnLimit = 50;
// Instead of:
// bool a =
// ([object a:42] == 0 || [object a:42
// b:42] == 0);
verifyFormat("bool a = ([object a:42] == 0 ||\n"
" [object a:42 b:42] == 0);");
Style.ColumnLimit = PreviousColumnLimit;
verifyFormat("bool a = ([aaaaaaaa aaaaa] == aaaaaaaaaaaaaaaaa ||\n"
" [aaaaaaaa aaaaa] == aaaaaaaaaaaaaaaaaaaa);");
// This tests that the formatter doesn't break after "backing" but before ":",
// which would be at 80 columns.
verifyFormat(
"void f() {\n"
" if ((self = [super initWithContentRect:contentRect\n"
" styleMask:styleMask ?: otherMask\n"
" backing:NSBackingStoreBuffered\n"
" defer:YES]))");
verifyFormat(
"[foo checkThatBreakingAfterColonWorksOk:\n"
" [bar ifItDoes:reduceOverallLineLengthLikeInThisCase]];");
verifyFormat("[myObj short:arg1 // Force line break\n"
" longKeyword:arg2 != nil ? arg2 : @\"longKeyword\"\n"
" evenLongerKeyword:arg3 ?: @\"evenLongerKeyword\"\n"
" error:arg4];");
verifyFormat(
"void f() {\n"
" popup_window_.reset([[RenderWidgetPopupWindow alloc]\n"
" initWithContentRect:NSMakeRect(origin_global.x, origin_global.y,\n"
" pos.width(), pos.height())\n"
" styleMask:NSBorderlessWindowMask\n"
" backing:NSBackingStoreBuffered\n"
" defer:NO]);\n"
"}");
verifyFormat("[contentsContainer replaceSubview:[subviews objectAtIndex:0]\n"
" with:contentsNativeView];");
verifyFormat(
"[pboard addTypes:[NSArray arrayWithObject:kBookmarkButtonDragType]\n"
" owner:nillllll];");
verifyFormat(
"[pboard setData:[NSData dataWithBytes:&button length:sizeof(button)]\n"
" forType:kBookmarkButtonDragType];");
verifyFormat("[defaultCenter addObserver:self\n"
" selector:@selector(willEnterFullscreen)\n"
" name:kWillEnterFullscreenNotification\n"
" object:nil];");
verifyFormat("[image_rep drawInRect:drawRect\n"
" fromRect:NSZeroRect\n"
" operation:NSCompositeCopy\n"
" fraction:1.0\n"
" respectFlipped:NO\n"
" hints:nil];");
verifyFormat("[aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa];");
verifyFormat("[aaaaaaaaaaaaaaaaaaaa(aaaaaaaaaaaaaaaaaaaaa)\n"
" aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa];");
verifyFormat("[aaaaaaaaaaaaaaaaaaaaaaa.aaaaaaaa[aaaaaaaaaaaaaaaaaaaaa]\n"
" aaaaaaaaaaaaaaaaaaaaaa];");
verifyFormat(
"scoped_nsobject<NSTextField> message(\n"
" // The frame will be fixed up when |-setMessageText:| is called.\n"
" [[NSTextField alloc] initWithFrame:NSMakeRect(0, 0, 0, 0)]);");
verifyFormat("[self aaaaaa:bbbbbbbbbbbbb\n"
" aaaaaaaaaa:bbbbbbbbbbbbbbbbb\n"
" aaaaa:bbbbbbbbbbb + bbbbbbbbbbbb\n"
" aaaa:bbb];");
verifyFormat("[self param:function( //\n"
" parameter)]");
verifyFormat(
"[self aaaaaaaaaa:aaaaaaaaaaaaaaa | aaaaaaaaaaaaaaa | aaaaaaaaaaaaaaa |\n"
" aaaaaaaaaaaaaaa | aaaaaaaaaaaaaaa | aaaaaaaaaaaaaaa |\n"
" aaaaaaaaaaaaaaa | aaaaaaaaaaaaaaa];");
// Variadic parameters.
verifyFormat(
"NSArray *myStrings = [NSArray stringarray:@\"a\", @\"b\", nil];");
verifyFormat(
"[self aaaaaaaaaaaaa:aaaaaaaaaaaaaaa, aaaaaaaaaaaaaaa, aaaaaaaaaaaaaaa,\n"
" aaaaaaaaaaaaaaa, aaaaaaaaaaaaaaa, aaaaaaaaaaaaaaa,\n"
" aaaaaaaaaaaaaaa, aaaaaaaaaaaaaaa];");
verifyFormat("[self // break\n"
" a:a\n"
" aaa:aaa];");
// Formats pair-parameters.
verifyFormat("[I drawRectOn:surface ofSize:aa:bbb atOrigin:cc:dd];");
verifyFormat("[I drawRectOn:surface //\n"
" ofSize:aa:bbb\n"
" atOrigin:cc:dd];");
// Inline block as a first argument.
verifyFormat("[object justBlock:^{\n"
" a = 42;\n"
"}];");
verifyFormat("[object\n"
" justBlock:^{\n"
" a = 42;\n"
" }\n"
" notBlock:42\n"
" a:42];");
verifyFormat("[object\n"
" firstBlock:^{\n"
" a = 42;\n"
" }\n"
" blockWithLongerName:^{\n"
" a = 42;\n"
" }];");
verifyFormat("[object\n"
" blockWithLongerName:^{\n"
" a = 42;\n"
" }\n"
" secondBlock:^{\n"
" a = 42;\n"
" }];");
verifyFormat("[object\n"
" firstBlock:^{\n"
" a = 42;\n"
" }\n"
" notBlock:42\n"
" secondBlock:^{\n"
" a = 42;\n"
" }];");
// Space between cast rparen and selector name component.
verifyFormat("[((Foo *)foo) bar];");
verifyFormat("[((Foo *)foo) bar:1 blech:2];");
Style.ColumnLimit = 20;
verifyFormat("aaaaa = [a aa:aa\n"
" aa:aa];");
verifyFormat("aaaaaa = [aa aa:aa\n"
" aa:aa];");
// Message receiver taking multiple lines.
// Non-corner case.
verifyFormat("[[object block:^{\n"
" return 42;\n"
"}] a:42 b:42];");
// Arguments just fit into one line.
verifyFormat("[[object block:^{\n"
" return 42;\n"
"}] aaaaaaa:42 b:42];");
// Arguments just over a column limit.
verifyFormat("[[object block:^{\n"
" return 42;\n"
"}] aaaaaaa:42\n"
" bb:42];");
// Arguments just fit into one line.
Style.ColumnLimit = 23;
verifyFormat("[[obj a:42\n"
" b:42\n"
" c:42\n"
" d:42] e:42 f:42];");
// Arguments do not fit into one line with a receiver.
Style.ColumnLimit = 20;
verifyFormat("[[obj a:42] a:42\n"
" b:42];");
verifyFormat("[[obj a:42] a:42\n"
" b:42\n"
" c:42];");
verifyFormat("[[obj aaaaaa:42\n"
" b:42]\n"
" cc:42\n"
" d:42];");
// Avoid breaking receiver expression.
Style.ColumnLimit = 30;
verifyFormat("fooooooo =\n"
" [[obj fooo] aaa:42\n"
" aaa:42];");
verifyFormat("[[[obj foo] bar] aa:42\n"
" bb:42\n"
" cc:42];");
// Avoid breaking between unary operators and ObjC method expressions.
Style.ColumnLimit = 45;
verifyFormat("if (a012345678901234567890123 &&\n"
" ![foo bar]) {\n"
"}");
verifyFormat("if (a012345678901234567890123 &&\n"
" +[foo bar]) {\n"
"}");
verifyFormat("if (a012345678901234567890123 &&\n"
" -[foo bar]) {\n"
"}");
Style.ColumnLimit = 70;
verifyFormat(
"void f() {\n"
" popup_wdow_.reset([[RenderWidgetPopupWindow alloc]\n"
" iniithContentRect:NSMakRet(origin_global.x, origin_global.y,\n"
" pos.width(), pos.height())\n"
" syeMask:NSBorderlessWindowMask\n"
" bking:NSBackingStoreBuffered\n"
" der:NO]);\n"
"}");
Style.ColumnLimit = 60;
verifyFormat("[call aaaaaaaa.aaaaaa.aaaaaaaa.aaaaaaaa.aaaaaaaa.aaaaaaaa\n"
" .aaaaaaaa];"); // FIXME: Indentation seems off.
// FIXME: This violates the column limit.
verifyFormat(
"[aaaaaaaaaaaaaaaaaaaaaaaaa\n"
" aaaaaaaaaaaaaaaaa:aaaaaaaa\n"
" aaa:aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa];");
Style = getChromiumStyle(FormatStyle::LK_ObjC);
Style.ColumnLimit = 80;
verifyFormat(
"void f() {\n"
" popup_window_.reset([[RenderWidgetPopupWindow alloc]\n"
" initWithContentRect:NSMakeRect(origin_global.x, origin_global.y,\n"
" pos.width(), pos.height())\n"
" styleMask:NSBorderlessWindowMask\n"
" backing:NSBackingStoreBuffered\n"
" defer:NO]);\n"
"}");
// Respect continuation indent and colon alignment (e.g. when object name is
// short, and first selector is the longest one)
Style = getLLVMStyle();
Style.Language = FormatStyle::LK_ObjC;
Style.ContinuationIndentWidth = 8;
verifyFormat("[self performSelectorOnMainThread:@selector(loadAccessories)\n"
" withObject:nil\n"
" waitUntilDone:false];");
verifyFormat("[self performSelector:@selector(loadAccessories)\n"
" withObjectOnMainThread:nil\n"
" waitUntilDone:false];");
verifyFormat(
"[aaaaaaaaaaaaaaaaaaaaaaaaa\n"
" performSelectorOnMainThread:@selector(loadAccessories)\n"
" withObject:nil\n"
" waitUntilDone:false];");
verifyFormat(
"[self // force wrapping\n"
" performSelectorOnMainThread:@selector(loadAccessories)\n"
" withObject:nil\n"
" waitUntilDone:false];");
// The appropriate indentation is used after a block statement.
Style.ContinuationIndentWidth = 4;
verifyFormat(
"void aaaaaaaaaaaaaaaaaaaaa(int c) {\n"
" if (c) {\n"
" f();\n"
" }\n"
" [dddddddddddddddddddddddddddddddddddddddddddddddddddddddd\n"
" eeeeeeeeeeeeeeeeeeeeeeeeeeeee:^(fffffffffffffff gggggggg) {\n"
" f(SSSSS, c);\n"
" }];\n"
"}");
}
TEST_F(FormatTestObjC, ObjCAt) {
verifyFormat("@autoreleasepool");
verifyFormat("@catch");
verifyFormat("@class");
verifyFormat("@compatibility_alias");
verifyFormat("@defs");
verifyFormat("@dynamic");
verifyFormat("@encode");
verifyFormat("@end");
verifyFormat("@finally");
verifyFormat("@implementation");
verifyFormat("@import");
verifyFormat("@interface");
verifyFormat("@optional");
verifyFormat("@package");
verifyFormat("@private");
verifyFormat("@property");
verifyFormat("@protected");
verifyFormat("@protocol");
verifyFormat("@public");
verifyFormat("@required");
verifyFormat("@selector");
verifyFormat("@synchronized");
verifyFormat("@synthesize");
verifyFormat("@throw");
verifyFormat("@try");
EXPECT_EQ("@interface", format("@ interface"));
// The precise formatting of this doesn't matter, nobody writes code like
// this.
verifyFormat("@ /*foo*/ interface");
}
TEST_F(FormatTestObjC, ObjCBlockTypesAndVariables) {
verifyFormat("void DoStuffWithBlockType(int (^)(char));");
verifyFormat("int (^foo)(char, float);");
verifyFormat("int (^foo[10])(char, float);");
verifyFormat("int (^foo[kNumEntries])(char, float);");
verifyFormat("int (^foo[kNumEntries + 10])(char, float);");
verifyFormat("int (^foo[(kNumEntries + 10)])(char, float);");
}
TEST_F(FormatTestObjC, ObjCSnippets) {
verifyFormat("@autoreleasepool {\n"
" foo();\n"
"}");
verifyFormat("@class Foo, Bar;");
verifyFormat("@compatibility_alias AliasName ExistingClass;");
verifyFormat("@dynamic textColor;");
verifyFormat("char *buf1 = @encode(int *);");
verifyFormat("char *buf1 = @encode(typeof(4 * 5));");
verifyFormat("char *buf1 = @encode(int **);");
verifyFormat("Protocol *proto = @protocol(p1);");
verifyFormat("SEL s = @selector(foo:);");
verifyFormat("@synchronized(self) {\n"
" f();\n"
"}");
verifyFormat("@import foo.bar;\n"
"@import baz;");
verifyFormat("@synthesize dropArrowPosition = dropArrowPosition_;");
verifyFormat("@property(assign, nonatomic) CGFloat hoverAlpha;");
verifyFormat("@property(assign, getter=isEditable) BOOL editable;");
verifyFormat("extern UIWindow *MainWindow(void) "
"NS_SWIFT_NAME(getter:MyHelper.mainWindow());");
verifyFormat("extern UIWindow *MainWindow(void) "
"CF_SWIFT_NAME(getter:MyHelper.mainWindow());");
Style.ColumnLimit = 50;
verifyFormat("@interface Foo\n"
"- (void)doStuffWithFoo:(id)name\n"
" bar:(id)bar\n"
" baz:(id)baz\n"
" NS_SWIFT_NAME(doStuff(withFoo:bar:baz:));\n"
"@end");
Style = getMozillaStyle();
verifyFormat("@property (assign, getter=isEditable) BOOL editable;");
verifyFormat("@property BOOL editable;");
Style = getWebKitStyle();
verifyFormat("@property (assign, getter=isEditable) BOOL editable;");
verifyFormat("@property BOOL editable;");
Style = getGoogleStyle(FormatStyle::LK_ObjC);
verifyFormat("@synthesize dropArrowPosition = dropArrowPosition_;");
verifyFormat("@property(assign, getter=isEditable) BOOL editable;");
}
TEST_F(FormatTestObjC, ObjCForIn) {
verifyFormat("- (void)test {\n"
" for (NSString *n in arrayOfStrings) {\n"
" foo(n);\n"
" }\n"
"}");
verifyFormat("- (void)test {\n"
" for (NSString *n in (__bridge NSArray *)arrayOfStrings) {\n"
" foo(n);\n"
" }\n"
"}");
verifyFormat("for (Foo *x in bar) {\n}");
verifyFormat("for (Foo *x in [bar baz]) {\n}");
verifyFormat("for (Foo *x in [bar baz:blech]) {\n}");
verifyFormat("for (Foo *x in [bar baz:blech, 1, 2, 3, 0]) {\n}");
verifyFormat("for (Foo *x in [bar baz:^{\n"
" [uh oh];\n"
" }]) {\n}");
}
TEST_F(FormatTestObjC, ObjCCxxKeywords) {
verifyFormat("+ (instancetype)new {\n"
" return nil;\n"
"}\n");
verifyFormat("+ (instancetype)myNew {\n"
" return [self new];\n"
"}\n");
verifyFormat("SEL NewSelector(void) { return @selector(new); }\n");
verifyFormat("SEL MacroSelector(void) { return MACRO(new); }\n");
verifyFormat("+ (instancetype)delete {\n"
" return nil;\n"
"}\n");
verifyFormat("+ (instancetype)myDelete {\n"
" return [self delete];\n"
"}\n");
verifyFormat("SEL DeleteSelector(void) { return @selector(delete); }\n");
verifyFormat("SEL MacroSelector(void) { return MACRO(delete); }\n");
verifyFormat("MACRO(new:)\n");
verifyFormat("MACRO(delete:)\n");
verifyFormat("foo = @{MACRO(new:) : MACRO(delete:)}\n");
verifyFormat("@implementation Foo\n"
"// Testing\n"
"- (Class)class {\n"
"}\n"
"- (void)foo {\n"
"}\n"
"@end\n");
verifyFormat("@implementation Foo\n"
"- (Class)class {\n"
"}\n"
"- (void)foo {\n"
"}\n"
"@end");
verifyFormat("@implementation Foo\n"
"+ (Class)class {\n"
"}\n"
"- (void)foo {\n"
"}\n"
"@end");
verifyFormat("@implementation Foo\n"
"- (Class)class:(Class)klass {\n"
"}\n"
"- (void)foo {\n"
"}\n"
"@end");
verifyFormat("@implementation Foo\n"
"+ (Class)class:(Class)klass {\n"
"}\n"
"- (void)foo {\n"
"}\n"
"@end");
verifyFormat("@interface Foo\n"
"// Testing\n"
"- (Class)class;\n"
"- (void)foo;\n"
"@end\n");
verifyFormat("@interface Foo\n"
"- (Class)class;\n"
"- (void)foo;\n"
"@end");
verifyFormat("@interface Foo\n"
"+ (Class)class;\n"
"- (void)foo;\n"
"@end");
verifyFormat("@interface Foo\n"
"- (Class)class:(Class)klass;\n"
"- (void)foo;\n"
"@end");
verifyFormat("@interface Foo\n"
"+ (Class)class:(Class)klass;\n"
"- (void)foo;\n"
"@end");
}
TEST_F(FormatTestObjC, ObjCLiterals) {
verifyFormat("@\"String\"");
verifyFormat("@1");
verifyFormat("@+4.8");
verifyFormat("@-4");
verifyFormat("@1LL");
verifyFormat("@.5");
verifyFormat("@'c'");
verifyFormat("@true");
verifyFormat("NSNumber *smallestInt = @(-INT_MAX - 1);");
verifyFormat("NSNumber *piOverTwo = @(M_PI / 2);");
verifyFormat("NSNumber *favoriteColor = @(Green);");
verifyFormat("NSString *path = @(getenv(\"PATH\"));");
verifyFormat("[dictionary setObject:@(1) forKey:@\"number\"];");
}
TEST_F(FormatTestObjC, ObjCDictLiterals) {
verifyFormat("@{");
verifyFormat("@{}");
verifyFormat("@{@\"one\" : @1}");
verifyFormat("return @{@\"one\" : @1;");
verifyFormat("@{@\"one\" : @1}");
verifyFormat("@{@\"one\" : @{@2 : @1}}");
verifyFormat("@{\n"
" @\"one\" : @{@2 : @1},\n"
"}");
verifyFormat("@{1 > 2 ? @\"one\" : @\"two\" : 1 > 2 ? @1 : @2}");
verifyIncompleteFormat("[self setDict:@{}");
verifyIncompleteFormat("[self setDict:@{@1 : @2}");
verifyFormat("NSLog(@\"%@\", @{@1 : @2, @2 : @3}[@1]);");
verifyFormat(
"NSDictionary *masses = @{@\"H\" : @1.0078, @\"He\" : @4.0026};");
verifyFormat(
"NSDictionary *settings = @{AVEncoderKey : @(AVAudioQualityMax)};");
verifyFormat("NSDictionary *d = @{\n"
" @\"nam\" : NSUserNam(),\n"
" @\"dte\" : [NSDate date],\n"
" @\"processInfo\" : [NSProcessInfo processInfo]\n"
"};");
verifyFormat(
"@{\n"
" NSFontAttributeNameeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee : "
"regularFont,\n"
"};");
verifyFormat(
"@{\n"
" NSFontAttributeNameeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee :\n"
" reeeeeeeeeeeeeeeeeeeeeeeegularFont,\n"
"};");
// We should try to be robust in case someone forgets the "@".
verifyFormat("NSDictionary *d = {\n"
" @\"nam\" : NSUserNam(),\n"
" @\"dte\" : [NSDate date],\n"
" @\"processInfo\" : [NSProcessInfo processInfo]\n"
"};");
verifyFormat("NSMutableDictionary *dictionary =\n"
" [NSMutableDictionary dictionaryWithDictionary:@{\n"
" aaaaaaaaaaaaaaaaaaaaa : aaaaaaaaaaaaa,\n"
" bbbbbbbbbbbbbbbbbb : bbbbb,\n"
" cccccccccccccccc : ccccccccccccccc\n"
" }];");
// Ensure that casts before the key are kept on the same line as the key.
verifyFormat(
"NSDictionary *d = @{\n"
" (aaaaaaaa id)aaaaaaaaa : (aaaaaaaa id)aaaaaaaaaaaaaaaaaaaaaaaa,\n"
" (aaaaaaaa id)aaaaaaaaaaaaaa : (aaaaaaaa id)aaaaaaaaaaaaaa,\n"
"};");
Style.ColumnLimit = 40;
verifyFormat("int Foo() {\n"
" a12345 = @{a12345 : a12345};\n"
"}");
verifyFormat("int Foo() {\n"
" a12345 = @{a12345 : @(a12345)};\n"
"}");
verifyFormat("int Foo() {\n"
" a12345 = @{(Foo *)a12345 : @(a12345)};\n"
"}");
verifyFormat("int Foo() {\n"
" a12345 = @{@(a12345) : a12345};\n"
"}");
verifyFormat("int Foo() {\n"
" a12345 = @{@(a12345) : @YES};\n"
"}");
Style.SpacesInContainerLiterals = false;
verifyFormat("int Foo() {\n"
" b12345 = @{b12345: b12345};\n"
"}");
verifyFormat("int Foo() {\n"
" b12345 = @{(Foo *)b12345: @(b12345)};\n"
"}");
Style.SpacesInContainerLiterals = true;
Style = getGoogleStyle(FormatStyle::LK_ObjC);
verifyFormat(
"@{\n"
" NSFontAttributeNameeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee : "
"regularFont,\n"
"};");
}
TEST_F(FormatTestObjC, ObjCArrayLiterals) {
verifyIncompleteFormat("@[");
verifyFormat("@[]");
verifyFormat(
"NSArray *array = @[ @\" Hey \", NSApp, [NSNumber numberWithInt:42] ];");
verifyFormat("return @[ @3, @[], @[ @4, @5 ] ];");
verifyFormat("NSArray *array = @[ [foo description] ];");
verifyFormat(
"NSArray *some_variable = @[\n"
" aaaa == bbbbbbbbbbb ? @\"aaaaaaaaaaaa\" : @\"aaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
"];");
verifyFormat(
"NSArray *some_variable = @[\n"
" aaaa == bbbbbbbbbbb ? @\"aaaaaaaaaaaa\" : @\"aaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaa\", @\"aaaaaaaaaaaaaaaa\", @\"aaaaaaaaaaaaaaaa\"\n"
"];");
verifyFormat("NSArray *some_variable = @[\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
"];");
verifyFormat("NSArray *array = @[\n"
" @\"a\",\n"
" @\"a\",\n" // Trailing comma -> one per line.
"];");
// We should try to be robust in case someone forgets the "@".
verifyFormat("NSArray *some_variable = [\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
" @\"aaaaaaaaaaaaaaaaa\",\n"
"];");
verifyFormat(
"- (NSAttributedString *)attributedStringForSegment:(NSUInteger)segment\n"
" index:(NSUInteger)index\n"
" nonDigitAttributes:\n"
" (NSDictionary *)noDigitAttributes;");
verifyFormat("[someFunction someLooooooooooooongParameter:@[\n"
" NSBundle.mainBundle.infoDictionary[@\"a\"]\n"
"]];");
Style.ColumnLimit = 40;
verifyFormat("int Foo() {\n"
" a12345 = @[ a12345, a12345 ];\n"
"}");
verifyFormat("int Foo() {\n"
" a123 = @[ (Foo *)a12345, @(a12345) ];\n"
"}");
Style.SpacesInContainerLiterals = false;
verifyFormat("int Foo() {\n"
" b12345 = @[b12345, b12345];\n"
"}");
verifyFormat("int Foo() {\n"
" b12345 = @[(Foo *)b12345, @(b12345)];\n"
"}");
Style.SpacesInContainerLiterals = true;
Style.ColumnLimit = 20;
// We can't break string literals inside NSArray literals
// (that raises -Wobjc-string-concatenation).
verifyFormat("NSArray *foo = @[\n"
" @\"aaaaaaaaaaaaaaaaaaaaaaaaaa\"\n"
"];\n");
}
TEST_F(FormatTestObjC, BreaksCallStatementWhereSemiJustOverTheLimit) {
Style.ColumnLimit = 60;
// If the statement starting with 'a = ...' is put on a single line, the ';'
// is at line 61.
verifyFormat("int f(int a) {\n"
" a = [self aaaaaaaaaa:bbbbbbbbb\n"
" ccccccccc:dddddddd\n"
" ee:fddd];\n"
"}");
}
TEST_F(FormatTestObjC, AlwaysBreakBeforeMultilineStrings) {
Style = getGoogleStyle(FormatStyle::LK_ObjC);
Style.ColumnLimit = 40;
verifyFormat("aaaa = @\"bbbb\"\n"
" @\"cccc\";");
verifyFormat("aaaa(@\"bbbb\"\n"
" @\"cccc\");");
verifyFormat("aaaa(qqq, @\"bbbb\"\n"
" @\"cccc\");");
verifyFormat("[aaaa qqqq:@\"bbbb\"\n"
" @\"cccc\"];");
verifyFormat("aaaa = [aaaa qqqq:@\"bbbb\"\n"
" @\"cccc\"];");
verifyFormat("[aaaa qqqq:@\"bbbb\"\n"
" @\"cccc\"\n"
" rr:42\n"
" ssssss:@\"ee\"\n"
" @\"fffff\"];");
}
TEST_F(FormatTestObjC, DisambiguatesCallsFromCppLambdas) {
verifyFormat("x = ([a foo:bar] && b->c == 'd');");
verifyFormat("x = ([a foo:bar] + b->c == 'd');");
verifyFormat("x = ([a foo:bar] + !b->c == 'd');");
verifyFormat("x = ([a foo:bar] + ~b->c == 'd');");
verifyFormat("x = ([a foo:bar] - b->c == 'd');");
verifyFormat("x = ([a foo:bar] / b->c == 'd');");
verifyFormat("x = ([a foo:bar] % b->c == 'd');");
verifyFormat("x = ([a foo:bar] | b->c == 'd');");
verifyFormat("x = ([a foo:bar] || b->c == 'd');");
verifyFormat("x = ([a foo:bar] && b->c == 'd');");
verifyFormat("x = ([a foo:bar] == b->c == 'd');");
verifyFormat("x = ([a foo:bar] != b->c == 'd');");
verifyFormat("x = ([a foo:bar] <= b->c == 'd');");
verifyFormat("x = ([a foo:bar] >= b->c == 'd');");
verifyFormat("x = ([a foo:bar] << b->c == 'd');");
verifyFormat("x = ([a foo:bar] ? b->c == 'd' : 'e');");
// FIXME: The following are wrongly classified as C++ lambda expressions.
// For example this code:
// x = ([a foo:bar] & b->c == 'd');
// is formatted as:
// x = ([a foo:bar] & b -> c == 'd');
// verifyFormat("x = ([a foo:bar] & b->c == 'd');");
// verifyFormat("x = ([a foo:bar] > b->c == 'd');");
// verifyFormat("x = ([a foo:bar] < b->c == 'd');");
// verifyFormat("x = ([a foo:bar] >> b->c == 'd');");
}
TEST_F(FormatTestObjC, DisambiguatesCallsFromStructuredBindings) {
verifyFormat("int f() {\n"
" if (a && [f arg])\n"
" return 0;\n"
"}");
verifyFormat("int f() {\n"
" if (a & [f arg])\n"
" return 0;\n"
"}");
verifyFormat("int f() {\n"
" for (auto &[elem] : list)\n"
" return 0;\n"
"}");
verifyFormat("int f() {\n"
" for (auto &&[elem] : list)\n"
" return 0;\n"
"}");
verifyFormat(
"int f() {\n"
" for (auto /**/ const /**/ volatile /**/ && /**/ [elem] : list)\n"
" return 0;\n"
"}");
}
TEST_F(FormatTestObjC, BreakLineBeforeNestedBlockParam) {
Style = getGoogleStyle(FormatStyle::LK_ObjC);
Style.ObjCBreakBeforeNestedBlockParam = false;
Style.ColumnLimit = 0;
verifyFormat("[self.test1 t:self callback:^(typeof(self) self, NSNumber *u, "
"NSNumber *v) {\n"
" u = v;\n"
"}]");
verifyFormat("[self.test1 t:self w:self callback:^(typeof(self) self, "
"NSNumber *u, NSNumber *v) {\n"
" u = v;\n"
"}]");
verifyFormat("[self.test1 t:self w:self callback:^(typeof(self) self, "
"NSNumber *u, NSNumber *v) {\n"
" u = c;\n"
"} w:self callback2:^(typeof(self) self, NSNumber *a, NSNumber "
"*b, NSNumber *c) {\n"
" b = c;\n"
"}]");
verifyFormat("[self.test1 t:self w:self callback:^(typeof(self) self, "
"NSNumber *u, NSNumber *v) {\n"
" u = v;\n"
"} z:self]");
Style.ColumnLimit = 80;
verifyFormat(
"[self.test_method a:self b:self\n"
" callback:^(typeof(self) self, NSNumber *u, NSNumber *v) {\n"
" u = v;\n"
" }]");
verifyFormat("[self block:^(void) {\n"
" doStuff();\n"
"} completionHandler:^(void) {\n"
" doStuff();\n"
" [self block:^(void) {\n"
" doStuff();\n"
" } completionHandler:^(void) {\n"
" doStuff();\n"
" }];\n"
"}];");
Style.ColumnLimit = 0;
verifyFormat("[[SessionService sharedService] "
"loadWindowWithCompletionBlock:^(SessionWindow *window) {\n"
" if (window) {\n"
" [self windowDidLoad:window];\n"
" } else {\n"
" [self errorLoadingWindow];\n"
" }\n"
"}];");
verifyFormat("[controller test:^{\n"
" doStuff();\n"
"} withTimeout:5 completionHandler:^{\n"
" doStuff();\n"
"}];");
verifyFormat(
"[self setupTextFieldSignals:@[\n"
" self.documentWidthField,\n"
" self.documentHeightField,\n"
"] solver:^(NSTextField *textField) {\n"
" return [self.representedObject solveEquationForTextField:textField];\n"
"}];");
}
TEST_F(FormatTestObjC, IfNotUnlikely) {
Style = getGoogleStyle(FormatStyle::LK_ObjC);
verifyFormat("if (argc < 5) [obj func:arg];");
verifyFormat("if (argc < 5) [[obj1 method1:arg1] method2:arg2];");
verifyFormat("if (argc < 5) [[foo bar] baz:i[0]];");
verifyFormat("if (argc < 5) [[foo bar] baz:i[0]][1];");
verifyFormat("if (argc < 5)\n"
" [obj func:arg];\n"
"else\n"
" [obj func:arg2];");
verifyFormat("if (argc < 5) [[unlikely]]\n"
" [obj func:arg];\n"
"else [[likely]]\n"
" [obj func:arg2];");
}
TEST_F(FormatTestObjC, Attributes) {
verifyFormat("__attribute__((objc_subclassing_restricted))\n"
"@interface Foo\n"
"@end");
verifyFormat("__attribute__((objc_subclassing_restricted))\n"
"@protocol Foo\n"
"@end");
verifyFormat("__attribute__((objc_subclassing_restricted))\n"
"@implementation Foo\n"
"@end");
}
} // end namespace
} // end namespace format
} // end namespace clang