Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/buschtoens/ember-route-task-helper
Easily access tasks defined on a route in your templates
https://github.com/buschtoens/ember-route-task-helper
ember ember-addon ember-concurrency ember-helpers
Last synced: 3 months ago
JSON representation
Easily access tasks defined on a route in your templates
- Host: GitHub
- URL: https://github.com/buschtoens/ember-route-task-helper
- Owner: buschtoens
- License: mit
- Created: 2017-08-03T20:52:26.000Z (over 7 years ago)
- Default Branch: master
- Last Pushed: 2023-12-15T20:24:19.000Z (about 1 year ago)
- Last Synced: 2024-10-29T20:02:06.449Z (3 months ago)
- Topics: ember, ember-addon, ember-concurrency, ember-helpers
- Language: JavaScript
- Homepage: https://buschtoens.github.io/ember-route-task-helper/
- Size: 1.44 MB
- Stars: 27
- Watchers: 3
- Forks: 4
- Open Issues: 26
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE.md
Awesome Lists containing this project
README
# ember-route-task-helper
[![Build Status](https://travis-ci.org/buschtoens/ember-route-task-helper.svg)](https://travis-ci.org/buschtoens/ember-route-task-helper)
[![npm version](https://badge.fury.io/js/ember-route-task-helper.svg)](http://badge.fury.io/js/ember-route-task-helper)
[![Download Total](https://img.shields.io/npm/dt/ember-route-task-helper.svg)](http://badge.fury.io/js/ember-route-task-helper)
[![Ember Observer Score](https://emberobserver.com/badges/ember-route-task-helper.svg)](https://emberobserver.com/addons/ember-route-task-helper)
[![code style: prettier](https://img.shields.io/badge/code_style-prettier-ff69b4.svg)](https://github.com/prettier/prettier)
[![Dependabot enabled](https://img.shields.io/badge/dependabot-enabled-blue.svg?logo=dependabot)](https://dependabot.com/)
[![dependencies Status](https://david-dm.org/buschtoens/ember-route-task-helper/status.svg)](https://david-dm.org/buschtoens/ember-route-task-helper)
[![devDependencies Status](https://david-dm.org/buschtoens/ember-route-task-helper/dev-status.svg)](https://david-dm.org/buschtoens/ember-route-task-helper?type=dev)The `route-task` template helper allows you to easily access tasks defined on a
route in the currently active route hierarchy. Essentially this addon is just
like [**ember-route-action-helper**][ember-route-action-helper] but for
[**ember-concurrency**][ember-concurrency] tasks.## Installation
```
ember install ember-route-task-helper
```This addon will work on Ember versions `2.4.x` and up only, due to use of the
new `RouterService`. If your Ember version does not natively support it yet, you
need to install the polyfill:```
ember install ember-router-service-polyfill
```Of course, you need to have [**ember-concurrency**][ember-concurrency]
installed. If you haven't already, run this command first:```
ember install ember-concurrency
```Minimum required version is `0.6.x`.
If you want to use this addon inside an engine, you'll need to have an
`EngineRouterService`. You can use my PoC linked in
[ember-engines#587][engine-router-service] until it's released officially as
part of ember-engines.[engine-router-service]: https://github.com/ember-engines/ember-engines/issues/587#issue-355568777
## Usage
### Template Helper: `(route-task taskName ...curryArguments)`
Wherever in a template you would access a task by its name, replace it with
`(route-task "taskName")` and move that task to a route. For instance:```hbs
{{task-button
task=myTask
}}{{!-- now becomes --}}
{{task-button
task=(route-task "myTask")
}}
```Notice the quotes around `"myTask"`.
You can also curry your tasks:
```hbs
{{task-button
task=(task myTask "Freddie" "Morecurry")
}}{{!-- now becomes --}}
{{task-button
task=(route-task "myTask" "Freddie" "Morecurry")
}}
```#### Exemplary Migration
Let's start with a traditional task defined on a component.
```js
// app/components/delete-user.js
import Component from '@ember/component';
import { get } from '@ember/object';
import { task, timeout } from 'ember-concurrency';export default Component.extends({
/**
* A User record.
* @type {DS.Model}
*/
user: null,/**
* Deletes the user after a timeout of 5 seconds.
* @type {Task}
*/
deleteUser: task(function*() {
timeout(5000); // give the user time to think about it
yield get(this, 'user').destroyRecord();
}).drop()
});
``````hbs
{{!-- app/templates/components/delete-user.hbs --}}{{#if deleteUser.isIdle}}
Delete {{user.name}}
{{else}}
Deleting {{user.name}} in 5 seconds. You can still abort.
Nah, spare their life.
{{/if}}
```And now we'll move it to a route, shall we?
```js
// app/routes/user.js
import Route from '@ember/routing/route';
import { get } from '@ember/object';
import { task, timeout } from 'ember-concurrency';export default Route.extends({
/**
* Deletes the current model after a timeout of 5 seconds.
* @type {Task}
*/
deleteUser: task(function*() {
timeout(5000); // give the user time to think about it
const user = this.modelFor(this.routeName);
yield user.destroyRecord();
}).drop()
});
``````hbs
{{!-- app/templates/user.hbs --}}{{#if (get (route-task "deleteUser") "isIdle")}}
Delete {{model.name}}
{{else}}
Deleting {{model.name}} in 5 seconds. You can still abort.
Nah, spare their life.
{{/if}}
```### Currying
```js
// app/routes/user.js
import Route from '@ember/routing/route';
import { get } from '@ember/object';
import { task, timeout } from 'ember-concurrency';export default Route.extends({
/**
* Deletes the current model after a timeout of 5 seconds.
* @type {Task}
*/
deleteUser: task(function*(user) {
timeout(5000); // give the user time to think about it
yield user.destroyRecord();
}).drop()
});
``````hbs
{{!-- app/templates/user.hbs --}}{{task-button
task=(route-task "deleteUser" model)
idleLabel=(concat "Delete " model.name)
runningLabel="Cancel deletion"
}}
```### Notes on Syntax
I personally dislike repeating `(route-task)` a bunch of times in my templates or even worse having to use the `(get)` helper to derive state from a task. You can avoid that by either only passing a task to a component (as shown in the `{{task-button}}` example above) or by using the `{{with}}` helper:
```hbs
{{#with (route-task "deleteUser" model) as |deleteUser|}}
{{#if deleteUser.isIdle}}
Delete {{model.name}}
{{else}}
Deleting {{model.name}} in 5 seconds. You can still abort.
Nah, spare their life.
{{/if}}
{{/with}}
```## Util: `routeTask(context, taskName, ...curryArguments)`
There also is a `routeTask` util, that's really similar to [**ember-invoke-action**][ember-invoke-action] and might come in handy for JS heavy components.
```js
// app/components/delete-user.js
import Component from '@ember/component';
import { get } from '@ember/object';
import { routeTask } from 'ember-route-task-helper';export default Component.extends({
/**
* A User record.
* @type {DS.Model}
*/
user: null,/**
* Deletes the user after a timeout of 5 seconds.
*/
click() {
const user = get(this, 'user');
routeTask(this, 'deleteUser').perform(user);
}
});
```### Currying
As with the `(route-task)` helper, you can curry the task with as many arguments as you like. So the above is interchangeable with:
```js
click() {
const user = get(this, 'user');
routeTask(this, 'deleteUser', user).perform();
}
```### `routeTaskFromRouter(router, taskName, ...curryArguments)`
Internally `routeTask` performs a lookup for the `Router` everytime you call it. If you already happen to have the router instance available in your current scope, you could also pass it directly to skip the lookup:
```js
// app/components/delete-user.js
import Component from '@ember/component';
import { get, computed } from '@ember/object';
import { getOwner } from '@ember/application';
import { routeTaskFromRouter } from 'ember-route-task-helper';export default Component.extends({
/**
* A User record.
* @type {DS.Model}
*/
user: null,/**
* The app's router instance.
* @type {Ember.Router}
*/
router: computed(function() {
return getOwner(this).lookup('main:router');
}).readOnly(),/**
* Deletes the user after a timeout of 5 seconds.
*/
click() {
const router = get(this, 'router');
const user = get(this, 'user');
routeTaskFromRouter(router, 'deleteUser').perform(user);
}
});
```### Notes on DDAU
In my opinion, using `routeTask` in components _generally_ isn't a good design pattern. I would much rather prefer to explicitly pass the route task as an attribute:
```hbs
{{my-component taskName=(route-task "taskName")}}
```Just by looking at the component invocation in the template, the user should be able to judge what's going in and what's coming out of a component (DDAU). This way components remain completely agnostic and make no assumptions about the environment they are invoked in.
Calling `routeTask` inside a component is really non-transparent and promotes an unhealthy invisible entanglement.
On the other hand, you can already call `(route-task)` in the component's template.
I've implemented it for feature parity. But just because it's there, doesn't mean you _have_ to use it. But don't let me stop you. :stuck_out_tongue_winking_eye:
## Contributing
I sincerely hope this addon serves you well. Should you encounter a bug, have a great idea or just a question, please do [open an issue][new-issue] and let me know. Even better yet, submit a PR yourself! :blush:
This addon is using the [Prettier code formatter][prettier]. It's embedded as a fixable eslint rule. If you're editor is set up to fix eslint errors on save, you're code is auto formatted. If not, please make sure you're not getting any linter errors.
## Attribution
The original idea for this addon was brought up in [machty/ember-concurrency#89][issue-89] by [@Luiz-N][luiz-n]. This addon is in many ways a straight copy and paste from [ember-route-action-helper][ember-route-action-helper] by [@DockYard][dockyard].
A huge **thank you** to goes out to [@machty][machty] for developing [ember-concurrency][ember-concurrency] in the first place. :heart:
[ember-concurrency]: https://github.com/machty/ember-concurrency
[ember-route-action-helper]: https://github.com/DockYard/ember-route-action-helper
[ember-invoke-action]: https://github.com/martndemus/ember-invoke-action
[issue-89]: https://github.com/machty/ember-concurrency/issues/89
[luiz-n]: https://github.com/Luiz-N
[dockyard]: https://github.com/DockYard
[machty]: https://github.com/machty
[new-issue]: https://github.com/buschtoens/ember-route-task-helper/issues/new
[prettier]: https://prettier.io/