I don't think anybody has suggested dropping any parameter labels, and I wouldn't want that to be possible.
I love this idea as pitched, and would use it a lot. However, I will be this "anybody" and say that dropping parameter labels when using tuple shorthand would address the related problem of initializers which were designed to require pointless argument labels.
For example, basically any time there's an x and y? That's a design from a language without tuples. We've got .0, .1, and parentheses around pairs. They don't need letter names that don't scale past three dimensions. Allowing for dropping those labels when everything would otherwise would be unambiguous is a win. (Note: past xyz, tackling readability of long tuple element indexes, I consider probably to be an IDE feature.)
Different, concrete example:
URLQueryItem has one initializer. Would we even have this type instead of a (String, String?)
tuple in a new Swift-only API? Probably not, but it's a necessary type for now.
The API we're supplied with is, as with Jens's Rect<Float>
, is noisy:
var components = URLComponents()
components.queryItems = [
.init(name: "a", value: "nil"),
.init(name: "b", value: "2")
]
XCTAssertEqual(components.string, "?a&b=2")
We can fix it, using Swift's "mechanism for getting rid of unhelpful argument labels", but only because it's an array :
components.queryItems = [("a", nil), ("b", "2")].map(URLQueryItem.init)
Not too bad, but the language's clearest form available is:
components.queryItems = ["a": nil, "b": "2"]
To get there, I say that some arrays are basically ordered dictionaries:
extension Array: ExpressibleByDictionaryLiteral where Element: ExpressibleByKeyValuePair {
public init(dictionaryLiteral elements: (Element.Key, Element.Value)...) {
self = elements.map(Element.init)
}
}
…and that a key-value pair is just a 2-tuple with a special relationship between its elements:
public protocol ExpressibleByKeyValuePair: ExpressibleBy2Tuple {
typealias Key = Element0
typealias Value = Element1
}
public protocol ExpressibleBy2Tuple {
associatedtype Element0
associatedtype Element1
init( _: (Element0, Element1) )
}
And that initializer, which throws out argument labels which ought not to exist, is where the clutter and busywork comes in:
extension URLQueryItem: ExpressibleByKeyValuePair {
public init( _ pair: (String, String?) ) {
self.init(name: pair.0, value: pair.1)
}
}
As nice as the dictionary literal syntax, that all that allows, is, if this pitch got into the language with the extension I'm suggesting, I'd be happy to throw all of this out and go with:
components.queryItems = [("a", nil), ("b", "2")]
(Although, if that meant that the compiler could understand the following without writing an initializer in the extension body, that could be handy. I'm not going to think about the ramifications of that idea yet.)
extension URLQueryItem: ExpressibleByKeyValuePair { }
I will not be happy to use code that utilizes the pitch, unaltered, which only allows for this—an improvement, yes, but only 1/3 of this case's problem:
components.queryItems = [
(name: "a", value: "nil"),
(name: "b", value: "2")
]
So, what I'm talking about is more work, and surely more contentious, but I think it or something that looks like it is the logical step after this pitch.