Support use of an optional label for the first trailing closure

My reply to you would be that, again, as a generalization of the existing syntax, there is nothing new; if anything, it's subtracting a special treatment for the first trailing label.

I agree with you that it creates a result that is unfamiliar. But again, by construction, if it's actually adopted by users, then it would cease to be unfamiliar and you would cease to be unsettled by it. If users can't get over the unfamiliar appearance, then they won't use it, and because it would then be absent you wouldn't be unsettled by it. The unfamiliarity problem is necessarily a transient and self-resolving one.

It is also explicitly discussed in the proposal text, so I'm not sure how much mileage we're going to get here dwelling on it. Your point is well taken and, when the time comes for the core team to evaluate the proposal on its merits, it will surely be on their minds.

5 Likes

AFAIK there is currently no place in the language where there are two plain text names separated only by a space. That is the spelling I'm saying is being added. That is the brand new thing users will have to learn to visually parse. People can get used to ~anything, but I'm sure you'd agree it's better if they don't have to.

1 Like

We really aren't saying anything different: the appearance of [1, 2, 3].first where: { ... } has not arisen before, and it is therefore unfamiliar.

There is no new rule to learn, however, neither for people nor for machines. (You can see that for yourself in the implementation: the substantive changes involve telling the parser to return early from several functions; the rest are typo corrections and internal refactoring.)

This is an implicitly circular argument. Let me show you:

  1. People can get used to a brand new Porsche, but I'm sure you'd agree it's better if they don't have to.

  2. People can get used to a (new) broken down Lada, but I'm sure you'd agree it's better if they don't have to.

I would agree to the latter statement, because a broken down Lada is bad, and therefore it's better if I don't have to get used to it. I would disagree with the former statement, because a brand new Porsche is very nice, so it's not better if I don't have to get used to it: in fact, I would very much say, "I could get used to that!"

So you see, it's not the getting used to that's the problem, it's what we're getting used to. Therefore, implicit in your statement is that what takes getting used to is not good. But your argument against the what (i.e., the proposed solution) is that it's unfamiliar (i.e., that it takes getting used to)!

[Unless, of course, you want to assert axiomatically that having to get used to anything, good or bad, is not good. In which case, well, I will concede the point and eagerly volunteer to suffer having to get used to your next new car/gadgets/other worldly goods :stuck_out_tongue:]

In this proposal, I explain several motivating problems to be solved, and how generalizing the existing syntax for labeled closures will solve those problems. Therefore, I have concluded that what I've proposed is a good thing (though perhaps not to the level of being equivalent to a brand new Porsche). So no, I do not agree that "it's better if [users] don't have to" get used to it; in fact, I think many will say, "I could get used to that!"

10 Likes

"Two [identifiers] separated only by a space" occurs very commonly in Swift, since any function declaration containing a parameter whose external label differs from its internal name has this characteristic:

func foo(label argName: Int)

(Bonus: the second identifier is also followed by a colon in both cases.)

The two situations aren't completely analogous (one is a declaration, the other is a call), but sticking to your original constraints, juxtaposition does exist in Swift already.

Speaking of parsing, there are constructions that this proposal fixes that the compiler cannot ever learn to parse unambiguously with the syntax supported by the language today, such as unlabeled trailing closures in conditional statements, or using trailing closure syntax when calling function overloads that differ only by the label of their final closure argument or when defaulted arguments are involved. These are concrete and known issues. On the other hand, users potentially being confused by seeing two identifiers side-by-side is hypothetical, can be easily learned, and I think the proposal addresses it sufficiently.

12 Likes

For what itā€™s worth, that was my exact reaction. It took a moment, but after seeing what it enabled, and typing it out for the first time, it was something that I wanted to use.

4 Likes

Trailing closure syntax arguably is broken, and has been since inception. The SE-0279 change to the API Design Guidelines demonstrates the existence of a real problem. Fixing that problem is well worth the adoption of that which is proposed, here.

4 Likes

Well done. I concur with the proposed minimal-impact approach to addressing the issue, but do hope to see a mandatory-label annotation and/or more aggressive migration at some point in the future.

Question: Might the recent change to the API Design Guidelines be amended? Perhaps, as follows:

Name functions assuming with awareness that the argument label of the first trailing a closure will may be dropped at the call site if a user calls the function with the closure as a first trailing closure. Include meaningful argument labels for all subsequent closures.

Or, perhaps, it would be best to leave that alone, for now.

6 Likes

Thanks for raising this and continuing the conversation :)

Honestly I keep thinking the same as in the previous proposal, I don't think we're fixing anything worth changing the language.

When I read

First, I don't like at all the floating label, I don't want to have to parse white space to read Swift code, that's okey for other languages but in Swift we have delimiters.
Second, I don't see what we gain compared to a fully call syntax.

I see the appeal of trailing closures because they simplify quite a lot the calling site (parenthesis, labels), but when you add a label you are not gaining anything.

x = [1, 2, 3].first(where: { $0 > 2 })
x = [1, 2, 3].first { $0 > 2 }
x = [1, 2, 3].first where: { $0 > 2 }

That said, because the previous proposal was already accepted I guess that given an official review period for this I would be kinda forced to say yes to it.

3 Likes

Keep in mind, this is highly motivated in the context of multiple trailing closures. Especially those that are symmetrical.

Yes thanks @clayellis, I'm aware of that, I just picked a simpler example I guess.

In any case checking at one with multiple closures

My two opinions still apply. At a quick glance is hard to parse, I need to realise there is no . to know that state is a parameter label and not the name of a function to the output of scope. I still think that these floating labels are a bad idea.

To compare with the existing Swift syntax I still don't see what's that huge difference and gain that makes people prefer these changes.

store.scope(
    state: { $0.login },
    action: { AppAction.login($0) }
)

We're literally talking about 1 set of parenthesis (that help clarify that all of this is a function call) and 1 comma (that is necessary when you do this in one line).

Talking about lines, the examples proposed all use a nice formatting that makes it clearer, I would argue that the same can be said about the original syntax.

store.scope(state: { $0.login }, action: { AppAction.login($0) })

Of course if you write this you may get parenthesis blindness (which will bring you objc memories ^^), but let's be honest, nobody writes it like this. You always endup using multiple lines with proper indentation which at the end is the same thing we're using in the examples for the new syntax.

In any case, I know that I'm probably in the very small minority here, and since a change already was accepted I don't see this another one not being accepted. So all of this given I'm sure I will still be in favour of this new proposal.

6 Likes

This proposal answered basically all questions I had about what happens after SE-279 and what are the possible future directions for trailing closures. Aligning declaration and call sites is something I'm very much in favour of and this proposal brings us there one step closer.

Consistency in how trailing closures work would, in my opinion, make them easier to understand by newcomers as there wouldn't be any exceptions based on what position the closure appears at.

(sorry @GetSwifty, this was not supposed to be a reply to your post, I can't find a way to remove that)

1 Like

Iā€™m not imagining this as something desirable (I called it hypothetical). My point is that the semantics of drop are partly conventional. There is nothing inherent in the name that suggests dropping from the front (or back). Somebody unfamiliar with the convention could reasonably think drop { $0.isEven } would drop all elements that match the predicate.

I agree that there are other directions that are better improvements than only requiring the existing label. But IMO, using the label in the current signature is a remains an improvement over eliding it.

8 Likes

I completely agree, and I think SE-279 makes the language worse (reduced clarity at the call site, harder to teach, can no longer use parentheses balancing to find where a long call-site ends, etc, etc) while offering no significant advantage. However given that it was already accepted, I much prefer a world where we can at least have all the labels, so I fully support this proposal.

15 Likes

Yes, and the reason they are is that a new user doesn't really understand what that closure is doing there: the space between the function root name and the closure doesn't signal clearly that the closure is a parameter of the function. That's why newcomers find trailing closures confusing. This proposal will make them less confusing. If the argument to focus on here is "clarity for newcomers", I can assure you that requiring labels for all trailing closures will be by far the best way for newcomers to understand what's going on.

Of the examples in you comparison, the following:

thing.bind
  get: { otherThing }
  set: { otherThing = $0 }

seems to me the most clear by far. It's the only one where the symmetry of arguments, related to the nature of the bind function, is spelled out the best, and will probably be the easiest for newcomers to undestand.

This, conversely, will probably be terrible for newcomers:

thing.bind(get:) { otherThing }
  set: { otherThing = $0 }

Other than being ugly, a newcomer will probably not understand at all what that (get:) is: a lot of Swift is structured to read like natural language, and this doesn't; wrapping the parameter label in parentheses next to a root function name is a language detail that many people, even long-term Swift users, don't know.

It's not much of an advantage if some syntax already exists in the language, but it's ugly and hard to parse. SE-279 introduced a new, pretty heavy syntactic element, that is, an argument label (with :, that's what means in Swift "argument label", the same that would mean for the first closure) for all closures but the first: adding it to the first is really just an extension of that proposal.

I personally consider this super-clear and definitely Swift-like:

myArray.first where: { $0 < 5 }

the reason being the colon after where: the colon in Swift, in the context of calling a function, is used to indicate argument labels, so this reads as an argument label with its argument but without surrounding parentheses. Are surrounding parentheses "Swift-like"? Well, it depends: on function calls, yes, but on statements like if and switch they're not, so the language didn't really make its mind for now. A good thing about SE-279 and this proposal is that they move the language in a direction of less parentheses, and I couldn't be happier: most of the time they're redundant, make readability worse and represent unneeded additional syntactic burden where what's going on is easily understandable from the context.

On the other hand, something like this would be problematic:

thing.bind
get: { otherThing }
set: { otherThing = $0 }
myArray.first
where: { $0 < 5 }

With no indentation and no spaces, this would be hard to parse, but this is really a styleguide problem, and it's mostly the same right now: even this is bad:

thing.bind(
get: { otherThing },
set: { otherThing = $0 })
myArray.first(
where: { $0 < 5 })

the additional final character (being (, ) or ,) doesn't improve the situation. A proper formatting of the whole thing would be:

thing.bind(
  get: { otherThing },
  set: { otherThing = $0 }
)
myArray.first(
  where: { $0 < 5 }
)

Here's the same without extra redundant characters:

thing.bind
  get: { otherThing }
  set: { otherThing = $0 }

myArray.first
  where: { $0 < 5 }

It reads better, no context is lost, and there's no unneeded characters. Notice that this could be the same also for regular parameters, not just trailing closures, but the latter allow for an easier parsing thanks to the curly braces.

5 Likes

Thank you all for a wonderful discussionā€”and for staying for the most part away from rehashing SE-0279. Many will agree, I hope, that itā€™s been just as important to prove to ourselves that we can engage with this topic in a productive and dignified way as it has been to tackle the technical details themselves.

I think weā€™ve captured many well articulated sentiments of the community in this thread (not to mention all the ink spilled before). Iā€™ve gently augmented the text to include all the interesting arguments elucidated here about caller appearance and created a PR for the proposal, with the hope that the core team will proceed to review soon!

26 Likes

"Treat warnings as errors" being turned on seems like standard practice for projects I work on these days that run a tight ship.

I'm not sure library authors should be deciding what syntax I as a client of the library get to use. Recommending (== making it a warning) seems fine, requiring (== an error) feels a bit heavy handed.

Just my twenty cents (inflation! :) ).

1 Like

Thank you very much for putting this together. Clearly stated and very helpful.

The fact that labels may be elided in trailing closures is an anomaly that (based on the recent discussions) is widely considered a mistake. We don't allow users to choose to elide any other labels a library author has chosen. I don't think this flexibility would be accepted today if it wasn't already a part of the language.

9 Likes

Thanks. I guess the kinds of places where eliding the labels makes sense and really does seem to improve readability are places that library authors tend to make them optional in the declaration. Good enough. I definitely support this proposal as written, was just not so sure about your push to make it an error.

I hope while we're considering updating these rules we'll also consider how this question should be answered if and when the rules are updated.