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

Compiler option for implicitly adding this context on class member methods #28548

Open
5 tasks done
flash1293 opened this issue Nov 15, 2018 · 2 comments
Open
5 tasks done
Labels
Add a Flag Any problem can be solved by flags, except for the problem of having too many flags In Discussion Not yet reached consensus Suggestion An idea for TypeScript

Comments

@flash1293
Copy link

Search Terms

this context class

Suggestion

A new compiler option --enforceClassMemberThisContext. It causes the compiler to implicitly set the this context of class methods which use this somewhere in the method body and behave as the user had done this manually on each method.

Use Cases

It is pretty common to pass class methods down to other parts of the program as event listeners (e.g. in react apps). In this case it is almost always necessary to bind the this context to the current class instance if the method uses this in some form. A common mistake is to miss this binding for a given method. It is possible to add a typescript check for this mistake (by specifying the this-"parameter" on each class method) but one would have to do this for each method by hand, which is also easy to miss. By specifying it as a compiler flag, it would become very hard to make this mistake.

Examples

The following example doesn't raise a typescript error in current typescript but would do so with the proposed compiler flag:

class MyComponent {
  otherMethod() {}

  clickHandler() {
    // ...
    this.otherMethod();
  }

  registerListeners() {
     //...
     buttonElement.onclick = this.clickHandler;
  }
}

This example produces a "The 'this' types of each signature are incompatible." error in current typescript and is identical to the example above with the proposed compiler flag:

class MyComponent {
  otherMethod() {}

  clickHandler(this: MyComponent) {
    // ...
    this.otherMethod();
  }

  registerListeners() {
     //...
     buttonElement.onclick = this.clickHandler;
  }
}

Problems/Discussion Points

  • This doesn't work with element.addEventListener at the moment because the EventListener interface doesn't specify or complain about wrong this contexts (but I think it should, maybe this is another separate issue?)
  • This doesn't work with the semi-common pattern of overwriting the method instances with its bound versions in the constructor. Maybe the docs should recommend something like this instead (should be roughly equivalent besides the different name for the bound handler If I'm not mistaken)?
class MyComponent {
  // ...
  clickHandler() { /* ... */ }
  boundClickHandler = this.clickHandler.bind(this);
  // ...
}
  • Maybe this could be implemented as a tslint check, but it would require the user to type out the this context for each method separately, which is cumbersome.

Checklist

My suggestion meets these guidelines:

  • This wouldn't be a breaking change in existing TypeScript/JavaScript code
  • This wouldn't change the runtime behavior of existing JavaScript code
  • This could be implemented without emitting different JS based on the types of the expressions
  • This isn't a runtime feature (e.g. library functionality, non-ECMAScript syntax with JavaScript output, etc.)
  • This feature would agree with the rest of TypeScript's Design Goals.
@weswigham weswigham added Suggestion An idea for TypeScript In Discussion Not yet reached consensus labels Nov 15, 2018
@DanielRosenwasser DanielRosenwasser added the Add a Flag Any problem can be solved by flags, except for the problem of having too many flags label Nov 16, 2018
@DanielRosenwasser
Copy link
Member

Is this a duplicate of #7968 (also discussed at #6739)?

@flash1293
Copy link
Author

@DanielRosenwasser whoops, you are right - can be closed as duplicate.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Add a Flag Any problem can be solved by flags, except for the problem of having too many flags In Discussion Not yet reached consensus Suggestion An idea for TypeScript
Projects
None yet
Development

No branches or pull requests

3 participants