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

Narrow object property indexed by bracket notation when the key is a known literal type, regardless of where the key comes from #61176

Open
6 tasks done
jcalz opened this issue Feb 14, 2025 · 0 comments
Labels
Help Wanted You can do this Possible Improvement The current behavior isn't wrong, but it's possible to see that it might be better in some cases
Milestone

Comments

@jcalz
Copy link
Contributor

jcalz commented Feb 14, 2025

πŸ” Search Terms

type guard, narrowing, control flow analysis, property, index, known type, literal type, bracket notation

βœ… Viability Checklist

⭐ Suggestion

This is a different re-opening or re-focusing of #10530, not the same as #56389, and was not fixed by #57847. The original motivating example for #10530 (treat obj["key"] like obj.key for narrowing) was fixed long ago, and more recently many of the remaining requests which had been added to #10530 were also fixed (narrowing for obj[key] when key is a variable whose literal type isn't known, for multiple uses of the same key). Those issues are closed. But it seems there are still some issues lumped in with #10530 which were not addressed, so here's the suggestion:

Please enable narrowing of object properties accessed via bracket notation based on the type of the key whenever the key's type is a known literal, even if multiple distinct variables or properties are used as keys. That is, if you have key1 and key2 and keyObj.prop and they are all of the literal type "foo" (or have been narrowed to that type), then treat obj[key1] and obj[key2] and obj[keyObj.prop] all as obj.foo for control flow purposes.

πŸ“ƒ Motivating Example

#51368 gives one, where the variable is a let variable annotated with a literal type or const asserted :

let k: "foo" = "foo";
// let k = "foo" as const; // <-- same behavior
const obj = { foo: Math.random() < 0.5 ? "abc" : undefined }
if (obj.foo) { obj[k].toUpperCase() } // error!
//             ~~~~~~ possibly undefined

Probably I'd say someone should use a const there instead of let.


There's also one from this Stack Overflow question, involving enum-like const-asserted objects:

const Enum = { FOO: "foo", BAR: "bar" } as const;
const obj = { [Enum.FOO]: Math.random() < 0.5 ? "abc" : undefined }
if (obj[Enum.FOO]) obj[Enum.FOO].toUpperCase() // ERROR!
//                 ~~~~~~~~~~~~~ possibly undefined

This one strikes me as quite unfortunate, since an actual enum works just fine here:

enum Enum { FOO = "foo", BAR = "bar" }
const obj = { [Enum.FOO]: Math.random() < 0.5 ? "abc" : undefined  }
if (obj[Enum.FOO]) obj[Enum.FOO].toUpperCase() // OKAY

πŸ’» Use Cases

  1. What do you want to use this for?

See #10530 and various issues closed as duplicates for more use cases.

  1. Current approaches and workarounds:

If TS knows the literal key then you do too, so you could always just use it directly:

const Enum = { FOO: "foo", BAR: "bar" } as const;
const obj = { [Enum.FOO]: Math.random() < 0.5 ? "abc" : undefined }
if (obj.foo) obj.foo.toUpperCase() // OKAY

That's reasonable, but I think it would be nice if there weren't this caveat for enum-like objects where you can't actually use them as keys directly for narrowing purposes.

Playground link to code

@RyanCavanaugh RyanCavanaugh added Help Wanted You can do this Possible Improvement The current behavior isn't wrong, but it's possible to see that it might be better in some cases labels Mar 24, 2025
@RyanCavanaugh RyanCavanaugh added this to the Backlog milestone Mar 24, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Help Wanted You can do this Possible Improvement The current behavior isn't wrong, but it's possible to see that it might be better in some cases
Projects
None yet
Development

No branches or pull requests

2 participants