π 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
- What do you want to use this for?
See #10530 and various issues closed as duplicates for more use cases.
- 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
π 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"]likeobj.keyfor narrowing) was fixed long ago, and more recently many of the remaining requests which had been added to #10530 were also fixed (narrowing forobj[key]whenkeyis a variable whose literal type isn't known, for multiple uses of the samekey). 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
key1andkey2andkeyObj.propand they are all of the literal type"foo"(or have been narrowed to that type), then treatobj[key1]andobj[key2]andobj[keyObj.prop]all asobj.foofor control flow purposes.π Motivating Example
#51368 gives one, where the variable is a
letvariable annotated with a literal type orconstasserted :Probably I'd say someone should use a
constthere instead oflet.There's also one from this Stack Overflow question, involving
enum-likeconst-asserted objects:This one strikes me as quite unfortunate, since an actual
enumworks just fine here:π» Use Cases
See #10530 and various issues closed as duplicates for more use cases.
If TS knows the literal key then you do too, so you could always just use it directly:
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