ember-render-helpers
Version:
Helpers that complement @ember/render-modifiers
150 lines (108 loc) • 4.82 kB
Markdown
[](https://github.com/buschtoens/ember-render-helpers/actions/workflows/ci.yml)
# ember-render-helpers
Use the `{{did-insert}}`, `{{did-update}}`, `{{will-destroy}}` modifiers from [`/render-modifiers`][render-modifiers] as template helpers.
The original idea came from [this Pre-RFC][pre-rfc].
[render-modifiers]: https://github.com/emberjs/ember-render-modifiers#readme
[pre-rfc]: https://github.com/emberjs/rfcs/issues/484
## Installation
```
ember install ember-render-helpers
```
<details>
<summary>Use Glint or <code><template></code> tag? ✨</summary>
- Update your template registry to extend this addon's. Check the [Glint documentation](https://typed-ember.gitbook.io/glint/using-glint/ember/using-addons#using-glint-enabled-addons) for more information.
```ts
import '/environment-ember-loose';
import type EmberRenderHelpersRegistry from 'ember-render-helpers/template-registry';
declare module '/environment-ember-loose/registry' {
export default interface Registry extends EmberRenderHelpersRegistry, /* other addon registries */ {
// local entries
}
}
```
- In a `<template>` tag, use the named import to consume the helpers.
```ts
import { action } from '/object';
import Component from '/component';
import { didInsertHelper } from 'ember-render-helpers';
export default class HelloComponent extends Component {
sayHello() {
console.log('Hello!');
}
<template>
{{didInsertHelper this.sayHello}}
</template>
}
```
</details>
## Usage
`ember-render-helpers` provides 3 helpers. To avoid name conflicts with `/render-modifiers`, these helpers are suffixed with `-helper`.
- `{{did-insert-helper}}`
- `{{did-update-helper}}`
- `{{will-destroy-helper}}`
All helpers expect a callback function as the 1st positional argument. You can pass parameters to this callback function as subsequent positional arguments, as named arguments, or using both.
```hbs
{{did-insert-helper this.sayHello "Zoey"}}
```
```ts
import Component from '/component';
import type { NamedParameters, PositionalParameters } from 'ember-render-helpers';
export default class HelloComponent extends Component {
sayHello(positional: PositionalParameters, _named: NamedParameters) {
const name = positional[0] as string;
console.log(`Hello, ${name}!`);
}
}
```
### Example
Clicking the `Toggle` button will toggle the `isVisible` flag. When it switches
to `true`, `onDidInsert` will be called, because the `{{did-insert-helper}}` helper is
inserted. When it switches to `false`, `onWillDestroy` will be called, because
the `{{will-destroy-helper}}` helper is removed.
Clicking the `Random` button will set `randomValue` to a new random value. Every
time this happens, while `isVisible` is `true`, `onDidUpdate` will be called,
because one of the parameters passed to the `{{did-update-helper}}` helper was updated.
Clicking the `Random` button _does not_ cause `{{did-insert-helper}}` or
`{{will-destroy-helper}}` to call `onDidInsert` and `onWillDestroy`, since these
helpers are not triggered by parameter updates.
```hbs
{{#if this.isVisible}}
{{did-insert-helper this.onDidInsert 1 2 3 this.randomValue foo="bar" qux="baz"}}
{{will-destroy-helper this.onWillDestroy 1 2 3 this.randomValue foo="bar" qux="baz"}}
{{did-update-helper this.onDidUpdate 1 2 3 this.randomValue foo="bar" qux="baz"}}
{{/if}}
<button {{on "click" this.toggleVisibility}}>Toggle</button>
<button {{on "click" this.rollTheDice}}>Random</button>
```
```ts
import { action } from '/object';
import Component from '/component';
import { tracked } from '/tracking';
export default class extends Component {
isVisible = false;
randomValue?: number;
toggleVisibility() {
this.isVisible = !this.isVisible;
}
rollTheDice() {
this.randomValue = Math.random();
}
onDidInsert(positional: unknown[], named: Record<string, unknown>) {
console.log({ positional, named });
// => { positional: [1, 2, 3, 0.1337...], named: { foo: 'bar', qux: 'baz' } }
}
onWillDestroy(positional: unknown[], named: Record<string, unknown>) {
console.log({ positional, named });
// => { positional: [1, 2, 3, 0.1337...], named: { foo: 'bar', qux: 'baz' } }
}
onDidUpdate(positional: unknown[], named: Record<string, unknown>) {
console.log({ positional, named });
// => { positional: [1, 2, 3, 0.1337...], named: { foo: 'bar', qux: 'baz' } }
}
}
```