✅ The extends: 'recommended'
property in a configuration file enables this rule.
This rule aides in the migration path for emberjs/rfcs#308.
Currently, the way to access properties on a components class is {{greeting}}
from a template. This works because the component class is one of the objects
we resolve against during the evaluation of the expression.
The first problem with this approach is that the {{greeting}}
syntax is
ambiguous, as it could be referring to a local variable (block param), a helper
with no arguments, a closed over component, or a property on the component
class.
Consider the following example where the ambiguity can cause issues:
You have a component class that looks like the following component and template:
import Component from '@ember/component';
import computed from '@ember/computed';
export default Component.extend({
formatName: computed('firstName', 'lastName', function() {
return `${this.firstName} ${this.lastName}`;
});
});
Given { firstName: 'Chad', lastName: 'Hietala' }
, Ember will render the
following:
<h1>Hello Chad Hietala!</h1>
Now some time goes on and someone adds a formatName
helper at
app/helpers/formatName.js
that looks like the following:
export default function formatName([firstName, lastName]) {
return `${firstName} ${lastName}`;
}
Due to the fact that helpers take precedence over property lookups, our
{{formatName}}
now resolves to a helper. When the helper runs it doesn't have
any arguments so our template now renders the following:
<h1>Hello !</h1>
This can be a refactoring hazard and can often lead to confusion for readers of
the template. Upon encountering {{greeting}}
in a component's template, the
reader has to check all of these places: first, you need to scan the
surrounding lines for block params with that name; next, you check in the
helpers folder to see if there is a helper with that name (it could also be
coming from an addon!); finally, you check the component's JavaScript class to
look for a (computed) property.
Like
RFC#0276
made argument usage explicit through the @
prefix, the this
prefix will
resolve the ambiguity and greatly improve clarity, especially in big projects
with a lot of files (and uses a lot of addons).
As an aside, the ambiguity that causes confusion for human readers is also a
problem for the compiler. While it is not the main goal of this proposal,
resolving this ambiguity also helps the rendering system. Currently, the
"runtime" template compiler has to perform a helper lookup for every
{{greeting}}
in each template. It will be able to skip this resolution
process and perform other optimizations (such as reusing the internal
reference
object and caches) with this addition.
Furthermore, by enforcing the this
prefix, tooling like the Ember Language
Server does not need to
know about fallback resolution rules. This makes common features like "Go To
Definition"
much easier to implement since we have semantics that mean "property on class".
- use ember-no-implicit-this-codemod
- upgrade to Glimmer components, which don't allow ambiguous access
- classic components have auto-reflection, and can use
this.myArgName
orthis.args.myArgNme
or@myArgName
interchangeably
- classic components have auto-reflection, and can use
The following values are valid configuration:
- boolean -
true
to enable /false
to disable - object -- An object with the following keys:
allow
-- An array of component / helper names for that may be called without arguments (string or regular expression)