That is strange. Your solution with the typealias might reveal why it's happening. The compiler is probably looking for the name Foo.Bar.Bar since you're in Bar's scope, and when it can't find that it throws that error. By making the typealias, you're creating a type Foo.Bar.Bar. In this case though it would be pretty trivial to just swap Bar for Self, which should work. Or, you can use fully-qualified names:
Name lookup in Swift is lexical: when you’re in the original declaration of Bar, name lookup looks in Bar, then the lexically-containing Foo (whether it’s an extension of Foo or not), then the top level. When you’re in an extension, the lookup looks in Bar (the current type), then the top level, cause that’s what contains the extension declaration.
It wouldn’t have to work this way, but (1) changing this would be source-breaking, so we’d have to have a strong motivation and limit it to a new language mode; and (2) with the current rules, moving a type in or out of another type, or from one parent type to another, can be a source-compatible change as long as you leave a typealias behind, and changing the lookup rules here would break that.
"if Self and Bar are equivalent how come I can use one but not another?!"
Well, they are not fully equivalent...
When you "cd" into a folder "/Users/joe/Desktop/Foo/Bar" and try to do "ls Bar" - that won't work (assuming you don't have another "Bar" in there!), although your could still do "ls .", similar to Self. To do "ls Bar" you'd need to step out of "Bar" into "Foo", but then "mkdir", etc would create items not where you want (you want them in "Bar").
If you "cd" into "Foo" you could type "mkdir Bar/bar" to make a directory inside "Bar". I hope we'd have the equivalent in Swift one day: