This meta-issue collects a potpourri of small Dart language enhancements that will thrill users and not cost too much to implement.
We want to choose a subset of these to implement in the next quarter(s).
This issue is for discussion of which of these features offer the most bang-per-buck, and whether there are other low-hanging fruity features to add.
To qualify, the feature should be well-defined enough that it doesn't require significant further design work (so either very trivial or already designed to a large degree), it should be localized enough that it's unlikely to interfere with other language features, and it should not involve the more complicated parts of the Dart semantics (like the type system). If it can be implemented entirely in the front-end (aka, it's syntactic sugar), then that's a great advantage.
The initial features here are mainly syntactical niceties (syntactic sugar) and features which are largely self-contained.
Go vote for your favorite feature by +1'ing its issue.
library; declaration (#1073)import/export URIs (#649)await (#25)&&= and ||= assignment operators (#122)>>> operator (#120)(See issues sorted by up-votes).
A rough summary of these features is included below.
Allow library; to be a library declaration with an empty name.
This is equivalent to no library declaration, but it provides a place to hang library annotations. Currently you have to give a library a name in order to, say, deprecate it, which seems (and really is) unnecessary.
Can also, over time, be used to move library dartdoc away from the import section.
Allow import shorthands like:
import dart:async; // Same as "dart:async"
import test; // Same as "package:test/test.dart"
import collection:equality // Same as "package:collection/equality.dart"
We'll allow a sequence of some non-space, non-; characters (potentially only ., /, :, letters and digits) to act as a a shorthand for the full URI. We still accept the full URI for cases that contain other characters.
There might be some third-party tools which parse imports using RegExps. If so, those will need to change.
Allow
(elementExpression)
as an elementExpression. That allows
var list = [if (t1) (if (t2) e1) else (if (t3) e2)];
which is otherwise not expressible to express without introducing artificial intermediates because the else would bind to the nearest if.
If parsing to an AST, grouping parentheses can be ignored, but the formatter needs to be aware of them.
Allow ? expression as an element expression, so [?x] is equivalent to [if (x != null) x], but only evaluating x once.
This is a sweet spot of null-awareness, without too much complexity, and it fits well with …?.
Allow (foo) => expr to be written as foo => expr. It only applies to a single untyped argument of an arrow-syntax function expression.
Might not be worth it compared to just having => expr have an implicit parameter of (it).
Allow named arguments to occur before positional arguments in an argument list.
Currently you cannot do expectAsync(count: 2, () { … }), but have to put the count argument after the large function body where it's harder to find.
By allowing named arguments before positional arguments, this becomes possible. Evaluation order of arguments is still left-to-right, so it cannot simply be desugared away by moving the argument. The front end can introduce let constructs to force the order of evaluation and then pass the resulting values in the same order as before.
Allow e.await in async functions to mean the same as (await e).
Since await as a reserved word in those functions, this will not conflict with any existing code.
It reads much better in long chains of operations because it doesn't need parentheses.
It will also have to work with e?.await, e..await and e?..await, and generally look like a getter. Maybe only allow it on expressions of type FutureOr<T> and Future<T>.
Can be desugared in the front-end.
&&= and ||= Short-circuit Assignments (#122)Allow x &&= y to mean x ? x = y : false (aka. x && x = y) and x ||= y to mean x ? true : x = y; (aka. x || x = y), where x is only evaluated once as usual.
These can be desugared entirely in the front-end.
Implement >>> as a user-declarable operator with the same precedence as >>.
This includes allowing #>>> and const Symbol(">>>") as symbols.
(We can then also implement int.>>> when possible).
Might need minimal backend support as well as front-end changes.
Allow #foo= to be a symbol literal for the setter name foo=. Must work for private symbols too.
Allow one or more _ characters between digits in number literals. The _s have no effect, they are ignored when figuring out the numeric value of the literal. They work for decimal, hexadecimal and floating point literals, in the latter case also in the exponent section, as long as they are flanked by digits on both sides (where hexadecimal literal digits include the letters a..f, and all other literals don't).
There is no change to int.parse or similar functions. Anyone needing to parse numbers containing _ characters can remove them first using String.replaceAll.
Can happen entirely in the front-end.
Allow binary integers similar to hexadecimal integers. That would be: 0b1001 for 9.
It's just a new way of writing an integer literal, it can happen entirely in the front-end.
(It's unlikely that we'll want a more general any-radix integer functionality, so this should not conflict with any other feature).
Allow c"x" as a way to write 0x79 (it's an integer literal). The string must contain exactly one code point, so it's a constant expression otherwise equal to "x".runes.first.
Can be implemented entirely in the front-end.
Currently Dart does not allow a generic function type, like T Function<T>(T) to be used as a type argument.
This restriction was introduced as a precautionary limit on the type system, but it has been hit by user code, and there is no easy workaround other than using Function and having more unsafe code.
The issue is amplified by they type inference happily inferring a generic type argument, and then the compiler rejects it immediately after.
We hope that simply removing the check will be sufficient, and that no back-end code is affected.
Allow const constructor invocations in metadata to have type arguments.
@Foo<List<int>>()
int bar = 0;
This is currently not allowed by the grammar. There should be on issues with it.
May I suggest adding https://github.com/dart-lang/language/issues/620 to the list?
It doesn't sound too complex (at least to me, I could be wrong – in which case just ignore it), but is fairly valuable as it gives more flexibility for package authors.
Currently, that restriction prevents exposing a nice API in some situations.
Otherwise, I'd vote for:
These two gives significant readability increase. Named everywhere fits nicely with Flutter. And suffix awaits are a very common use-case
I'd also argue that:
>>> operator (#120)don't seem valuable _to me_.
Imports are added by the IDE automatically. I have yet to see a situation where I needed the library keyword. And Triple shift/symbols/char code constants are very edge-case.
The ones I haven't mentioned yet looks interesting. Not game-changer, but I can see myself using them.
All in all, the way I'd prioritize it _for me_, ignoring how long they take, would be in order:
Named arguments everywhere (#1072)
Unparanthesized function expression parameter (#320)
The link for &&= and ||= is wrong.
@srawlins Good catch, fixed. (It was #122, not #112)
I agree with @rrousselGit about #620, it would be very valuable.
Regarding the proposed list, I would rank them in this order, from more valuable to less valuable:
await (#25) 🥉import/export URIs (#649)>>> operator (#120)library; declaration (#1073)&&= and ||= assignment operators (#122)Please add short assignment!
MyWidget(child, style) instead of MyWidget(child: child, style: style)
Please add short assignment!
That could be similar to https://github.com/dart-lang/language/issues/831.
I don't thing #620 is in a state where it's clear what the solution would be. Multiple features are discussed, including omitting/eliding one type parameter from a method taking more than one, or allowing pattern matching to extract nested types from a single parameter type. The former might be a small feature once it's properly designed, depending on the design, the latter definitely won't.
Whatever the solution will be, it will require a significant amount of design work, which disqualifies it from this list. These features are either trivial in design, or already designed to a sufficient degree, and the features are fairly local in scope (not something where we expect large-scale crosscutting concerns with other features). Anything touching type inference is unlikely to satisfy that.
Is #336 a lot of effort to implement? It would make naming classes in large codebases much easier because many classes could be nested within others. Right now we find that we have to name classes with really long names to avoid conflicts and/or confusion.
@samandmoore Yes, #336 (nested class declarations) is probably non-trivial.
The compilation pipe-line would require a completely new internal structure to hold the nested classes in order to be able to correctly handle the identifier resolution (or they need to desugar it somehow, so it's only the front-end, but then that too will require some amount of work). And that's even if it's only static nested classes.
Waiting for the >>> operator. Currently using the workaround given here
For me the top three would be:
But even more useful than #323 would be let expressions in collection literals (the idea is mentioned in #323 and perhaps narrowly defined enough that they could be considered as part of this discussion).
Apart from that, shorter function expressions at least for getters (instead of (x) => x.getter, of course #320 would help here a bit) would be much appreciated, but I guess that's better solved using more versatile tear offs and I assume that those are too big of a feature to be included here?
Much as I'd love a let expression, I don't think the design is as clear-cut as it looks. Mainly because we might want to go a complete different way instead.
Say instead you could have embedded variable declaration expressions with a scope that is everything dominated by that expression: foo(var list = [], list) would pass the same list twice. That's possibly a better match for Dart than let list = [] in foo(list, list). So, there is some design space here to explore yet before we're ready to add anything to the language.
I would strongly favor a let expression in the example you gave. Not only is it likely to be a more familiar construct to users coming from other languages, but I'd argue that it also fits better into the mental model of the current collection literal design (since you just have to remember that the three constructs if, for and let introduce branching, looping and variable assignment using a very similar syntax). In fact, the same syntax that is used for if and for could also be used for let:
final ifExample = [if (a == b) c];
final forExample = [for (final a in b) c];
final letExample = [let (final a = b) c];
final letExampleWithMultipleBindings = [let (final a = b, final c = d) e];
Right now, trying to develop in a declarative style using lots of immutable objects is still quite a pain in Dart, with the exception of collection literals, which really shine in these scenarios. But it is currently frustrating to constantly encounter the road block of having to use the same expression twice and then having to factor out what could have been a declarative collection into a more imperative version.
await (#25)import/export URIs (#649)I think the shorthand for import/export is a home run. It makes code a lot simpler to understand for those coming from Python/other modern languages; it's also just simpler to type and read.
I would ask to consider including Binary Integer Literals (#581) with Number Digit Separators (#2).
How about optional semicolon?
How about optional semicolon?
I doubt this is a "small" feature.
Added #581 to the list, #2 was already there.
@lrhn what about context based enum member resolution (similar to Swift), where instead of E.m one can just write .m if context type is already known to be E. I think this is extremely simple feature to implement - yet it has a very delightful effect, code becomes less repetetive and easier to read (in certain cases).
@mraleph While I very much like the idea of enum value shorthand (#357), I don't think the idea is well-defined or mature enough to make this list yet. It's not clear how it should handle == or the {e1?.foo:e2} grammar ambiguity, and we probably also need to check the design around switch statements (it might be sufficient to say that the case expressions get the static type of the switch expression as context type). I'm not convinced that it's trivial, so we need to do the design. Needing to do non-trivial design is a disqualifier for this list.
If we do the design, we can come back and decide that it really is trivial to implement, and then I'd love to add it to the list.
(I notice that I already prematurely gave it the label, so I've removed that again).
@lrhn what about context based enum member resolution (similar to Swift), where instead of
E.mone can just write.mif context type is already known to beE. I think this is extremely simple feature to implement - yet it has a very delightful effect, code becomes less repetetive and easier to read (in certain cases).
Replied here: https://github.com/dart-lang/language/issues/357#issuecomment-690617838
For me definitely Named arguments everywhere (#1072) then we might be able someday to get rid of the named child children parameter and put them as positional parameter at the last position.
I don't know if that is possible without too much work. I would like to see a Result<T>, or MayBeError type. I know we don't support unions yet but Dart already has a special Union FutureOr<T> type built in.
IMHO with the introduction of non nullability We need a sound replacement to returning null as a flag to signal errors.
What about Named Arguments Shorthand #1123? Specifically this proposed syntax: https://github.com/dart-lang/language/issues/1123#issuecomment-673138091. It seems like it should be trivial to implement (only adding a bit of syntax and a front end transformation).
Please be stupid, Please be simple.
Most helpful comment
Please add short assignment!
MyWidget(child, style)instead ofMyWidget(child: child, style: style)