Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Python-style list comprehensions to GDScript #81788

Closed
wants to merge 1 commit into from

Conversation

kahveciderin
Copy link

@kahveciderin kahveciderin commented Sep 17, 2023

This adds python-style list comprehensions to GDScript.

Examples:

[i for i in 10]

[Vector2(x, 0) for x in range(3)]

[x for x in range(5) if x % 2 == 0]

To Do:

  • List comprehensions
  • Dict comprehensions?
  • Using the iterator inside the list comprehensions
  • Filtering (if) inside the list comprehension
  • Chaining multiple for-if's
  • Better type inference?
  • Fix the issues listed below

Issues:

  • [ Partly fixed by making the error quiet ] "non-zero temp variables" when exiting a block on list comprehensions
  • [ FIXED ] This results in a runtime error: Invalid operands 'Nil' and 'int' in operator '>'.
var x = 123
var arr = [x for x in (range(3) if x > 1 else range(5))]
print(arr)
  • [ FIXED ] Trying to check compatibility of unset value type
var arr = [Vector2(x, 0) for x in range(3)]
  • [ FIXED ] Arrays are completely broken
print([1, 2, 3]) # 3
  • [ FIXED ] Flattening arrays does not work because the iterator isn't accessible from the list of chained comprehension:
var arr = [[1, 2, 3],
      [3, 6, 7],
      [7, 5, 4]]
			
print([x for x in a for a in arr])

Closes godotengine/godot-proposals#2972

@kahveciderin
Copy link
Author

Just tested what Python does in case of an if inside the list comprehension:

This is valid (obviously):

[x for x in range(3) if x > 1]

so is this (ternary on the list, not a filtering operation):

x = 2
[x for x in (range(3) if x > 1 else range(5))]

but this isn't:

x = 2
[x for x in range(3) if x > 1 else range(5)]

So, it seems like just disallowing ternary at that place should work. if a ternary really is needed for the list, it should be placed in ().

@kahveciderin
Copy link
Author

Apart from dictionary comprehensions (which I'm not sure if it should be a part of this PR or not), this PR should be ready for review.

@kahveciderin kahveciderin marked this pull request as ready for review September 17, 2023 18:50
@kahveciderin kahveciderin requested a review from a team as a code owner September 17, 2023 18:50
@kahveciderin kahveciderin reopened this Sep 17, 2023
@kahveciderin
Copy link
Author

That was an accidental close

@AThousandShips
Copy link
Member

Please use git push --force to update your branch instead of doing merge commits

Copy link
Member

@dalexeev dalexeev left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please note that we don't have a consensus on this feature yet. There are concerns about the readability of the syntax and the need for this feature in the context of game development. Regular for loops do not look cumbersome, they are just 3-4 short lines instead of 1 long line. In my opinion, this is purely syntactic sugar, which does not add new functionality, and complicates the grammar of the language. Also arrays have map() and filter() methods.

I took a look at the implementation and added a few comments. Note that you don't have to do anything about this for now, since there is no consensus on the proposal yet.

Additional notes

1. We should not just imitate Python and always copy its features and syntax. It is worth comparing other languages, how they implement or do not implement the feature. Perhaps we will find some other option less confusing. See also the comment about the when token instead of if as a separator for filter expression.

2. The proposal and the PR forgot about typed arrays. In the case of a regular loop, we can specify the array type:

var a: Array[int]
for x in list_expr:
    a.append(f(x))

But in the case of list comprehensions this is not provided as part of the syntax, you can't do it like this (see also #71336):

var a := Array[int]([f(x) for x in list_expr])

I haven't looked into this PR closely, but it seems to always generate an untyped array, and even the following won't work:

var a: Array[int] = [f(x) for x in list_expr]

See also godotengine/godot-proposals#7364. We haven't fully resolved all design issues with typed arrays.

ExpressionNode *list = nullptr;
IdentifierNode *variable = nullptr;
ExpressionNode *condition = nullptr;
bool use_conversion_assign = false;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Adding use_conversion_assign without datatype_specifier doesn't make sense. See #80247. Perhaps list comprehensions should also support a static type specifier.

Comment on lines +1509 to +1510
ExpressionNode *parse_expression(bool p_can_assign, bool p_stop_on_assign = false, bool p_disallow_ternary = false);
ExpressionNode *parse_precedence(Precedence p_precedence, bool p_can_assign, bool p_stop_on_assign = false, bool p_disallow_ternary = false);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We have a similar question in #80085 (see discussion in godotengine/godot-proposals#4775). Perhaps instead of disabling the ternary operator in some cases, we should introduce a new token (when) that will act as a separator. No decision has been made yet.

@@ -3745,6 +3791,20 @@ void GDScriptAnalyzer::reduce_identifier(GDScriptParser::IdentifierNode *p_ident
}
}

// Check if we are inside a comprehension. This allows the comprehension to access the variable it is iterating over.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure that identifier is resolved correctly. We recently fixed a bug with scopes (see #79880). It is worth checking that the resolution here is consistent: a local identifier should not shadow class or global identifier in the list expression. Also, the implementation is probably a little over-complicated (but I'm not sure).

@@ -1501,6 +1520,7 @@ class GDScriptParser {
ExpressionNode *parse_ternary_operator(ExpressionNode *p_previous_operand, bool p_can_assign);
ExpressionNode *parse_assignment(ExpressionNode *p_previous_operand, bool p_can_assign);
ExpressionNode *parse_array(ExpressionNode *p_previous_operand, bool p_can_assign);
ComprehensionNode *parse_comprehension(ExpressionNode *expression);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
ComprehensionNode *parse_comprehension(ExpressionNode *expression);
ComprehensionNode *parse_comprehension(ExpressionNode *p_expression);

Prefix arguments with p_

@@ -2734,8 +2745,40 @@ GDScriptParser::ExpressionNode *GDScriptParser::parse_await(ExpressionNode *p_pr
return await;
}

GDScriptParser::ComprehensionNode *GDScriptParser::parse_comprehension(ExpressionNode *expression) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
GDScriptParser::ComprehensionNode *GDScriptParser::parse_comprehension(ExpressionNode *expression) {
GDScriptParser::ComprehensionNode *GDScriptParser::parse_comprehension(ExpressionNode *p_expression) {

@@ -108,10 +108,10 @@ class GDScriptCompiler {
generator->start_block();
}

void end_block() {
void end_block(bool expect_no_locals = true) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
void end_block(bool expect_no_locals = true) {
void end_block(bool p_expect_no_locals = true) {

@@ -79,7 +79,7 @@ class GDScriptCodeGenerator {
virtual void end_parameters() = 0;

virtual void start_block() = 0;
virtual void end_block() = 0;
virtual void end_block(bool expect_no_locals) = 0;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
virtual void end_block(bool expect_no_locals) = 0;
virtual void end_block(bool p_expect_no_locals) = 0;

@@ -468,7 +470,7 @@ class GDScriptByteCodeGenerator : public GDScriptCodeGenerator {
virtual void end_parameters() override;

virtual void start_block() override;
virtual void end_block() override;
virtual void end_block(bool expect_no_locals) override;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
virtual void end_block(bool expect_no_locals) override;
virtual void end_block(bool p_expect_no_locals) override;

do {
ComprehensionNode::ForIf for_if;

if (consume(GDScriptTokenizer::Token::IDENTIFIER, R"(Expected loop variable name in list comprehension)")) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
if (consume(GDScriptTokenizer::Token::IDENTIFIER, R"(Expected loop variable name in list comprehension)")) {
if (consume(GDScriptTokenizer::Token::IDENTIFIER, R"(Expected loop variable name in list comprehension.)")) {

@@ -2249,6 +2249,7 @@ GDScriptParser::ExpressionNode *GDScriptParser::parse_precedence(Precedence p_pr
// Allow keywords that can be treated as identifiers.
token_type = GDScriptTokenizer::Token::IDENTIFIER;
}

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change

Unrelated whitespace change

List<int> used_temporaries;

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change

@@ -86,7 +86,9 @@ class GDScriptByteCodeGenerator : public GDScriptCodeGenerator {

Vector<StackSlot> locals;
Vector<StackSlot> temporaries;

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change

Comment on lines +1994 to +1997
void GDScriptAnalyzer::resolve_iteration(GDScriptParser::ExpressionNode *list,
GDScriptParser::IdentifierNode *variable,
bool *use_conversion_assign,
GDScriptParser::TypeNode *datatype_specifier) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
void GDScriptAnalyzer::resolve_iteration(GDScriptParser::ExpressionNode *list,
GDScriptParser::IdentifierNode *variable,
bool *use_conversion_assign,
GDScriptParser::TypeNode *datatype_specifier) {
void GDScriptAnalyzer::resolve_iteration(GDScriptParser::ExpressionNode *p_list, GDScriptParser::IdentifierNode *p_variable,
bool *r_use_conversion_assign, GDScriptParser::TypeNode *r_datatype_specifier) {

@nukeop

This comment was marked as abuse.

@kahveciderin
Copy link
Author

@nukeop i think in yesterday's gdscript meeting, the discussion for the inclusion of this feature was postponed to 4.3

@YuriSizov
Copy link
Contributor

What needs to happen for there to be a consensus?

Not every proposal ends up with a consensus. If there are concerns which cannot be addressed then something may not get implemented/approved. While there are clearly people who want this feature, it doesn't mean it's going to be added if language maintainers don't find it fitting the language design.

@nukeop

This comment was marked as abuse.

@YuriSizov
Copy link
Contributor

A consensus, at least in the sense that we use it here, does mean a positive decision to approve the proposed change. It's different from the final decision that maintainers can make to approve or reject the feature. A consensus comes from everyone agreeing on the next step, and while there are still people who request this feature, the only consensus that we can have is to accept it. Of course, if the request is withdrawn after concerns are raised, that would also be a consensus, but that seems unrealistic to me in this case.

So a consensus is not always possible, and this can die due to the lack of a consensus, because we prefer to preserve the status quo when we can't agree on the next step. So the final decision by maintainers can come from the fact that there is no consensus, and that can be the end of the story.

@JohnnyUrosevic
Copy link

@nukeop i think in yesterday's gdscript meeting, the discussion for the inclusion of this feature was postponed to 4.3

How come it was postponed?

@AThousandShips
Copy link
Member

I assume it's because it's a major feature and we're almost at feature freeze so little time to do proper testing

@nukeop

This comment was marked as abuse.

@AThousandShips
Copy link
Member

AThousandShips commented Sep 25, 2023

This PR was opened last week so no that isn't the same, there was no PR for it then

Also what do you mean by "excuse"?

This PR was opened very close to feature freeze, but since the support for this particular feature is very mixed and many strong arguments against it has been raised it is not a simple question of "just add it".

So even if this wasn't still a non-agreed upon feature with mixed support there's no way this could have been rushed through before feature freeze, a week simply isn't enough time to review and test

@nukeop

This comment was marked as abuse.

@timothyqiu
Copy link
Member

I'm also wondering what's the exact reasons for postponing the discussion. Is there something like a meeting summary?

I mean, even when the discussion is postponed due to a lack of consensus, there should be a summary of the viewpoints from both those in favor and those opposed. So people could improve what they have now accordingly.

p.s., There are two different reasons given for the postpone: because a lack of consensus, and because we don't have time to do proper testing. It does sound like a perfunctory response.

@AThousandShips
Copy link
Member

I am not aware of the specific reasons, I gave my guess, but the fact that we are now in feature freeze week or thereabouts such a major feature being added isn't really realistic

It's very frustrating to see valuable contributions stuck in a limbo where they're suspended indefinitely

This is just exaggeration, this PR posted a week ago, it hasn't been lingering at all, a feature such as this isn't something you merge in a week 🙃

@nukeop

This comment was marked as abuse.

@timothyqiu
Copy link
Member

Feature freeze starts from next thursday. I don't think it's reasonable to start rejecting something because we're one or two weeks from feature freeze. Isn't that the point of setting a specific feature freeze date?

@AThousandShips
Copy link
Member

AThousandShips commented Sep 25, 2023

Well this hasn't been reviewed and the specifics are still being discussed, so there's the question of "should we spend time on debating this on short timeframe or leave it for next cycle"

The argument of "rejecting something because we're one or two weeks from feature freeze" only really applies if something is ready to go right?

Again I don't know the reasons here, but there has been questions raised about correctness of some code here, and about the specific syntax, and the need for this feature in the first place, so packing that entire thing, deciding syntax, reviewing code, and testing it into at most 10 days is a bit of a stretch no?

Getting a major feature like this merged in 18 days is fast, very fast

@AThousandShips
Copy link
Member

AThousandShips commented Sep 25, 2023

But again, just my guesses, but the reaction here is very agitated, and presents it as if this feature is being killed when it's just postponed, and presenting it as if this PR is forced to linger when it is very very new

Have a nice day, but those are my theories, let's not get up in arms about our favourite feature not getting here fast enough (I'm specifically talking about the exchange I was having right here, not about anyone else)

@YuriSizov
Copy link
Contributor

YuriSizov commented Sep 25, 2023

Hey everyone!

I think there is a little bit of misinterpretation of what @kahveciderin has said. The discussion itself has been postponed, as in there has been no discussion and more pressing matters have been prioritized. There is nothing to summarize from the meeting, no consensus or lack thereof from it to discuss here.

To put it plainly: this feature is not at the top of the priority list for the GDScript team. It's controversial enough to not be an easy decision to make, and the team is not ready to discuss the idea right now, preferring their time to be used on things which can be resolved in the meantime.

I'm very concerned that this could turn into another case of someone wasting their time working on a full implementation of a feature while it's actually already been decided behind the scenes that this won't be allowed to make it into GDScript.

It's very frustrating to see valuable contributions stuck in a limbo where they're suspended indefinitely with no actionable feedback until one day they're too old to be merged because too much has changed in GDScript in the meantime.

@nukeop You are very much aware that @vnen has made several clear statements, including to yourself specifically, that warn about this feature being controversial and questioning its usefulness. We cannot stop anyone from implementing a PR despite those statements, but you can't claim that there has been lack of clear communication.

Now, it starts to look like you are purposefully fanning the flames under this discussion. If it's genuine frustration, I'm sorry, but you have to learn to deal with the fact that not every suggestion, no matter how popular it may appear, is going to end up being accepted. You have been warned about the entitled tone of your messages already. I would like to remind you about our code of conduct and advise you to keep your comments to constructive feedback. What you're doing is not helping with the development, the review process, or a positive attitude towards the feature itself.


I would strongly suggest everyone to take a break until there is anything worth discussing, i.e. until after the GDScript team has a chance to deliberate on this at some later date.

@nukeop

This comment was marked as off-topic.

@nukeop

This comment was marked as off-topic.

@JohnnyUrosevic
Copy link

JohnnyUrosevic commented Sep 25, 2023

This PR was opened very close to feature freeze, but since the support for this particular feature is very mixed and many strong arguments against it has been raised it is not a simple question of "just add it".

what strong arguments have been raised against it? I haven't seen any personally

Have a nice day, but those are my theories, let's not get up in arms about our favourite feature not getting here fast enough

sorry I am not angry or "up in arms", I'm just confused why this feature has been delayed again and again with no real communication

@YuriSizov
Copy link
Contributor

YuriSizov commented Sep 25, 2023

@JohnnyUrosevic You can refer to the linked proposal, godotengine/godot-proposals#2972, for arguments for and against the feature, as well as official communication on the matter. Do note that even if you or somebody else don't consider arguments against as valid, doesn't mean we can ignore them. It would be very improper for maintainers to only take seriously those who want the feature.

I would also like to repeat myself: everyone interested in this feature should take a break from discussing it and wait for more news. There is absolutely no point in asking the same question over and over again to get the same answers. I appreciate that you may not care that some community members and maintainers don't like the idea of this feature and you can say that they are free not to use it. That still puts the proposal into the controversial territory.

Language design is hard. There needs to be a consensus why we are making the syntax, the parser, the grammar more complicated.

@JohnnyUrosevic
Copy link

JohnnyUrosevic commented Sep 25, 2023

@YuriSizov I have been following that thread for years. As far as I can tell, no one in charge of designing the language has said anything in that thread about what concerns are preventing them from implementing this. Last I heard was "we will discuss it after Godot 4" and then silence. I just think some transparency would be nice for such a heavily suggested feature. I have been taking a break and waiting for news for years at this point, I just want to see what the big issue is

@YuriSizov
Copy link
Contributor

YuriSizov commented Sep 25, 2023

@JohnnyUrosevic The summary from the people behind the language has been given 5 years ago:

We've discussed this on IRC with many core developers, and the general feeling is that the syntax of list comprehensions makes code quite hard to read, and goes against the design principles of GDScript to have a straightfoward and easy to read syntax.

Most use cases for list comprehensions can easily be replaced with one (or nested) for loops, which are a lot more readable even if a couple lines more, so we don't see a real use case that would warrant the syntactic complexity.

Again, you might disagree, and that's fine. Nobody is trying to convince you that your preferred syntax is wrong. We just didn't find it fits the goals of the language, and the argument in favor of it being readable is mostly supported by people who are already used to the syntax from years of Python usage. GDScript doesn't require you to learn Python. It doesn't require you to build Python-specific experience and familiarity. And if we take that out of question, it becomes less of a strictly positive change.

We also have to consider, as I've mentioned, making the parser and the grammar more complicated, covering all corner cases, making sure that the new language feature can be used cohesively, doesn't conflict with other features. Just because it's a solved problem in Python (I assume), doesn't mean it would be any easier to solve it here. GDScript is a different language with only vague similarities with Python. It owns Python nothing in terms of adapting its syntax.

Finally, for such a passionately discussed language feature, it is little else than syntactic sugar. It doesn't really unblock new ways to code logic and implement algorithms. It just lets you write code that is usually represented in blocks as a one-liner. So we are discussing adding a new complex syntax that offers little functionality and may lead to people writing GDScript that other community members will find hard to comprehend. If you can't see why this is a controversial matter, I don't think any answer that maintainers can give you would satisfy you.

Whether or not this puts the end to this discussion, I'm leaving it at this. Any further discussion unrelated to the actual code and feature review on this PR lowers its chances of being reviewed, because these long discussions on PRs get in the way of reviewers.

@JohnnyUrosevic
Copy link

@YuriSizov so if they have made their decision and aren't gonna change it why is the proposal not closed?

@kahveciderin
Copy link
Author

I normally wouldn't be a part of this discussion here, as I felt like I (with the help of other people in the discussion as well) expressed why this feature is needed, but after seeing this discussion here as well I felt the need to chime in.

As the owner of the proposal & this PR, and also as someone who has worked with nearly every language under the sun in the past ~10 years, I'd like to offer my perspective. I understand that GDScript is a domain-specific language, tailored to gamedev, but at least to me, it feels incomplete most of the time. When I first started my Godot journey, GDScript seemed like the least polished part of Godot, and I honestly still think the same. The weird limitations it imposes are a hurdle in my development process, and thus GDScript actually reduces my productivity when working on Godot. GDScript lacks basic functionality that most other programming languages do have, and therefore isn't "done" (like v0.x.y of a product). I don't remember feeling this confined in a programming language ever, and that's not in a good way. Seeing how the proposal evolved, I began to wonder how many more awesome GDScript features are waiting for approval, just like this one, with (in my opinion) weird reasons (like how complexity is quoted in this proposal), and how complete it has the potential to feel when the language becomes just a bit more mature. (I don't want to be misconstrued, it's not my intention to disrespect anyone who has worked with the development of GDScript at all, I'm just stating my experiences, and all my assesments are subjective, maybe GDScript is someone's favorite language, I don't know)

Moving on to the topic in hand, the purpose of list comprehensions is more than just line reduction; it syntactically expresses that we're creating a list out of an iterator (sometimes conditionally). In my humble opinion, I don't find list comprehensions hard to read. Following a for loop is harder than reading a list comprehension. You need to make sure the for loop doesn't have any other side effects while reading, but list comprehensions do one thing and one thing only, which is to create a list from an iterator.

Even if we consider that list comprehension syntax might be a bit confusing for an absolute beginner programmer, I honestly don't think GDScript should be someone's first programming language. Just like SQL, it's a specialized tool, and beginners typically learn programming concepts in more general-purpose languages before diving into domain-specific ones.

List comprehensions and similar constructs are already part of languages such as Python, C#, Fortran, Elixir, Erlang, F#, Haskell, and more. In the proposal, someone has also mentioned that the representation is also there in math ({ n : n ∈ { 0, 1, 2, ..., 99 }, 2 | n }), so it isn't something out of the ordinary for most people I would say.

I understand that the core maintainers are trying to see this feature through a game developer's glasses and assign meaning to it in that context. However, saying that this feature isn't strictly about game development is similar to saying, let's say, the XOR operator isn't strictly about game development. Truth is, it has it's place. I (and I assume some of the supporters of this proposal) feel rightfully limited when working with iterators & lists, and this limitation honestly decreases my productivity a lot because now I either need to write a function (or a lambda) that does what I want, which is one more thing to keep track of.

Also I would like to add that there are some misunderstandings in the proposal. Some people, for example, talked about the list comprehensions not being able to use break or continue etc. so they were against the proposal, which leads me to think that there might be a huge misunderstanding about what list comprehensions are - they aren't just fancier for loops. The loop is just an implementation detail. List comprehensions allow us programmers to pseudo-declaratively express that we are creating a list out of some iterator, which is more readable than writing the raw for loop. I don't care how the list is created - it could be delivered by carrier pigeons for all it's worth, I just want to express that I'm creating a list from an iterator, and having a native construct for this exact purpose seems like a valid reason to me to have this feature in GDScript, even if it doesn't "unblock new ways". At that rate, a language is turing complete when there's a loop and a conditional. Does this mean we can just remove everything except those (lambdas, map/filter/reduce stuff, dictionaries, arrays, functions, else etc.) since those don't unblock new ways for a problem to be solved as well. At the end of the day, everything is syntactic sugar on top of the CPU-specific machine code.

Regarding the delays in the discussion phase, and also the old (I think now 6-year-old?) PR getting closed after a lack of discussion, I understand that this isn't the #⁠1 priority right now, and I wouldn't expect this to be included in the release anytime soon. But I invite everyone, especially those who are strictly for this change, along with the core GDScript maintainers, to use this version of GDScript with list comprehensions for a while to see how it improves things. Those who have a use case that could be worth mentioning can simply mention it in the proposal discussion, so we would have more strong reasons to (hopefully) get this accepted.

Anyways, just my two cents.

@LinuxUserGD
Copy link
Contributor

Closes godotengine/godot-proposals#2972

Also supersedes #51997

@nukeop

This comment was marked as abuse.

@Calinou
Copy link
Member

Calinou commented Oct 12, 2023

Since v4.1.2 has been released last week, I'm assuming the feature freeze is over.

No, as feature freeze for 4.2 has just started 🙂

@FurkanKambay
Copy link

FurkanKambay commented Oct 17, 2023

I was researching a missing feature in C# and came across this Stack Overflow answer that seemed relevant to this topic. Let me explain.

Basically, C# left out a useful feature because of reasons similar to ones given here like language purity, not increasing complexity, fitting the language design, and a notion of how you are "supposed" to code.

Yes, it's not a 1:1 comparison since the C# feature isn't syntactic sugar, and they did have another reason for not implementing it (which was debatable too, see 2nd answer). However, it is similar in the sense that there is a demand for it, it's been done in other languages before and proven useful, and the only reason for not implementing it is to keep the language simple.

List comprehensions are an elegant way to write code, they are more readable (subjective, sure) than multiple lines of a for loop, and everything else others have probably already said. Yes, they are technically syntactic sugar but it improves the developer experience, which is, in my opinion, something we should care more about than the approachability of the language by beginners. A useful feature that's in demand shouldn't be left out because of reasons like keeping the language more beginner-friendly. Also, languages borrow features from each other all the time, and it's not like there's going to be a new groundbreaking syntax for some incredible novel functionality. Comprehensions aren't exclusive to python, and GDScript doesn't have to shy away from implementing it with such shallow reasoning.

That being said, I am aware that features require maintaining, checking for edge cases, etc. I've never worked on a compiler before so I'm not aware of all the possible roadblocks but my intention with this comment is not to say "what are we waiting for, just merge it already", but to just voice my support for the feature and share my thought process on this matter since I think it will apply to other feature requests down the road (or current ones). If we want Godot to improve and be used by more and more developers, it's inevitable that people will ask for features like this that some people might find unfit for the language design. However, I think it's also important to realize that programming languages are not prescriptive, they are descriptive. They evolve over time because they are just tools at the end of the day and developers' needs change and more people use the language so it needs to adapt. That's not to say that it needs to implement every feature and become bloated, of course.

I can't even write a statically typed Dictionary[string, object] right now. Not trying to complain, just stating the reality. I love what Godot stands for and want to see it succeed. Hopefully GDScript can improve and mature, because at this stage, it needs to.


This felt a bit dramatic to type, but I felt the arguments against this feature were just shallow. Sorry if my language sounded rude or anything. I recognize that I haven't contributed anything to Godot and I respect the previous contributors for their work.

@lemonJumps
Copy link

Hi, is dictionary comprehension going to be a part of this pr as well?
I kinda need this feature lol.

@kahveciderin
Copy link
Author

Hi, is dictionary comprehension going to be a part of this pr as well?
I kinda need this feature lol.

@lemonJumps i can work on it if that's desired, feel free to share your use case though at it allows us to get more recognition from the people who are responsible for deciding if this gets merged or not.

@nukeop

This comment was marked as off-topic.

@Zireael07

This comment was marked as off-topic.

@godotengine godotengine locked as too heated and limited conversation to collaborators Oct 17, 2023
@YuriSizov
Copy link
Contributor

This feature won't be a part of 4.2, and until there is a decision by the GDScript team, there is not much that can be discussed. So let's give it a bit of time, shall we?

@adamscott
Copy link
Member

For the reasons mentioned here, I'm closing the PR.

Thanks for your work, nonetheless, @kahveciderin. Don't hesitate to join us more often in our weekly meetings.

@adamscott adamscott closed this Dec 4, 2023
@godotengine godotengine unlocked this conversation Dec 4, 2023
@YuriSizov YuriSizov removed this from the 4.x milestone Dec 4, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Add Python-style list comprehensions to GDScript