I have an extension to walk up the view controller chain (even through container views, which is very handy)
public extension UIViewController // go up to a ce
(Aside, NB, Swift3 needs the "!" on p.next: unfortunately I'm not sure exactly why.)
Because .next
returns an Optional, but p
is not optional. This will crash if you run out of responders.
1) It seems ... dangerous ... to change General to General.self - in fact is it safe and is the meaning the same as General in Swift <3 ?
I'm surprised that worked in earlier versions of Swift without the .self
, but yes, .self
is required in order to directly reference a metatype. Referencing metatypes is somewhat rare in Swift, and can lead to surprising behaviors if done unintentionally, so it requires an extra piece of syntax to say "yes, I really mean the type."
why is that a metatype? Did I "do something wrong" and make it ask for a metatype rather than just a type?
You did this correctly.
2) What the hell is a "metatype"?
The type of a type. An instance of a metatype is a type. Consider:
func f(x: Int)
To call this, you pass an instance of Int
. So similarly:
func f<T>(x: T.Type)
To call this, you pass an instance of the metatype T.Type
, which is a type.
Unrelated, but I would probably rethink this code along these lines. First, it is convenient to be able to treat the responder chain as a sequence. Here's one way to do that:
public extension UIResponder {
public func nextResponders() -> AnySequence<UIResponder> {
guard let first = next else { return AnySequence([]) }
return AnySequence(sequence(first: first, next: { $0.next }))
}
}
Then getting the next responder that matches a type is cleaner and clearer IMO (and works on any responder chain):
public extension UIResponder {
public func firstResponder<T: UIResponder>(ofType _: T.Type)-> T? {
return nextResponders()
.flatMap { $0 as? T }
.first
}
}
...
self.firstResponder(ofType: General.self)?.clickedHamburgerMenuButton()