-
Notifications
You must be signed in to change notification settings - Fork 12.6k
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
Refactoring to convert to "named parameters" #23552
Comments
CC @rauschma @appsforartists in case you want to 👍 |
Apologies for being ungrateful: this doesn’t reflect how I use this pattern – I don’t see the parameters as a whole, I see each parameter individually. As a work-around, I inline the type and don’t define it externally. Let me try to convince you, one last time (then I’ll stop pestering you), of the usefulness of a better notation for destructuring (nested destructuring will profit, too!). The following is an extreme example, but there are many similar functions in my code (“Showoff” is the name of my slide framework): function slidesToNodeTree(
{conf, configShowoff, inputDir, slideDeckDir, slideFileName, parentPartNode, visitedSlideFiles}
: {conf: ConfigSlideLink, configShowoff: ConfigShowoff, inputDir: string,
slideDeckDir: ServePath, slideFileName: string, parentPartNode: PartNode,
visitedSlideFiles: SlideFileDesc[]}) {
···
} With a better notation: function slidesToNodeTree(
{ conf as ConfigSlideLink, configShowoff as ConfigShowoff, inputDir as string,
slideDeckDir as ServePath, slideFileName as string, parentPartNode as PartNode,
visitedSlideFiles as SlideFileDesc[]}) {
···
} Alternatively: function slidesToNodeTree(
{ (conf: ConfigSlideLink), (configShowoff: ConfigShowoff), (inputDir: string),
(slideDeckDir: ServePath), (slideFileName: string), (parentPartNode: PartNode),
(visitedSlideFiles: SlideFileDesc[])}) {
···
} Note how, in the last two examples, you actually see parameters with names (vs. a single type for all parameters). The first example looks messy, the last two examples don’t. If you have ever used a programming language with named parameters and liked them there – isn’t this a compelling use case? Given the thumbs-up at a recent comment of mine, there are quite a few people who agree. This is the only aspect of my plain JavaScript code that became less usable after I moved to TypeScript. |
a useful feature 👍 |
I love that you're exploring this, but I agree with @rauschma. I think there's a bit of a cargo-cult bias in the TS community to always prefer interfaces (because they can be extended). Type literals seem like a more appropriate model for the kwargs pattern, because they allow users to treat each named argument individually. If a third party made a "convert to named arguments" command, I would probably find it useful to be able to write function signatures normally and then convert them to an interface/type literal. However, I'd rather the language supported setting the type for a named arg adjacent to the declaration of its name and default value. It's both hard to read and cumbersome to maintain when the types are separated from the rest of the definition. |
This refactor also must refactor all references to the method in the whole project right ? But a very helpful refactor , in my experience, happened many times when an API signature that needs to be backwards compatible, is defined with multiple params and then you keep adding parameters to implement new features or even change parameter type to OR, like existing: PreviousType|NewSemanticType.... |
IMO, it should |
So I was playing a lot lately with Language Service APIs and friends and I have several refactors more or less working fine. This is the one suggested here (I think) : You can easily install them in vscode as an extension: https://marketplace.visualstudio.com/items?itemName=cancerberosgx.vscode-typescript-refactors Probably TypeScript team will implement these more elegantly but in the meanwhile at least is fun. This is kind of a crazy tool that was really helpful to develop plugins quickly and learn the API: https://github.com/cancerberoSgx/typescript-plugins-of-mine/blob/master/typescript-plugin-ast-inspector/doc/evalCodeTutorial.md And now I'm playing with some IPC communication between plugins in tsserver and host editor plugins / |
This is great! Have you thought about the default arguments with no types? Would you infer types there? function foo(a = 1, b?: string) Also would it work in constructor function with class Foo {
constructor(private a: string) {}
} where would you generate a type name if function is anonymous? (function (a = 1, b?: string){}).call(1) |
This would be super helpful to me, even if others would rather a new destructuring syntax for function parameters. So, if the proposed refactoring somehow loses favour over new destructuring syntax, I would still like for this particular one to be implemented. I personally use interfaces instead of inlining object parameter types, especially for large projects. And projects I write that get consumed by others. I've had to wrap a function many times. And if the library doesn't have those function arguments as an interface, then I have to copy-paste their inlined code instead of just using an interface that should already be there. People like to think of interfaces as being "code duplication". But in the big picture, not having those interfaces causes code duplication. So, in my personal opinion, inlining = save time now, but cause code duplication in the future/for downstream users of your library. Interface = a little extra time now, but reduce code duplication and make it easier for downstream users to extend/reuse/wrap code. |
@mohsen1 you're now my favorite QA contact for refactorings 😉
Good test case! Should "fall out" from the naive implementation.
It probably should not for now.
As a first-pass, it's probably reasonable to say that this would only generate a named type for
We could always come back to this and add it for signatures like call type literals, construct type literals, call/construct signatures, and method signatures (i.e. the ambient stuff). |
ummmmmm, I'd like to work on it (If I didn't disrupt your release plan) |
Options your say, why not Params? |
@Kingwl this one might involve more work than other refactorings, but nobody's started working on it yet, so go for it! |
@rauschma then you have two syntaxes to be maintained. Because your version is not compatible with the current version that allows sharing of interfaces between methods. So you get an inconsistency there, because you can't just abolish the current syntax because of exactly this reason. |
@RyanCavanaugh |
For the record, I'd like to not continue discussion around new syntax here. This feature could theoretically work with that syntax anyway so it shouldn't have any bearing on the implementation. |
✨ 🚲 🏠 ✨ Naming poll! Vote:
|
I call them named parameters, but just throwing another idea out there: "convert parameters to destructed object" |
I wish we wouldn't encourage the pattern of making functions take a single object - it thwarts currying. |
I know @rauschma said he'd stop and seems to have kept his word, but he's absolutely right that doesn't express what people want to express. Have people discussed using function foo({a is number, b is string, c? is boolean}): void {
a; b; c
} And the TypeScript code base itself is chock full of long lists of parameters with optionals etc. Empirical evidence for where this could help. |
There were some discussions about I haven't thought through every issue, but I'm wondering about using So the most verbose version of this would be: function someFunc({ foo: bar: string = 'default' }): void {
bar;
} ... whereas the common case would be: function someFunc({ foo:: string }): void {
foo;
} |
There are cases where it is beneficial to take a single object with many parameters. For example, that object can be an immutable data structure and the function is just performing some operation on it and returning a new instance of the data structure. In some cases, those data structures can have dozens of properties... You're not really suggesting we curry in those cases as well, right? |
@DanielRosenwasser what's the reason this was closed? |
@jasonwilliams it was closed because it was implemented |
I love this refactoring, but is there a a reason why it does not work on constructors? I sometimes find myself creating a class where the number of constructor parameters grows out of control, with too many optional ones and I wished that I had instead made one "config" or "options" interface that can be passed in to the constructor. The advantages being:
(I guess these are the same reasons why it's useful on functions, really) I notice that @mohsen1 asked about using it on constructors earlier in this thread and @DanielRosenwasser kiboshed the idea unceremoniously (simply saying "It probably should not for now."). Maybe that's just because it was early on in the development of this refactoring and you all were trying to keep things as simple as possible? What extra complications do using this for constructors create that aren't present for functions? Since this issue is old and closed should I open a new issue called "Make 'Convert parameters to destructured object' refactoring work for constructors"? |
Yes, please open a new issue. Thanks! |
If anyone tried this and wonder why it doesn't work, it might be because you are selecting multiple lines. |
It would look something like this
The text was updated successfully, but these errors were encountered: