Comments (9)
There's definitely many duplicates of this
#27808, extends oneof
issue, I've seen too many of them.
@noahtallen just use the distributive behavior to make it work
function doSomethingWithData<T extends SomeThings>({thing, data}: T extends unknown ? ArgType<T> : never) {
if (thing === SomeThings.One) {
console.log(data.x); // OK
}
}
Note that you also have to make the literal object foo
declared as const.
const foo = {thing: SomeThings.One, data: {x: 1}} as const;
from typescript.
Hm interesting. It's odd to me that something like T extends T ? ... : never
is required to get inference to work, given that T always extends T. So this is basically just tricking/forcing the compiler to distribute here?
Either way, happy that there's a solution!
from typescript.
Because within the function the compiler doesn't know which type is passed along. It could be SomeThings.One
, it could be SomeThings.Two
, it could also be the type SomeThings.One | SomeThings.Two
. There's definitely many duplicates of this, just don't have one at hand right now.
from typescript.
The compiler should be able to know that SomeThings.One
was passed within this block. It should also be know (based on the contracted specified by the ArgType
interface) that it's impossible to pass mismatched arguments where the data doesn't match the enum:
// We know which one was passed now:
if (thing === SomeThings.One) {
console.log(data.x);
}
// Correctly shows an error, so it does narrow the type of `data` based on the enum here:
doSomethingWithData({thing: SomeThings.One, data: {y: 'xyz'});
I think it evaluates ArgType<T>
too early into a union of all possible values, where if it evaluated it within the if statement, it should know that it resolves to a single possibility
from typescript.
You misunderstand how generics work. Just because thing
equals SomeThings.One
does not mean you're dealing with a { x: number }
data.
This is a perfectly valid and legal call:
doSomethingWithData<SomeThings.One | SomeThings.Two>({ thing: SomeThings.One, data: { y: "abc" }})
from typescript.
That's great, thanks for sharing! looks like as const
isn't needed if you do:
const foo: ArgType<SomeThings.One> = {thing: SomeThings.One, data: {x: 1}};
if (someCondition) {
foo.data.x = 100;
}
// works!
doSomethingWithData(foo);
doSomethingWithData({ thing: SomeThings.Two, data: {y: 'hi'}});
Anyways, I'm curious how this works. Why T extends unknown
when T is known?
from typescript.
Why T extends unknown when T is known?
Because unknown
is the top type that anything can be assigned to it, to make sure that the conditional type always get in the "true" branch. You can replace it with T extends any
or T extends T
, all of them are OK.
And for "distributive" behavior, see https://www.typescriptlang.org/docs/handbook/2/conditional-types.html#distributive-conditional-types.
from typescript.
So this is basically just tricking/forcing the compiler to distribute here?
Yes. Conditional types distribute over type parameters, so:
type ArrayOf<T> = T extends T ? T[] : never;
type Test = ArrayOf<string | number>; // string[] | number[];
whereas without distribution you’d get (string | number)[]
instead.
from typescript.
from typescript.
Related Issues (20)
- Object circular assignment never error HOT 2
- Only resolve source return type when actually needed during inference when generic is used HOT 2
- Target signature provides too few arguments. Expected 1 or more, but got 0 HOT 1
- Convert params to destructed object: Cannot apply refactoring HOT 2
- getPasteEdits returns edits even when no imports are added
- getPasteEdits returns wrong range for primary edit
- Some long function return expressions no longer evaluated in 5.5 HOT 3
- Generic is "`any`-like" for relational operators HOT 2
- inherit interface/type block comment HOT 1
- JSDoc: property access within static initialization block of @extends annotated class from different file
- Isolated Declarations in TS 5.5: State of the feature
- Isolated declarations quick fix suggests `boolean` return type for predicates HOT 1
- Is the "Roadmap" wiki page still maintained? HOT 2
- Typescript 5.5 is not working with yarn v4 and nodeLinker pnp, it forces switching to nodeLinker node-modules HOT 4
- update Object.values and Object.entries docs to mention they look only at own properties HOT 1
- `@import` JSDoc tag doesn't seem to account for `resolution-mode` import attribute in TS v5.5.2 HOT 1
- `/// <reference lib` is stripped out of `d.ts` file HOT 2
- HookWebpackError: Maximum call stack size exceeded in Release 5.5.2 HOT 1
- TS 5.5 type guards overly strict in some cases HOT 3
- Unresolved `any` behaves different from `any` HOT 3
Recommend Projects
-
React
A declarative, efficient, and flexible JavaScript library for building user interfaces.
-
Vue.js
🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.
-
Typescript
TypeScript is a superset of JavaScript that compiles to clean JavaScript output.
-
TensorFlow
An Open Source Machine Learning Framework for Everyone
-
Django
The Web framework for perfectionists with deadlines.
-
Laravel
A PHP framework for web artisans
-
D3
Bring data to life with SVG, Canvas and HTML. 📊📈🎉
-
Recommend Topics
-
javascript
JavaScript (JS) is a lightweight interpreted programming language with first-class functions.
-
web
Some thing interesting about web. New door for the world.
-
server
A server is a program made to process requests and deliver data to clients.
-
Machine learning
Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.
-
Visualization
Some thing interesting about visualization, use data art
-
Game
Some thing interesting about game, make everyone happy.
Recommend Org
-
Facebook
We are working to build community through open source technology. NB: members must have two-factor auth.
-
Microsoft
Open source projects and samples from Microsoft.
-
Google
Google ❤️ Open Source for everyone.
-
Alibaba
Alibaba Open Source for everyone
-
D3
Data-Driven Documents codes.
-
Tencent
China tencent open source team.
from typescript.