stage | start-date | release-date | release-versions | teams | prs | project-link | meta | |||||
---|---|---|---|---|---|---|---|---|---|---|---|---|
accepted |
2018-10-22 00:00:00 UTC |
|
|
|
This RFC introduces new router helpers that represent a decomposition of functionality of what we commonly use {{link-to}}
for. Below is a list of the new helpers:
This represents a super set of the functionality provided by Ember Router Helpers which has provided this RFC that confidence that a decomposition is possible.
This RFC does not deprecate {{link-to}}
or {{query-params}}
. These deprecations will come in the form of a deprecation RFC.
{{link-to}}
is the primary way for Ember applications to transition from route to route in your application. While this works for a lot of cases there are some use cases that are not well supported or supported at all by the framework. Below is an enumeration of cases that {{link-to}}
does not address.
We currently do not have a good solution for transitioning solely based on HTML anchors defined in the templating layer. For instance let say you are using Ember Intl to do internationalization for your application. Ember Intl uses the ICU message format for the actual translation strings and supports having HTML within the string. Now lets say you want to put a link in a translation string and have it work like {{link-to}}
works. In that case you either have to roll your own solution or use something like Ember-href-to. Another example where this would be useful is that links within markdown produced by addons like Ember-CLI-Showdown would just work. API's like RouterService#transitionTo
can transition an application using relative URLs and we have an opportunity to leverage this functionality to support this use case.
In 2.11 we moved LinkComponent
from private
to public
largely because there was no other way to modify the behavior of {{link-to}}
and it had effectively become de-facto public API. That being said, it is less than desirable to reopen
or extend
framework objects to gain access to the functionality to create some application specific primitive. For example Ember Bootsrap extends the LinkComponent
and then layers more functionality on top of it. Addons would be better served if they had access to more primitive functionality.
{{link-to}}
adds some convienent, yet not obvious, classes to the element. These classes are:
active
: applied to any{{link-to}}
that is on the "active" pathdisabled
: applied depending on the evaluation ofdisabled=someBool
loading
: applied if one or more of the models passed areundefined
ember-transitioning-in
: applied to links that are about to beactive
ember-transitioning-out
: applied to links that are about to be deactivated
The issue with these class names is that they are not declared anywhere in your templated and are provided by the LinkComponent
as classNameBindings
. This effectively creates a set of reserved class names that are highly prone to colissions in your typical application.
Furthermore, addons like ember-cli-active-link-wrapper and ember-bootstarp-nav-link do a ton of work arounds to get things like the .active
class to show up on wrapping elements instead of the element directly. This is a great example that shows we are missing some primitives.
Lastly, {{link-to}}
has very strange behavior when it comes to serializing query params. On a controller you declare the query params for a specific route. These query params can have defaults for them. For example if you have a controller that looks like:
// app/controllers/profile.js
import Controller from '@ember/controller';
export default Controller.extend({
queryParams: ['someBool'],
someBool: true,
})
and you to link to it like this:
In the DOM you will have an href
on the anchor that gets serializes as:
<a href="/profile?someBool=true" class="active ember-view">Profile</a>
Looking at a template you would have no idea that rendering the {{link-to}}
would result in the query params being serialized. From an implementation point of view, this is problematic as we are forced to lookup
the Route
and the associated Controller
to grab the query params. This can add a non-trivial amount of overhead during rendering, especially if you have many {{link-to}}
s on a route that link many different parts of your application. As a side-note, this is one of the things that needs to be dealt with if we are ever to kill controllers.
Since angle bracket invocation does not support positional params, {{link-to}}
has to adapt its public API.
{{link-to}}
has a lot of functionality, however this functionality does come at a cost for every instance of {{link-to}}
. This is not ideal especially if you're just using {{link-to}}
to generate a url that can be transitioned to. By providing fine grain control of the functionality, applications should see a performance boost.
Below is a detailed design of all of the template helpers.
The following helpers are to be used to construct a valid root-relative URL that will be used by the event dispatcher to perform a transition. These helpers do not pass the in memory model, meaning the model hook will always run for the route you are transitioning to.
Or
{{url-for}}
generates a root-relative URL as a string (which will include the application's rootUrl). When the link is clicked it will cause a transition to occur. See the Event Dispatcher Changes. It will not serialize the default query params on the controller.
Using the example above:
Positional Params
routeName
String: A fully-qualified name of this route, like"people.index"
model
...Object|Array|Number|String: Optionally pass an arbitrary amount of models or identifiers for each dynamic segment to be use for generation.
Named Params
models
Object|Array|Number|String: Same as the positional parameter. It is a compiler error if the positional params and the named param is used.queryParms
Object: Optionally pass key value pairs that will be serialized
Returns
- String: a root-relative URL as a string (which will include the application's
rootUrl
)
{{root-url}}
simply returns the value from Application.rootURL
. It can be used to prefix any href
values you wish to hard code.
Will result in the following for the default configuration:
<a href="/profile">Profile</a>
{{root-url}}
does not take any parameters.
The following helpers are all context dependent, not global. For instance you might have two copies of (is-active "posts")
in your app simultaneously where one is true
and one is false
, because you're in the middle of an animated transition, or because you're pre-rendering a route that hasn't been entered yet.
or
The arguments to {{is-active}}
have the same semantics as {{url-for}}
, however the return value is a boolean. This should provide the same logic that determines whether to put an active
class on a {{link-to}}
.
Using the example above:
Positional Params
routeName
String: A fully-qualified name of this route, like"people.index"
model
...Object|Array|Number|String: Optionally pass an arbitrary amount of models to use for generation
Named Params
models
Object|Array|Number|String: Same as the positional parameter. It is a compiler error if the positional params and the named param is used.queryParms
Object: Optionally pass key value pairs that will be used to determine if the route is active.
Returns
- Boolean: Determines if the route is active or not.
Or
The arguments to {{is-loading}}
have the same semantics as {{url-for}}
and {{is-active}}
, however if any of the model(s) passed to it are unresolved e.g. evaluate to undefined
the helper will return true
, otherwise the helper will return false
. This should provide the same logic that determines whether to put an loading
class on a {{link-to}}
.
Using the example above:
Positional Params
routeName
String: A fully-qualified name of this route, like"people.index"
model
...Object|Array|Number|String: Optionally pass an arbitrary amount of models to use for generation
Named Params
models
Object|Array|Number|String: Same as the positional parameter. It is a compiler error if the positional params and the named param is used.queryParms
Object: Optionally pass key value pairs that will be used to determine if the route is loading.
Returns
- Boolean: Determines if the route is loading or not.
Or
The arguments to {{is-transitioning-in}}
have the same semantics as all the other route state helpers, however {{is-transitioning-in}}
only returns true
when the route is going from an non-active to an active state. This should provide the same logic that determines whether to put an ember-transition-in
class on a {{link-to}}
.
Using the example above:
Positional Params
routeName
String: A fully-qualified name of this route, like"people.index"
model
...Object|Array|Number|String: Optionally pass an arbitrary amount of models to use for generation
Named Params
models
Object|Array|Number|String: Same as the positional parameter. It is a compiler error if the positional params and the named param is used.queryParms
Object: Optionally pass key value pairs that will be used to determine if the route is transitioning in.
Returns
- Boolean: Determines if the route is transitioning in.
Or
{{is-transitioning-out}}
is just the inverse of {{is-transitioning-in}}
.
Using the example above:
Positional Params
routeName
String: A fully-qualified name of this route, like"people.index"
model
...Object|Array|Number|String: Optionally pass an arbitrary amount of models to use for generation
Named Params
models
Object|Array|Number|String: Same as the positional parameter. It is a compiler error if the positional params and the named param is used.queryParms
Object: Optionally pass key value pairs that will be used to determine if the route is transitioning out.
Returns
- Boolean: Determines if the route is transitioning out.
In the past, only HTMLAnchorElement
s that were produced by {{link-to}}
s would produce a transition when a user clicked on them. This RFC changes to the global EventDispatcher
to allow for any HTMLAnchorElement
with a valid root relative href
to cause a transition. This will allow for us to not only allows us to support use cases like the ones described in the motivation, it makes teaching easier since people who know HTML don't need know an Ember specific API to participate in routing transitions.
While the vast majority of the time developers want root relative URLs to cause a transition there are cases where you want root relative urls to cause a normal HTTP navigation. In the router map you can define wildcard / globbing that makes this problematic as any root relative url can be catched by a wildcard route. To solve this issue this RFC proposes expanding the route options to allow for a black list of urls that are allowed to cause a normal HTTP navigation.
Router.map(function() {
this.route('not-found', { path: '/*path', blacklist: ['/contact-us', '/order/:order_id'] });
});
When an event comes into the EventDispatcher
we will cross check the blacklist to see if the event should be let through to the browser or if it should be handled internally.
This RFC introduces the notion of an attribution
to the Transition
. The TransitionAttribution
is a read-only object that has 2 fields event
and source
.
interface TransitionAttribution {
readonly event: Maybe<Event>;
readonly source: unknown
}
interface Transition {
readonly attribution: TransitionAttribution;
}
On initial render event
and source
will be null
. On subsequent transitions, the event
will be the DOM event that caused the transition and element
will be populated with HTMLElement
that the user interacted with to cause the transition. See Appendix A for example usage. In the event that the transition occurs programmatically through an API like replaceWith
or transitionTo
the event
will be null
but can be populated by the caller.
In cases where you need to programatically transition with transitionTo
or replaceWith
we will allow for you to pass your own TransitionAttribution
. See Appendix B for an example.
interface Options {
queryParams?: Dict<string|number>,
attribution?: TransitionAttribution;
}
interface Router /* Route, RouterService */ {
//...
transitionTo(routeName: string, models?: string|number|object, options?: Options): Transition;
replaceWith(routeName: string, models?: string|number|object, options?: Options): Transition;
}
The attribution
in a Transition
is guaranteed to be carried through the completion of the route transition. This includes abort
s, redirect
s and retry
s of the transition. The attribution
field is readonlu and the TransitionAttribution
is readonly and frozen.
Since this RFC does not deprecate {{link-to}}
you can continue to use it. That being said {{link-to}}
has static semantics therefore we can write a codemod using Ember Template Recast to migrate the code. Below are numerous before and after examples of how the codemod would migrate. It's important to note that the behavior of the application will change if you are relying on the passing of the in-memory model. Because of this the codemod would need different levels of conversion.
Before:
After:
Before:
After:
Before:
After:
One of the trickier parts about this migration is knowing how the autogenerated CSS classes are being used. Because of this, adding the route state helpers must explicitly be turned on in the codemod. For instance if you are making heavy use of the .active
class, you will be suited best by turning pass the codemod the correct configuration to do a transform like the following:
Before
After
If you were to transform all {{link-to}}
s verbatim in terms of functionality this would be the result.
Before
After
As the kitchen sink example shows, {{link-to}}
is packed with functionality. While this convienent, it comes with a cost per {{link-to}}
and is the reason why addons like Ember-href-to were created. In reality the vast majority of applications only need a subset of this functionality and only in rare cases need things like the transition and loading states.
In many ways this vastly simplifies Ember's approach to linking within the app. It removes the requirement for a proprietary API and instead embraces the power of URLs.
In the cases where you do need to do more complicated things like pass in memory models to a route, things should feel very similar to {{link-to}}
as they have the exact same signature. In the case of query param serialization, I believe we are actually aligning a mental model as to how URL generation should work.
This RFC expands the surface area of the templating layer by exposing the primitives that make up {{link-to}}
. This may cause confusion of choosing between using simple basic anchor tags, {{url-for}}
and {{link-to}}
, however I believe that each one of the these APIs are solving a real problem that we have in Ember today.
By proxy this may cause people to encapsulate all of these primitives into a single component and thus creating a user-land version of {{link-to}}
. This could be seen as a framework misstep if the majority of applications end up depending on the addon.
We could just start deprecating and removing functionality from {{link-to}}
itself. That being said, it is hard to understand how much of the community is reliant on certain feature of {{link-to}}
. This also doesn't help with usecases like the i18n and markdown use cases.
TBD?
// app/utils/tracking.js
const TRACKING_DATA = new WeakMap();
export default TRACKING_DATA;
// app/components/track-link.js
import Component from '@ember/component';
import TRACKING_DATA from '../utils/tracking';
export default Component.extend({
tagName: 'a',
attributeBindings: ['href'],
didInsertElement() {
TRACKING_DATA.set(this.element, this.contextName);
}
})
// app/routes/application.js
import Route from '@ember/routing/route';
import { inject as service } from '@ember/service';
import TRACKING_DATA from '../utils/tracking';
export default Route.extend({
router: service('router'),
init() {
this._super(...arguments);
this.router.on('routeDidChange', transition => {
let { source, event } = trasition.attribution;
let trackingInfo = {
cause: event.type,
contextName: null
};
if (TRACKING_DATA.has(trasition.source)) {
trackingInfo.contextName = TRACKING_DATA.get(source);
}
ga.send('pageView', {
from: transition.from ? transition.from.name : 'initial',
to: transition.to.name,
attribution: trackingInfo
});
});
}
})
// app/utils/tracking.js
const TRACKING_DATA = new WeakMap();
export default TRACKING_DATA;
// app/routes/profile.js
import Route from '@ember/routing/route';
import { inject as service } from '@ember/service';
import TRACKING_DATA from '../utils/tracking';
export default Route.extend({
store: service('store'),
actions: {
changeName(name) {
this.model.set('name', name);
},
changeAge(age) {
this.model.set('age', age);
},
submit(e) {
if (isValid(this.model)) {
let attribution = {
event: e,
source: e.target,
};
TRACKING_DATA.set(e.target, 'profile.submit');
this.model.save().then(() => {
this.transitionTo('profile.success', { attribution });
}, () => {
alert('Issue saving... please try again.');
});
} else {
alert('Data is not valid!');
}
}
}
})
// app/routes/application.js
import Route from '@ember/routing/route';
import { inject as service } from '@ember/service';
import TRACKING_DATA from '../utils/tracking';
export default Route.extend({
router: service('router'),
init() {
this._super(...arguments);
this.router.on('routeDidChange', transition => {
let { source, event } = trasition.attribution;
let trackingInfo = {
cause: event.type,
contextName: null
};
if (TRACKING_DATA.has(trasition.source)) {
trackingInfo.contextName = TRACKING_DATA.get(source);
}
ga.send('pageView', {
from: transition.from ? transition.from.name : 'initial',
to: transition.to.name,
attribution: trackingInfo
});
});
}
})