Parsley Index

This index collects all Parsley syntax and methods.

_index

When looping through a content model with an each loop, you can use variable_name._index to get the current count of how many times you have run through the each loop. _index begins counting from 0. To start counting at 1 use _num instead.

<div class="row"> 
{{ each team_members as member }}
{{ if {member._index} % 3 == 0 }} (** Checks for every third loop, can be used to end and stop rows when using fixed width columns in a grid  **)
</div><div class="row">
{{ end-if }}
<div class="column">
<p>{{ member.name }}</p>
<p>{{ member._index }}</p>
</div>
{{ end-each }}
</div>

_length

When looping through a content model with an each loop, you can call {{ variable_name._length }} to get the total number of times the loop will run.

<div class="article-tags">
{{ each article_tag as tag }}
<p>{{ tag.name }}{{ if {tag._num} != {tag._length} }}, {{ end-if }}</p>
{{ end-each }}
</div>
(** As long as the loop's num does not equal its length a comma will be appended to the list of tags **)

_num

When looping through a content model with an each loop, you can use variable_name._num to get the current count of how many times you have run through the each loop. _num begins counting from 1. To start counting at 0 use _index instead.

<div class="article-tags">
{{ each article_tag as tag }}
<p>{{if {tag._num} == 1 }}This is the first item{{ end-if }}</p>
{{ end-each }}
</div>
(** As long as the loop's num does not equal its length a comma will be appended to the list of tags **)

activate_links()

Converts plain text URLS into anchor tags URL. Useful for content where text comes in with HTML stripped.

{{ page.textarea.activate_links() }}

api.dribbble

{{ each api.dribbble.getShots(USER_NAME) as shot }}
    <img src="{{ shot.image_teaser_url }}" alt="{{ shot.title} }" />
{{ end-each }}

breadcrumbs

The breadcrumbs call outputs an automated breadcrumb trail of links, listing the parent pages assigned to the current page. If there are no available links, then an empty span is outputted.

{{ breadcrumbs }}

Example Output:
<span itemprop="breadcrumb">
    <span class="crumb"><a href="/blog/">Blog</a></span> 
    <span class="sep">»</span>
    <span class="crumb"><a href="/blog/example-category/">Example Category</a></span>
    <span class="sep">»</span>
    <span class="crumb">Example Post</span>
</span>

text_breadcrumbs

The text_breadcrumbs call outputs an automated breadcrumb trail as a text string, listing the parent pages assigned to the current page in a comma-separated format. If there are no available pages, then an empty string is returned.

{{ text_breadcrumbs }}

Example Output:
Blog, Example Category, Example Post

clippings (see Globals)

Comments (** **)

Use these comments to prevent Parsley or other code from processing in HTML and Endpoint files. Anything within these comments are server side comments and will not render out to the source code.

(** old code  **)
(** 
< h1>{{ page.title }}</h1>
**)
(** new code **)
<h3>{{ page.new_title }}</h3>

current_view

The {{ current_view }} call is a unique Zesty.io statement that will load the Page View associated with the current website url. {{ current_view }} is used once in the loader snippet. This allows a website to have consistent template elements above and below the unique aspect of each page.

{{ include header }}
<div class="content">
      {{ current_view }}
</div>
{{ include footer }}

date(format)

Uses the php date format to create the date string shown to the user.

<em>{{ page.custom_date.date(l, F jS, Y \a\t g:ia) }}</em> 
// <em>Sunday, January 1st, 2011 at 10:50pm</em>

dateDiffForHumans()

This method creates an expression of duration comparing the date and time from a content field to the current date and time.

{{ page.date.dateDiffForHumans() }}
<br/>
Returns:
<br/>
[x time] ago

each

The Parsley Each statements, also known as loops, allow you to iterate (loop) through each entry of a content model.

{{ each team_members as member }}
<p>{{ member.name }}</p>
{{ end-each }}

else

The else statement allows for a a final case if none of the defined conditions in the if statement have been met.

{{ if {page.field} == 1 }}
Example where first condition is met
{{ else-if {page.field} == 2 }}
Example where second condition is met
{{ else }}
Example where no condition is met
{{ end-if }}

else-if

The else-if statement is used for the defining alternate conditions, if the original if condition is not met. There can be as many else-if statements as needed after the opening if statement and before the end-if.

{{ if {page.field} == 1 }}
Example where first condition is met
{{ else-if {page.field} == 2 }}
Example where second condition is met
{{ else }}
Example where no condition is met
{{ end-if }}

end-each

Required to close each loops.

{{ each team_members as member }}
<p>{{ member.name }}</p>
{{ end-each }}

end-if

Required to close an if statement.

{{ if {page.field} == 1 }}
Example where first condition is met
{{ else-if {page.field} == 2 }}
Example where second condition is met
{{ else }}
Example where no condition is met
{{ end-if }}

escapeForJs()

The escapeForJs() call returns text with all the characters that could break a Javascript or JSON string escaped with \.

[
 {
 "title": "{{ page.title.escapeForJs() }}"
 }
]

filter()

The .filter() call can be used to access specific data based on the scope.

{{ scope.filter(z.zuid = '{this.one-to-one-field}').scope-field }}

{{ another-set.filter(z.zuid = '{this.one-to-one}').page_title }}

find_in_set

The find_in_set query allows for looking for a specific number or string in a comma separated list. find_in_set is always used to filter a Content Model related with a One-to_Many relational field.

{{ each tags as tag where find_in_set(tag.zuid, '{this.tags}') }}
<span>{{ tag.title }}</span>
{{ end-each }}

first()

The first() call will access the content available in the first entry of any content model.

{{ content_model.first().title }}
// "This is the title for the first entry of a content model"

format_currency()

The format_currency() call transforms a float number from a content field to the format $XX.XX.

{{ each products as product }}
    {{ product.price.format_currency() }}
{{ end-each }}

get_var

The get_var call returns the value of any query parameter in the current URL by referencing the key. When printing out any query parameter using get_var, be sure to use htmlentities() to prevent Cross-site Scripts and code injections.

// at the page /blog/?category=news

{{ get_var.category.htmlentities() }}

getfeedurl()

Every content model in Zesty (single page, page groups or datasets) can access a public RSS feed. It will always be content_model_reference_name-feed.rss/

{{ content_model.getfeedurl() }}

getImage(width, height, type)

The getImage() function takes optional arguments for width, height, and type (fit or crop) and returns a URL to that image. Fit is the default and does not need to be explicitly declared. Crop type requires all 3 arguments declared. Fit is similar to the css style "contain" and crop is similar to the css style "cover".

<img src="{{ page.photo.getImage() }}" />
<img src="{{ page.photo.getImage(100) }}" />
<img src="{{ page.photo.getImage(,100,fit) }}" /> 
<img src="{{ page.photo.getImage(100,,fit) }}" />
<img src="{{ page.photo.getImage(100,100,fit) }}" />
<img src="{{ page.photo.getImage(100,100,crop) }}" />

getImageTitle()

The getImageTitle() call returns the image title set in the Media Tab.

{{ page.image.getImageTitle() }}

getMediaURL()

The getMediaURL() call returns the original URL to that file. This is how to access files stored in the Media Tab that are not images, such as PDFs and MP4s. You can also use this call to access the original file of an image without the default optimization provided by Zesty.io.

<a href="{{ page.download_file.getMediaURL() }}" >Download</a>
// <a href="https://domain.com/file.pdf" >Download</a>
// If looping through an image field with multiple images use this call
{{ my-var.image.getmediaurl() }}

getNextURL(number)

The getNextURL() call returns the current page url with a query parameter p increased by the number specified. If there is no query parameter p, it will set it equal to the number specified. If no number is provided, the default is 10. This can be used with get_var.p to do pagination with an each loop.

{{ page.getNextURL(10) }}
// blog/?p=10

getPreviousURL(number)

The getPreviousURL() call returns the current page url with a query parameter p decreased by the number specified. If there is no query parameter p, it will set it equal to 0. If no number is provided as an argument, the default is 10. This can be used with get_var.p to do pagination with an each loop.

// on the page /blog/?p=30
{{ page.getPreviousURL(10) }}
// /blog/?p=20

getUrl()

The getUrl() call returns the relative path to the entry.

<a href="{{ blog.first().getUrl() }}">Link</a>
// <a href="/blog/">Link</a>

globals

The globals call is used to access content stored in the Globals (formerly Content Clippings) set.

{{ globals.phone }}

// Returns: 
555-555-5555

Google Analytics

Parsley provides a short cut to creating google analytics tag. The advantage of using this Parsley call is that as Google Analytics updates their syntax, it will be automatically updated on your website.

This call {{gaEvent(eventCategory,eventAction,eventLabel,eventValue)}} will trigger an event to your Google Analytics integration.

eventCategory: text required Typically the object that was interacted with (e.g. 'Video')

eventAction: text required The type of interaction (e.g. 'play')

eventLabel: text optional Useful for categorizing events (e.g. 'Fall Campaign')

eventValue: integer optional A numeric value associated with the event (e.g. 42)

Read the spec for GA Events here

<pre><a href="https://vimeo.com/144792991" {{gaEvent(Vimeo, Opened-video)}}>My Video</a></pre>

// returns 
<pre><a href="https://vimeo.com/144792991" onclick="ga('send','event', 'Vimeo', ' Opened-video', ' ',undefined,'{'nonInteraction': 1}');">My Video</a></pre>

gravatar()

The gravatar() function takes an email address and requests an image from the Gravatar API.

"<img class="highlight" src="{{ page.author_email.gravatar() }}" />"
// <img class="highlight" src="http://www.gravatar.com/avatar/205e460b479e2e5b48aec07710c08d50" />

hasChildren()

The hasChildren() function returns true or false, depending if any pages are parented to the entry. This can be used to find out if a page has pages that live under it. This is helpful when using logic to create secondary navigations.

{{ if {page.hasChildren()} }}
    // subnav code
{{ end-if }}

html_entity_decode()

The html_entity_decode call converts all HTML entities in the provided string to their applicable characters.

<p>{{ page.description.html_entity_decode() }}</p>

htmlentities()

The htmlentities() call convert all applicable characters to HTML entities. Should be used whenever printing out a get or post variable to prevent Cross-site Scripting and code injection.

{{page.html.htmlentities()}}
{{ get_var.search.htmlentities() }}

if

The if statement evaluates conditions using ==, !=, >=, <=, <, and >. Within the if statement, multiple conditions can be combined using && for and, and || for or. If statements can also execute math.

{{ if {page.field} == 1 && {page.price} + 10 > 100 }}
Example
{{ end-if }}

include

The include call is a prompt to reference the code in any View which includes views, snippets, and endpoints.

{{ include header }}

instance.host_env

Shows the instance's webengine URL. This only works in preview - not on live.

{{ instance.host_env }}

instance.host_live

Shows you the instance's first registered domain. This only works in preview - not on live.

{{ instance.host_live }}

instance.host_preview (or instance.host_relative)

Shows an instance's preview URL. This only works in preview - not on live.

{{ instance.host_preview }}
{{ instance.host_relative }}

instance.host_protocol

Shows the instance's protocol based on its settings. This only works on preview - not on live.

{{ instance.host_protocol }}

last()

The last() call will access the content available in the last entry of any content model.

{{ articles.last().title }}

limit

The limit statement is used in an each loop to specify how many content entries will result. The limit statement can also be used to skip a number of entries and show the next set if two arguments are provided.

{{ each articles as post where post.featured = 1 sort by post.date desc limit 20 }}
// entries 1 through 20
{{ end-each }}
{{ each articles as post where post.featured = 1 sort by post.date desc limit 10,20 }}
// entries 11 through 30
{{ end-each }}

length()

The length of a given string.

{{this.title.length()}}
// outputs a number, ex. 11

{{if {this.title.length()} > 5}}
 // do this 
{{end-if}}

math(statement)

The math() call executes an equation and returns a number.

(** plain output **)
{{ math( 1 + 2 ) }}

(** in use with a page field **)
{{ math( round({page.product_cost} * 1.5) ) }}

(** in use with a variable **)
{{ $my_num = 4 }}
{{ math( {$my_num} + 2 ) }}

navigation(number)

The navigation() call will access your Zesty.io Navigation Structure and output it as an HTML list.

site.navigation(INT depth, BOOL bootstrap, BOOL json)

{{site.navigation()}} // all defaults to html output
{{site.navigation(1)}}  // depth of the parent child levels
{{site.navigation(3, false, true)}}  // json output

The call will mimic the content navigation layout you find in the Zesty.io Content Manager and will ignore pages marked as unlisted. A number argument will specify what level of children to access within the navigation. Blank will list all pages, 1 will only be top level pages, 2 will be top level and subpages, 3 will be top level, subpages and tertiary pages, and so on.

{{ navigation() }}

// Returns:
<ul id="nav" class="nav">
     <li><a href="..."></a>
          <ul>
              <li><a href="..."></a></li>
          </ul>
     </li>
     <li><a href="..."></a></li>
</ul>

number_format(number)

The number_format() call returns a number with the specified number of decimals as the argument. Blank will remove any decimals.

{{ $number = 3.14 }}

{{ $number.number_format(1) }} 

// Returns: 
3.1

obfuscate()

The obfuscate() call returns obscure text to web crawlers, like email or phone numbers.

<a href="mailto:{{setting.contact-form.sending_email.obfuscate()}}">{{setting.contact-form.sending_email.obfuscate()}}</a>

xssProtect()

The xssProtect() call encode, decodes, strips html and turns characters into entities.

{{get_var.search.xssProtect()}}

page

The page call represents the content entry for the current page. It provides access to the data associated with the page. Any field in the Content Model can be accessed using this method when on the associated page.

{{ page.field_name }}

paypalStandard(action, product, price, shipping)

Quickly create a Paypal Standard buy, donate, or add to cart button in Parsley. To start using Paypal Standard a Paypal account must be created. The Paypal account information and tax rate are set in the Config Tab, under Settings-> Paypal Standard. These must be set to send money to the appropriate account. Action can be buy, addtocart and donate.

{{ each products as p }}
    <h1>{{ p.title }}</h1>
    <p>{{ p.price }}</p>
    {{ paypalStandard(buy,{p.title},{p.price},{p.shipping}) }}
 {{ end-each }}

post_clean()

The post_clean() call replaces \n \r\n or \ with a blank string.

{{ page.imported_content.post_clean() }}

post_var

The post_var call references data available in the requests post body. A couple common ways data can be added to the request post body is by submitting a form with a method="post" or using javascript POST. A common practical use for this call is to check is a form has been submitted by checking if the POST variable for the name of a required input is available.

{{ if {post_var.email} }}
Thank you for submitting your form, please check {{ post_var.email }} for a verification email.
{{ else }}
// form code
{{ end-if }}

preg_replace(pattern, replacement)

The preg_replace() call is a regular expression replacement. To learn more about Regular expression go to http://www.regular-expressions.info/

{{ page.content.preg_replace(/i/,*) }}

//Returns:
I'm gonna f*x that last joke by tak*ng out all the words and add*ng new ones.
- M*tch Hedberg

random()

The random() call is used to reference a random content entry for a content model.

{{ articles.random().title }}
// "the 'title' from a random item in 'articles'"

replace(pattern, replacement)

The replace() call does a simple replace without any regular expression.

// $description = "We wrote this blog post"
{{ $description.replace(We, I) }}
// "I wrote this blog post"

request.path()

Gets the full path from request excluding query parameters, for example lets use the url /store/12345/my-product/

{{ request.path() }} // outputs: /store/12345/my-product/

request.fullpath()

Gets the full path from request including query parameters, for example lets use the url /store/12345/my-product/?query=newvalue

{{ request.fullpath() }} // outputs: /store/12345/my-product/?query=newvalue

request.pathPart(num)

Get the "parts" of a url, for example lets use the url /store/12345/my-product/

{{ request.pathPart(1) }} // output: store
{{ request.pathPart(2) }} // output: 12345
{{ request.pathPart(3) }} // output: my-product
{{ request.pathPart(4) }} // output: (empty string)

request.queryParam(num)

Get the query parameters from url, for example lets use the url /store/12345/my-product/?test=hello

{{ request.queryParam(test) }} // output: hello
{{ request.queryParam(lala) }} // output: (empty string)

response.addHeader(name, ...value)

This call will add a header to the response. This call does not output anything.

{{response.addHeader(ab-test, 4)}}

// adds header to "ab-test" with value "4"

{{response.addHeader(Cache-Control, max-age=300, public)}}

// adds header to "Cache-Control" with value "max-age=300, public"

response.redirect301(url)

This call will add a redirect to the page.

{{response.redirect301(https://www.google.com)}}

response.redirect302(url)

This call will add a redirect to the page.

{{response.redirect302(https://www.google.com)}}

response.return404(message)

This call will response to your default 404 page, if there is no default 404 page, a generic 404 page will display with the "message" param as an h1 tag.

{{response.return404(My Default Message)}}

sectionlinks(number)

The sectionlinks() call looks at the current page and searches up to its top-level parent to create a navigation structure as an HTML list. Depth can be specified by as a numeric argument as sectionlinks(2) or you can choose not to display the top level page by passing sectionlinks(off)

{{ sectionlinks() }}

// <ul class="sectionLinks" id="sectionLinks">
      <li><a href="/about-us/" title="About Us">About Us</a>
         <ul class="sectionLinks">
         <li><a href="/about-us/team/" title="">Team</a></li>
         <li><a href="/about-us/qualifications/" title="Qualifications">Qualifications</a></li>
      </ul>
    </li>
</ul>

seo_meta_description

Use this call to access an item's meta description.

{{ this.seo_meta_description }}

seo_link_title

Deprecated: this functionality is only available to test in preview, it is not available for production websites because they are statically cached.

This call will access the Navigation Title for an item which is located under an item's Meta information in the Content section.

The sever_var call provides access to some of the most common server variables holding information about headers, paths, and script location. Some available variables are:

{{ this.seo_link_title }}

seo_meta_title

This call will access the Meta Title for an item which is located under an item's Meta information in the Content section.

{{ this.seo_meta_title }}

setting

The settings call returns data stored in the Instance Settings in the Config Tab.

<a href="{{ setting.social-links.linkedin_link }}">Zesty on Linkedin</a>
// <a href="https://www.linkedin.com/company/zesty-io/">Zesty on Linkedin</a>

toJSON()

A function outputs a JSON string that can be use in inline JavaScript. The function can be called off this, data calls, or loop variables. Parsley REPL Example https://parsley.zesty.io/turn-any-content-item-to-json/

{{this.toJSON(depth, showMeta)}}

  • depth is an integer for hydration depth, max is 5
  • showMeta true or false: true gives all meta, routing, and zuid details, and locale details. false is just content
{{this.toJSON(3,false)}} 

{{model.first().toJSON(2, true)}}

{{each model as item}}
    {{item.toJSON()}}
{{end-each}}

Example output for {{this.toJSON()}} call on zesty.io homepage

{
    title: "Your Content, Anywhere",
    content: "<h1>A CONTENT PLATFORM THAT POWERS ENTERPRISE GRADE WEBSITES AND APPLICATIONS</h1>",
    image: {
        type: "images",
        totalItems: 1,
        data: [
            {
                type: "image",
                zuid: "3-5c58033-eb8cq",
                url: "https://zestyio.media.zestyio.com/Zesty.io---9-5seg.mp4"
            }
        ]
    },
    customer_logo_heading: "Trusted by enterprises, startups, and everyone in between",
    main_headline: "Managing content at scale is difficult.",
    main_description: "<ul><li><p>Security issues, and out of date software</p></li><li><p>Editing the same content in multiple places</p></li><li><p>Developers needed for content updates</p></li></ul><h1>..but it doesn’t have to be</h1><ul><li><p>Centralized content management</p></li><li><p>Automated security and product improvements</p></li><li><p>Easy Distribution at global scale</p></li></ul>",
    pagina_nueva: null,
    og_image: {
    type: "images",
    totalItems: 1,
    data: [
            {
                type: "image",
                zuid: "3-a4f5ca6-a25px",
                url: "https://kfg6bckb.media.zestyio.com/zesty-share-image-generic.png"
            }
        ]
    },
    meta: {
        type: "item",
        model_name: "homepage",
        zuid: "7-31209c-g7qsjg",
        createdAt: "2020-10-01 06:33:30",
        updatedAt: "2020-10-01 06:33:29",
        listed: "1",
        version: "200",
        locale: {
            id: "1",
            name: "English (United States)",
            code: "en-US",
            default: "1",
            active: "1",
            enabled: "1"
        },
        model: {
            type: "model",
            zuid: "6-31079c-vdg69q",
            name: "homepage",
            label: "Homepage",
            resourceURI: "https://www.zesty.io/-/instant/6-31079c-vdg69q.json"
        },
        web: {
            url: "https://www.zesty.io/",
            uri: "/",
            fragment: "zesty_home",
            canonical_tag_mode: "1",
            sitemap_priority: "-1.0",
            sitemap_last_updated: "2020-10-01 06:33:29",
            canonical_query_param_whitelist: null,
            canonical_tag_custom_value: null,
            seo_link_text: "Homepage",
            seo_meta_title: "Zesty.io: The Headless CMS for Marketers + Developers",
            seo_meta_description: "Built for teams to manage and distribute content to multiple sites, devices, and anywhere else it needs to go.",
            seo_meta_keywords: null
        }
    }
}

json()

Accessing Objects in JSON.
Quickly access an object of an object to access its value.

{{this.json_object.json(obj_within.name)}}

Live example: https://parsley.zesty.io/accessing-a-json-object-within-an-object/

Iterating String Arrays

This refers to support for objects that store arrays of strings without keys.

{{each {this.json_object}.json(arr_strings) as item}} 
    {{item._string}}
{{/each}}

Live example: https://parsley.zesty.io/iterating-through-a-json-array-of-strings/

json(arr)

JSON Content Iteration.
JSON stored in a content model field can be iterated by using parsley, the same as content items.

{{each {this.json_object}.json(arr) as item}}
    {{item.title}}
{{end-each}}

Preview a live JSON iteration example here: https://parsley.zesty.io/iterating-over-a-content-stored-json-object-with-an-array/

instance

The instance call returns data specific to the instance, such as the CDN URL and the domain or globally relevant data, such as the current data-time or the current language.

{{ instance.cdn }}

instance.env

The instance.env call returns live for published content and views, and dev for versioned preview views and content.

{{ instance.env }}

instance.date()

The instance.date() call returns the current date time and accepts arguments to modify the date based on the PHP data-time. More information available at http://php.net/manual/en/function.date.php

{{ instance.date(l jS \of F Y h:i:s A) }}

//Returns:
Sunday 31st of July 2005 08:45:12 PM

instance.lorem(number)

The instance.lorem() call auto generates lorem ipsum placeholder content.

{{ instance.lorem(20) }}

// Returns : Lorem ipsum dolor sit amet, consectetur adipiscing elit. Aliquam volutpat quam quis erat tristique pretium.

instance.searchformatted(string,limit)

The instance.searchformatted() call takes a string and a limit as arguments and outputs instance-wide search results based on meta title, meta description, and path.

{{instance.searchformatted({get_var.search},10)}}

slugify()

The slugify() call will concatenate strings by replacing spaces with dashes (-), lowercasing all letters, and stripping out all punctuation.

// if the page title is the string: My New Homepage!

{{ page.title.slugify() }}

// calling slugify on the page title will return: my-new-homepage

sort by / order by

The sort by or order by statement can be used in an each loop to specify a field of the content model used to sort the content entries and which direction. Multiple fields can be specified using a comma separated list. If no sort is specified, the default is creation date. If no direction is specified ascending is the default.

{{ each articles as post where post.featured = 1 sort by post.date desc, post.title limit 10 }}
{{ end-each }}

striptags(start, length)

The striptags call will return a string with HTML tags removed. It also acts as a substring method if a start and length parameter are provided.

// $html = "Hello <h1>Zesty.io</h1> users"
{{ $html.striptags() }}
// "Hello Zesty.io users"

substr(start, length)

The substr() call takes the text value it is attached to and returns a substring with the desired length. This call takes both positive and negative numbers. Negative numbers, for example, substr(-5) will return the last five characters in a string. Does not consider whitespace as part of length.

// $description = "This is a longer description that needs shortened!"
{{ $description.substr(0,9) }}...
// "This is a lo..."

subWords(number_of_words)

The subWords() call returns the first specified number of words from a body of text.

{{ page.content.subWords(3) }}

trim_zeros()

The trim_zeros() call with take a number like 8.0000000 and just show 8. It will also trim leading zeros as well so a number like 00800.00 will show 800.

// $num = 8.000000000
{{ $num.trim_zeros() }}
// 8

truepath(zuid)

The truepath() call will determine the web URL to access a content entry based on a ZUID. Since one-to-one and internal page fields in Zesty.io store ZUIDs, truepath() is a quick way to access the direct url to that content entry without needing to do an each loop or filter call.

{{ truepath({page.link_to_article}) }}

truepath() can be used in each loops as well, for example {{ truepath(this.zuid) }} or {{ this.zuid.truepath() }} and you can assign it to a Parsley variable as well.

{{each articles as art limit 5}}
    {{art.zuid}} <br>
    {{art.zuid.truepath()}}<br>
    {{$var = {truepath({art.zuid})} }}
    VAR: {{$var}}<hr>
{{end-each}}

where

The where statement is used to filter content entries in an each loop. Where statements use SQL style conditional statements to determine which entries to pass through the loop. A single each can use multiple conditions using and and or.

{{ each articles as post where post.featured = 1 or post.category = '{page.featured_category}` sort by post.date desc limit 10 }}
{{ end-each }}

Each Statement Updates

Each statement's code gets reduced with shortcut statements and easier iteration of content relationships.

Content Relationship Iteration

The latest update simplifies the logic and reduces code length by allowing the relationship to be directly referenced in each loop. It also provides a shortened end-each option.

Old Style

{{each tags as tag where find_in_set(z.zuid,'{article.tags}')}}

    {{tag.name}}

{{end-each}}

New Style

{{each {article.tags} as tag}}

    {{tag.name}}

{{/each}}

Live example: https://parsley.zesty.io/iterate-through-a-model-relationship-field/

Array Commas

Outputting custom JSON data endpoints with Parsley content syntax requires extra logic if you are outputting an array of objects. This update streamlines that logic. Here is what use to look like to output 10 random articles to JSON.

Old Style

[

{{each articles as article limit 10 order by rand() }}

   {{article.toJSON()}}

  {{if {article._num} < {article._length} }},{{end-if}}

{{end-each}}

]

Note the logic does not have a trailing comma. That is what we replace with _arraycomma. _arraycomma will always know the last item in your array and will not put a comma after the last value. Otherwise, a comma will be appended after each value.

New Style

[

{{each articles as article limit 10 order by rand() }}

   {{article.toJSON()}} {{article._arraycomma}}

{{/each}}

]

Live example: https://parsley.zesty.io/generate-json-output/

Ending Each Iterations and If Logic Statements

A simple change to shorten code for an each statement {{end-each}} can now be written as {{endeach}} or {{/each}}.

If statements can now be terminated with {{/if}}, {{endif}} or {{end-if}}.

Data Access Calls

Data access calls are streamlined and gain two new functions.

Previous way to access a single item with parsley content syntax:

Old Style

{{articles.filter(z.zuid = '{this.articlezuid}').title}}

New Style

{{articles.filter({this.articlezuid}).title}

Live example: https://parsley.zesty.io/directly-access-an-item-of-a-content-model-by-zuid/

Using filter to search

{{articles.filter(title like '%parsley%').title}}

Example of a custom JSON endpoint to get a single item outputted to a fully hydrated JSON object. Make a custom file in code, /data/article.json with code

{{articles.filter({request.queryParam(zuid)}).toJSON()}}

Then call the endpoint with a ZUID www.domain.com/data/article.json?zuid=7-x1y2z3-x1y2z3