Hands-On Design Patterns with Swift
上QQ阅读APP看书,第一时间看更新

Setting Objective-C names from Swift

When you write Swift class names, we follow the recommendation of not prefixing our class names or extensions with a two or three letter code. However, back in Objective-C, those conventions are quite important for a number of reasons, but principally to avoid naming collisions with other objects from different frameworks.

Let's consider the following code snippet that defines a movie:

class Movie {
let title: String
let director: String
let year: Int
/* Initializers */
}

As it is right now, it is not possible to use it in Objective-C as the Movie object doesn't inherit NSObject. Also, because you're exposing your class to Objective-C and not following the naming conventions of Objective-C with the prefix, we should probably rename the class with the @objc(...) declaration:

@objc(FVMovie)
class Movie: NSObject {
/* Original code */
}

The previous class will be exposed to Objective-C as FVMovie and not Movie and can be used properly and naturally in your Objective-C code.

Let's now imagine, you have written an extension for String in Swift called leftPad and you'll need it in Objective-C now. In Objective-C, we deal with NSString, not String, so the extensions are not exposed by default:

extension String {
static let padCharacter = " "
func leftPad(to length: Int, with character: String = .padCharacter) -> String {
/* Your left pad implementation */
}
}

There are multiple steps required to expose it to Objective-C:

  • @objc cannot be applied to String as it's a struct, and Swift structs can't be exported to Objective-C
  • The leftPad name should be prefixed, as per Objective-C recommendations
  • String doesn't exist in Objective-C, but NSString does

Let's create an extension on NSString:

extension NSString {
func leftPad(to length: Int, with character: String = .padCharacter) -> String {
return (self as String).leftPad(to: length, with: character)
}
}

You'll realize that the method is still not exposed to Objective-C as the extension is not exposed by default. You can remedy that by marking the full extension @objc:

@objc
extension NSString { /* */ }

Now you can use the leftPad method in Objective-C:

NSString * padded = [@"Hello" leftPadTo:10 with:@" "];
// padded == @" Hello"

There are a few things we should improve before we can move on with that implementation:

  • The default character used in padding (String.padCharacter) isn't available
  • The method name isn't the best for Objective-C
  • This doesn't follow the recommended prefixed naming conventions

In order to expose the default implementation that requires only the target length, we need to add an additional method:

extension NSString {
func leftPad(to length: Int) -> String {
return (self as String).leftPad(to: length)
}
}

This will leverage the original default implementation in Swift, so it's usable in Objective-C as well:

NSString * padded = [@"Hello" leftPadTo:10];
// padded == @" Hello"

Now let's put proper names on the methods:

extension NSString {
@objc(flv_leftPadToLength:)
func
leftPad(to length: Int) -> String

@objc(flv_leftPadToLength:withCharacter:)
func leftPad(to length: Int, with character: String) -> String
}

Now you'll be able to use those implementations in Objective-C with the following:

[@"Hello" flv_leftPadToLength:10]; // @"     Hello"
[@"Hello" flv_leftPadToLength:10 withCharacter:@"*"]; // @"*****Hello"

In this section, we've focused on adapting our Swift code to Objective-C so it's easier to use and more natural in the Objective-C style. In the next section, we'll focus on the opposite, making our Objective-C code more natural to Swift.