In Smarty, is there a standard function or an easy way to generate json from an array, as json_encode()
does in php?
I could not see it in Smarty documentation, wanted to ask here.
In Smarty, is there a standard function or an easy way to generate json from an array, as json_encode()
does in php?
I could not see it in Smarty documentation, wanted to ask here.
Now deprecated
This should work. The @ makes smarty run the modifier against the whole array, otherwise it does it for each element.
{$myarray|@json_encode}
If $escape_html is enabled, you will need to use nofilter
:
{$myarray|@json_encode nofilter}
{$myarray|@json_encode:64 nofilter}
would leave slashes un-escaped. https://mcmap.net/q/658464/-json_encode-expects-parameter-2-to-be-long-string-given –
Disarming $smarty->registerPlugin('modifier', 'json_encode', function($json) { return json_encode($json, JSON_THROW_ON_ERROR); });
–
Albrecht While {$myarray|@json_encode}
does in fact emit the array encoded in json, it also escapes special characters, making the array unusable in javascript.
To avoid escaping special characters and also be able to use the array in javascript use the nofilter flag:
{$myarray|@json_encode nofilter}
You have to use json_encode()
in your php code then assign the value to smarty using
$smarty->assign()
function. After that you have to parse that value in your template file using
javascript.
code snippet:
{literal}
<script>
var json = JSON.parse('{/literal}{$your_json_encoded_array}{literal}');
//another statement
</script>
{/literal}
While {$myarray|@json_encode nofilter}
will work, there is a security hole here since we are doing variable escaping. Variable escaping (via nofilter
) is highly discouraged because malicious code can be displayed and executed easily.
There is another approach to consider, using the 'javascript'
escaping and replacing '"'
with "
:
{literal}const my_javascript_array = JSON.parse(`{/literal}{json_encode($myarray)|escape:'javascript'}{literal}`.replaceAll('"', '"'));
It is a bit convoluted, but you will thank yourself down the line for learning this trick :)
{literal}
<script type="text/javascript">
<!--
var newVar ={/literal}{$myarray|@json_encode nofilter};{literal}
// -->
</script>
{/literal}
My solution
Now with smarty 4.3.4, the @json_encode build in modifier is deprecated. You'll need to write your own modifier:
$smarty->registerPlugin('modifier', 'json_encode', function($json) {
return json_encode($json, JSON_THROW_ON_ERROR);
});
Then you can use it in the template:
<script type="text/javascript">
// we are setting some variables
// nofilter disables html entity escaping
var variables = {$variables|json_encode nofilter};
</script>
<!-- if you need to write the json_encoded string as html data attribute add |escape:'html' : -->
<script type="text/javascript" data-userinfo="{$variables|json_encode|escape:'html'}" src="..."></script>
I don't know of any. You could assign the json_encode()'s result to a smarty variable in your 'php code' with $smarty->assign( ... ), and then use it in your template.
Also there is a Smarty extension for json_decode(). It shouldn't be hard to write your own extension for the opposite based on this.
© 2022 - 2024 — McMap. All rights reserved.