I see that ember has a very nice mechanism for wrapping content in a component using the {{yield}}
mechanism documented here.
So, to use the example in the documentation, I can have a blog-post
component template defined like so:
<script type="text/x-handlebars" id="components/blog-post"> <h1>{{title}}</h1> <div class="body">{{yield}}</div> </script>
I can then embed blog-post
into any other template using the form:
{{#blog-post title=title}} <p class="author">by {{author}}</p> {{body}} {{/blog-post}}
My question is, can I specify two different {{yield}}
outlets in the components template?
Something like this is possible via Named Outlets in Ember.Route#renderTemplate
like so:
Handlebars:
<div class="toolbar">{{outlet toolbar}}</div> <div class="sidebar">{{outlet sidebar}}</div>
JavaScript:
App.PostsRoute = Ember.Route.extend({ renderTemplate: function() { this.render({ outlet: 'sidebar' }); } });
I'm not sure I can take this path for a component which will not know what route's template would be rendering it.
EDIT 1:
For the sake of clarity, I'm trying to implement the Android Swipe for Action Pattern as an Ember component.
So, I'd like users of this component to be able to specify two different templates:
I want to make this into a component, because quite a lot of javascript goes into handling the touch(start/move/end) events, while still managing smooth touch based scrolling of the list. Users would supply the two templates and this component would manage handling of touch events and necessary animations.
I've managed to get the component working in the block form, where the block's contents are treated like (1). The second template (2) is specified through a parameter (actionPartial
below) which is the name of a partial template for the actions:
Component Handlebars Template: sfa-item.handlebars
<div {{bind-attr class=":sfa-item-actions shouldRevealActions:show" }}> {{partial actionPartial}} </div> <div {{bind-attr class=":sfa-item-details isDragging:dragging shouldRevealActions:moveout"}}> {{yield}} </div>
Calling Handlebars Template:
{{#each response in controller}} <div class="list-group-item sf-mr-item"> {{#sfa-item actionPartial="mr-item-action"}} <h5>{{response.name}}</h5> {{/sfa-item}} </div> {{/each}}
Where the mr-item-action
handlebars is defined like so:
mr-item-action.handlebars:
<div class="sf-mr-item-action"> <button class="btn btn-lg btn-primary" {{action 'sfaClickedAction'}}>Edit</button> <button class="btn btn-lg btn-primary">Delete</button> </div>
Problem is, actions from the user supplied partial, sfaClickedAction
above, are not bubbled up from the component. A fact which is mentioned in the docs in para 4.
So, now I do not know how a user could capture actions that he defined in the supplied actions template. A component cannot catch those actions because it doesn't know about them either.
EDIT 2
I sprung a follow up question here
The content of the tag is also referred to as the block. The {{yield}} syntax yields to the block once the block is passed into the component.
Ember components are used to turn markup text and styles into reusable content. Components consist of two parts: a JavaScript component file that defines behavior, and its accompanying Handlebars template that defines the markup for the component's UI. Components must have at least one dash in their name.
This blog post describes the most elegant solution for Ember 1.10+: https://coderwall.com/p/qkk2zq/components-with-structured-markup-in-ember-js-v1-10
In your component you pass yield names into {{yield}}
s:
<header> {{yield "header"}} </header> <div class="body"> {{yield "body"}} </div> <footer> {{yield "footer"}} </footer>
When you invoke your component, you accept the yield name as a block param... and use an esleif chain!
{{#my-comp as |section|}} {{#if (eq section "header")}} My header {{else if (eq section "body")}} My body {{else if (eq section "footer")}} My footer {{/if}} {{/my-comp}}
PS eq
is a subexpression helper from the must-have ember-truth-helpers addon.
PPS Relevant RFC: proposal, discussion.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With