-
-
Notifications
You must be signed in to change notification settings - Fork 220
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
Support #[export(range = (radians_as_degrees, suffix="XX"))]
#783
Conversation
API docs are being generated and will be shortly available at: https://godot-rust.github.io/docs/gdext/pr-783 |
radians: bool, | ||
degrees: bool, | ||
hide_slider: bool, | ||
suffix: Option<String>, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can't you just store the TokenStream
and convert it later? Kinda weird that min
, max
, and step
are all expression but suffix
isn't.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, that seems to save one to_string
call and solves a question I had
8b6e322
to
4cbbdd4
Compare
4cbbdd4
to
c85b6a3
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks a lot!
Could you document the new keys, e.g. here?
https://godot-rust.github.io/docs/gdext/pr-783/godot/register/property/export_info_functions/fn.export_range.html
I know others don't have docs, but #[export(range = ...)]
is particularly complex and has a syntax that isn't used anywhere else. So it would be good to show examples of different possible invocations.
quote! { | ||
#export_func; | ||
::godot::__deprecated::emit_deprecated_warning!(export_range_radians); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Indentation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is what rustfmt produces. It wouldn't pass CI if it weren't like this.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Are you sure? Rustfmt is typically rather conservative when it comes to formatting the inside of macro invocations (as in, often doesn't format at all). That's also the value proposition of tools like Gene Michaels.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
$ rustfmt --check godot-macros/src/class/data_models/field_export.rs
Diff in /home/fp/3rd/gdext/godot-macros/src/class/data_models/field_export.rs at line 402:
}?;
let deprecation_warning = if *radians {
quote! {
- #export_func;
- ::godot::__deprecated::emit_deprecated_warning!(export_range_radians);
- }
+ #export_func;
+ ::godot::__deprecated::emit_deprecated_warning!(export_range_radians);
+ }
} else {
quote! { #export_func }
};
$ rustfmt --version
rustfmt 1.7.0-stable (129f3b9 2024-06-10)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Weirdly, this doesn't produce any diff for me, despite being seemingly the same.
impl Something {
fn produce_ugly_diff() {
match something {
AnonymouStructVariant {
f0,
f1,
f2,
f3,
f4,
f5,
f6,
f7,
f8,
f9,
f10,
} => {
let warning = if *radians {
quote! {
#export_func;
::godot::__deprecated::emit_deprecated_warning!(export_range_radians);
}
} else {
quote! { #export_func }
};
}
}
}
}
My hunch though is that it has something to do with the anonymous struct variant since we need to turn off a lint disallowing large numbers of parameters. But clearly it's not so easy to replicate.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Assuming it's a bug, I've reported it rust-lang/rustfmt#6233
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks a lot for reporting! 👍
So for now you can leave the bad formatting, maybe add a comment that links to this conversation:
https://github.com/godot-rust/gdext/pull/783#discussion_r1669105958
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done
godot-macros/src/util/list_parser.rs
Outdated
pub(crate) fn next_allowed_key_optional_value( | ||
&mut self, | ||
allowed_bool_keys: &[&str], | ||
allowed_kv_keys: &[&str], | ||
) -> ParseResult<Option<(Ident, Option<KvValue>)>> { | ||
let allowed_keys = || { | ||
let allowed_bool_keys = allowed_bool_keys.join(","); | ||
let allowed_kv_keys = allowed_kv_keys.join(","); | ||
[allowed_bool_keys, allowed_kv_keys].join(",") | ||
}; | ||
match self.next_key_optional_value()? { | ||
Some((key, None)) if !allowed_bool_keys.contains(&key.to_string().as_str()) => { | ||
if allowed_kv_keys.contains(&key.to_string().as_str()) { | ||
return bail!(key, "`{key}` requires a value `{key} = VALUE`"); | ||
} | ||
bail!(key, "expected one of \"{}\"", allowed_keys()) | ||
} | ||
Some((key, Some(_))) if !allowed_kv_keys.contains(&key.to_string().as_str()) => { | ||
if allowed_bool_keys.contains(&key.to_string().as_str()) { | ||
return bail!(key, "key `{key}` mustn't have a value"); | ||
} | ||
bail!(key, "expected one of \"{}\"", allowed_keys()) | ||
} | ||
key_maybe_value => Ok(key_maybe_value), | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is quite similar to the top-level KvParser
logic,with the main difference that "keys" (standalone) are now "values". I guess there isn't really anything to reuse?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not really, no. And if I did, it'd look a lot like this.
godot-macros/src/util/list_parser.rs
Outdated
pub(crate) fn next_allowed_key_optional_value( | ||
&mut self, | ||
allowed_bool_keys: &[&str], | ||
allowed_kv_keys: &[&str], | ||
) -> ParseResult<Option<(Ident, Option<KvValue>)>> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would be good to keep terminology consistent everywhere:
- "flag/flags" for standalone boolean keys
- "kv/kvs" for key-value pairs
As for the method name, the "optional" is a bit unclear (I know it refers to "value", but had to read multiple times). Maybe it's not needed given the return type has already Option
? So, next_allowed_flag_or_kv
might be a different way to state it... or next_allowed_key_value
.
Is "allowed" needed at all?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'll use flags, sure.
So the though behind the name was that it's doing the same thing as next_key_optional_value
, but with the possible flag names and kv keys specified, similar to next_allowed_ident(allowed_ids)
. So combining these two names, I landed on what's here. I agree that the name is a lot to parse, but I think it's reasonable to a user of this struct. An API user here mostly likely will be learning the API by reading the source and will first discover those methods I'm basing this name on before this one. It might be reasonable to rethink the naming, but I don't think that's in scope for this PR.
The outer Option
is Some
when there is a value to parse and None
after then end of the sequence, similar to std::iter::Iterator::next()
. If a not-allowed ident is a key, it'll return Err
, so the Option
type doesn't come into play. This is consistent with next_allowed_ident
.
So you're right that this is a bit funny, but it's because I'm combining a couple of existing funny things; I don't think I added any funniness. I will add a docstring to clarify the function, though.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So the though behind the name was that it's doing the same thing as
next_key_optional_value
, but with the possible flag names and kv keys specified, similar tonext_allowed_ident(allowed_ids)
. So combining these two names, I landed on what's here.
That makes sense, in that case you can keep it. Thanks for elaborating!
I will add a docstring to clarify the function, though.
Great 👍
5eee4e8
to
53e8b78
Compare
`radians_as_degrees` was added in Godot [4.2][1] to replace `radians` with identical functionality and a better name. `suffix` was added in [4.0][2] and was seemingly missed when `#[export(range)]` was first implemented. This is the first instance where Godot has deprecated an API and gdext has needed to implement it. Hopefully this will serve as a useful template for future work. This also fixes a previously unknown bug where a `KEY=VALUE` expression in the third slot of `range = (...)` would yield strange error messages. [1]: godotengine/godot#82195 [2]: godotengine/godot#50009
53e8b78
to
d1a4fee
Compare
Thank you! |
radians_as_degrees
was added in Godot 4.2 to replaceradians
with identical functionality and a better name.suffix
was added in 4.0 and was seemingly missed when#[export(range)]
was first implemented.This is the first instance where Godot has deprecated an API and gdext has needed to implement it. Hopefully this will serve as a useful template for future work.
This also fixes a previously unknown bug where a
KEY=VALUE
expression in the third slot ofrange = (...)
would yield strange error messages.Resolves #731