-
-
Notifications
You must be signed in to change notification settings - Fork 10.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Get helper improvements #5993
Comments
refs TryGhost#5993 - Don't merge the result set with the existing template data. - If available, return `meta.pagination` as the second blockParam
refs TryGhost#5993 - deps: jsonpath@0.2.0 - adds `resolvePaths` method - supports handlebars style arrays with `.[]` - supports shorthand post.tags and post.author for common usecases - adds more tests & improves existing ones
Regarding
So, basically, taking into account singular/plural form of a resource (post, posts) and returning results accordingly. This also eliminates some extra code for singular post rendering, like |
Re: It also would be nice to support E.g. |
@ErisDS I'm seeing this on the list, but is nesting get helpers something you guys are currently working on? For example: {{#get "tags"}}
{{#foreach tags}}
<h1>{{name}}</h1>
{{#get "posts" filter="tags:{{id}}"}}
<ul class="posts">
{{#foreach posts}}
<li class="post"><a href="{{url}}">{{title}}</a></li>
{{/foreach}}
</ul>
{{/get}}
{{/foreach}}
{{/get}} |
Hi @jerrylopez, nesting the get helper is not something that's currently in our plans. Not being able to nest it is a fundamental limitation of asynchronous helpers. Asynchronous helpers are something which are added as an extension by express-hbs due to lack of support in handlebars. This means it's entirely a bit of a hack. The intention of the get helper feature is to give users an easy way to pull additional data into a template but in a deliberately limited way as all of this is being done server side. Nesting get helpers as in your example could cause a page speed issue unless you're heavily caching your blog. This level of customisation is probably better done client side, using AJAX and the API directly. |
Having played with the get helper some more since it got merged, there are some details which need addressing as well as some updates needed to make it work nicely with the filter parameter.
I've kept everything in one issue because I hope that'll help to make it all make sense - the blockParams and data naming stuff is highly confusing.
1. Bugfix: Data is getting merged
The data fetched by the helper is being incorrectly merged with the existing data for the template. This means that in certain cases one posts object will get overwritten by another. And you won't get the data that you expect when inside the
{{#get}}
helper.I remember I was attempting to resolve an issue when I added the
_.merge()
line, but what it was escapes me and I'm very sure it's the wrong thing to do, so this needs to be removed.2. Addition: Pagination block param
Although I added support for blockParams I only did this for the main result data, and not the pagination data. What this means is that you can 'name' the list of posts output by the get helper like this:
{{#get "posts" filter="featured:true" as |featured|}}
- the|featured|
part is what handlebars calls a block param (see the rfc).I think it makes sense to be able to do
{{#get "posts" filter="featured:true" as |featured pagination|}}
which is easy to make work but not currently possible with my implementation. Note that without doing this, access to pagination looks like `{{meta.pagination}} - see the part about data naming inconsistencies in 4.3. Addition: Get helper path support for filters
The get helper will currently work with any kind of filter query, for example
{{#get "posts" filter="featured:true"}}
or{{#get "posts" filter="tags:photo"}}
which makes it possible to do queries to fetch extra data for your theme. However, there is no way currently to build dynamic queries based on data you've already got:E.g.
{{#get "posts" filter="tags:[{{post.tags}}]"}}
might reasonably fetch the posts with the same tags as the current post. Or even{{#get "posts" filter="tags:[{{post.tags}}]+id:-{{post.id}}"}}
could fetch posts with the same tags as the current post which also aren't the current post. To make this work, we need to explicitly add path resolution to the get helper, so that it will process handlebars-like strings out of the filter query and grab the requested data in a useful format.Handlebars has its own path syntax, supporting dot syntax like
post.author.name
and an array syntax likepost.tags.[0].slug
. Very similar to this is jsonpath - a library which provides an xPath-like syntax for grabbing data from a JSON object.JSONPath syntax is very slightly different to handlebars path syntax in that its array syntax doesn't require a dot, perhaps more like you would expect:
post.tags[0].slug
and it can understand wildcards in arrays, sopost.tags[*].slug
would return an array containing the slug of each tag associated with post.It's incredibly powerful, and used with the get helper it makes for powerful tool. I propose that we use JSONPath for parsing paths, with two slight modifications:
{{post.tags}}
for{{post.tags[*].slug}}
and{{post.author}}
for{{post.author.slug}}
to cover common use cases.4. Discussion: Get helper data naming vs basic theming
In the normal theme world, on a post listing page you do
{{#foreach posts}}
and on a post page you do{{#post}}
. Although the API returns the same plural form for multiple and single items, we specially preprocess the data before the theme gets it to make this more intuitive.The get helper, currently, doesn't do this preprocessing for you. This results in two inconsistencies:
Inconsistency 1 - posts or post?
Both
{{#get "posts" filter="featured:true"}}
and{{#get slug="my-post"}}
do two different queries to the API - the first is a browse and the latter is a read, so the first returns multiple results and pagination, the second returns a single result. However both result in a newposts
(plural) array of object(s), so the way you handle the data inside the{{#get}}
helper is the same for both:I'm not sure if it's confusing that the output from the second one is also 'posts' or if it is better that the two are consistent as there is little to tell between them. Note that if you wanted you could name the second one 'post' yourself:
This works because, weirdly
{{#foreach posts}}{{title}}{{/foreach}}
and{{#posts}}{{title}}{{/posts}}
both work although they behave slightly differently.An alternative would be to duplicate the object with both the plural and single name, but this seems like overkill.
Inconsistency 2 - pagination
When returned to a theme currently, pagination is a top level key so you do
{{pagination}}
. When returned from the API to the get helper, it is{{meta.pagination}}
. I think it unlikely that there will be much need for creating pagination inside of{{#get}}
queries, but this could trip people up. We could move it to a top level key, but this would be inconsistent with API documentation.Providing the second point in this issue is solved, it would be possible to solve this again by giving the data a name yourself:
{{#get "posts" filter="featured:true" as |featured pagination|}}
Ultimately, these are both, I think, quite edge case issues that will trip a few people up. This can be mitigated by writing the get helper documentation to encourage the use of block params to name the data that is returned, so that theme developers have absolute control.
We could also duplicate the data under different keys, or apply the same pre-processing that we do for
themes, the question is - which is least confusing?
I don't think there is an easy answer, so it may make sense to move the
{{#get}}
helper behind the Public API labs flag for the first release, get some feedback, make any necessary changes and then take the flag off in the following release.Todo list:
The text was updated successfully, but these errors were encountered: