-
Notifications
You must be signed in to change notification settings - Fork 27
Jedi 参考手册
CRLF(\r\n) 和 CR(\r) 会被统一解析为 LF(\n)。
## 标签(1)一个单独的单词就是一个标签:
html
会被解析为:
<html></html>
(2)包涵ID的标签:
div#container
会被解析为:
<div id="container"></div>
(3)包涵class的标签:
div.user-details
会被解析为:
<div class="user-details"></div>
(4)同时包含class和id:
div.bar.baz#foo
会被解析为:
<div id="foo" class="bar baz"></div>
注意: class(.xxx)必须写在id(#xxx)的前面,class可以有多个,id只能有一个。 不能在这里插值。
(5)标签后面直接跟‘=’,可以直接输出一个表达式:
span = 12345 + 54321
会被解析为:
<span>66666</span>
(6)在同一行写子标签
td > div.linkstd
会被解析为:
<td><div class="linkstd"></div></td>
注意:‘>‘后面只能跟子标签。
(7)标签后面直接跟随文本:
div.text '12345
会被解析为:
<div class="text">12345</div>
(8)标签的属性:
a @href="www.baixing.com"
'button
会被解析为:
<a href="www.baixing.com">
button
</a>
a @href="{url}"
'button
会被解析为:
<a href="<?= $data->url ?>">
button
</a>
(8)标签的属性可以后置到它下一级子block的任何位置:
div
@name1="example"
'12345
@name2="text"
会被解析为:
<div name1="example" name2="text">
12345
</div>
写在单引号 ' 和双引号 " 后面的内容会作为文本输出:
'hello!
"world!
会被解析为:
hello!world!
(2)大段文本可以写在子block中:
'
| foo bar baz
| rawr rawr
| super cool
| go jade go
会被解析为
| foo bar baz | rawr rawr | super cool | go jade go
(3)一个单独的'代表一个空格:
'
'hello
会被解析为:
hello
(4)文本开头处的空格会被忽略
' hello
' world
会被解析为:
hello world
(5)单引号开头的文本写在同一行时,可以在末尾加引号 ' ,末尾的引号 ' 不会被输出。
'hello world'
会被解析为:
hello world
(6)以双引号开头的文本可以插入表达式
"12345 {text}
会被解析为:
12345 <?= htmlspecialchars($data->text) ?>
(7)需要输出 {}
时,可以用‘\’转义掉它。
"12345 \{something}
会被解析为:
12345 {something}
会被解析为:
<?= adsenseScript ?>
(1)使用'//'可以注释掉一行代码,但是'//'必须写在行首。
// just some paragraphs
p 'foo
p 'bar
会被解析为:
<?php
// just some paragraphs
?>
<p>foo</p>
<p>bar</p>
(2)使用字符 '--' 可以简单的注释掉整个block:
--div.test
'example
span
'12345
会被解析为:
<?php
/*
*div.test
* 'example
* span
* '12345
*/
?>
(3)用'!'可以把文本注释在HTML页面中:
!hello world
会被解释为:
<!--hello world-->
(4)用'!'注释整段:
!
hello
world
会被解释为:
<!--hello world-->
(5)JS代码可以用'!'来写。
script
!
_defer.push('/follow.js');
alert('hello world');
会被解释为:
<script>
_defer.push('/follow.js');
alert('hello world');
</script>
注意:使用'!'写JS代码时,JS中不能插入表达式。
(6)若要在JS中插入表达式,可以用双引号 " 定义好[变量],然后在后面使用。
script
"
var categoryEnglishName = '{category.id}';
!
alert(categoryEnglishName);
会被解释为:
<script>
var categoryEnglishName = '<?= category.id ?>';
alert(categoryEnglishName);
</script>
会被解析为:
<?php
foreach ($data->ads as $ad) {
echo htmlspecialchars($ad);
}
?>
(2)key,value形式
:for (key, ad) in ads
"{key} {ad}
会被解析为:
<?php
foreach ($data->ads as $key => $ad) {
echo htmlspecialchars($key), htmlspecialchars(' '), htmlspecialchars($ad);
}
?>
(3)多重嵌套的foreach循环:
:for x in list1, y in list2
"{x}, {y}
会被解析为:
foreach ($data->list1 as $x) {
foreach ($data->list2 as $y) {
echo htmlspecialchars($x), htmlspecialchars(', '), htmlspecialchars($y);
}
}
注意,后面的表达式不能引用前面的循环变量,如:
:for i in list1, j in i
上面代码会出现编译错误。
:for x in [0..3]
"{x}
会被解析为:
foreach ([0..3] as $x) {
echo htmlspecialchars($x);
}
(1):if 关键字
:if a > b
'hello
会被解析为:
<?php
if ($data->a > $data->b) {
echo 'hello';
}
?>
(2):else 与 :else if 关键字:
:if a > b
'a bigger than b
:else if a < b
'a smaller than b
:else
'a equals b
会被解析为:
<?php
if ($data->a > $data->b) {
echo 'a bigger than b';
} elseif ($data->a < $data->b) {
echo 'a smaller than b';
} else {
echo 'a equals b';
}
?>
(1):let关键字
:let x = 'hello world'
"{x}
会被解析为:
<?php
call_user_func(function($x) {
echo htmlspecialchars($x);
}, 'hello world');
?>
(2)使用:let赋值时,被赋值的变量只在它的子block中有效:
"{x}
:let x = 'hello world'
"{x}
"{x}
会被解析为:
echo htmlspecialchars($data->x);
<?php
call_user_func(function($x) {
echo htmlspecialchars($x);
}, 'hello world');
?>
echo htmlspecialchars($data->x);
<?php if ($data->a > $data->b) { ?>
<div class="content"></div>
<?php } ?>
会被解析为:
<?php
$words = ['Hello', 'world!']
foreach ($words as $w)
{
var_dump($greeting);
}
?>
注意编译器会自动加上分号或大括号。
(2) 如果要访问传给模板的数据,需要用到 $data 变量。
= name
:let name = 'hax'
- echo 'inner name: ', name
- echo 'outer name: ', $data->name
会被编译为:
<?php
echo htmlspecialchars($data->name);
call_user_func(function ($name) use ($data) {
echo 'inner name: ', $name;
echo 'outer name: ', $data->name;
}, 'hax')
?>
注意,现在编译器存在一个bug,即 use ($data) 只有在内部访问了模型数据才会被编译器加上,临时解决方案是引用任何一个模型数据内容,比如假设存在 title,则可将上面的代码写成:
= name
:let name = 'hax'
:let hack = title
- echo 'inner name: ', name
- echo 'outer name: ', $data->name
(3)注入时注意末尾的';',Jedi的自动添加可能会带来Bug。
-for ($i = 0; $i < 10; $i++) {
-$varb = 'hello world';
-}
会被解析为:
<?php
for ($i = 0; $i < 10; $i++) {;
$varb = 'hello world';
};
?>
(4)'-'只会影响本行内容,它的子block会继续按照Jedi语法解析,并会自动添加大括号'{ }'。
-for ($i = 0; $i < 10; $i++)
-$varb = 'hello world'
会被解析为:
<?php
for ($i = 0; $i < 10; $i++)
{
$varb = 'hello world';
}
?>
使用类似mixin的语法,script = expression
。
该语句会被编译成JS函数(IIFE),其参数是data
,data的值就是expression,在PHP中通过json_encode传给js
示例:
// suppose formatImages = true
script = [formatImages]
!
window.__formatImages = data[0]
会被编译成类似
<script>function(data) {
window.__formatImages = data[0]
}(true)</script>
上面的Bigsheep.jedi会被翻译为
//layout.jedi是一个模板
head
'我是head
body
//Bigsheep.jedi引用了模板layout.jedi
//在body的内容前增加内容
'我是Bigsheep
//替换body的内容
'我也是Bigsheep
//在body的内容后增加内容
'我还是Bigsheep
注意:目前Jedi的模板只能继承一层,不支持诸如:B继承A,C再继承B。但Jedi支持继承多个模板。
## 闭包 (1)Jedi中的闭包类似于定义一个PHP函数 ```shell :: time.friendly () '12345 time.friendly = '' ```会被解析为:
<?php
call_user_func(function ($context) {
echo '<time class="friendly">';
echo '12345';
echo '</time>';
}, '');
?>
(2)引用闭包时,可以传参数进去。闭包中通过星号 '*' 使用参数。
:: time.friendly ()
"{*}
time.friendly = '12345'
会被解析为:
<?php call_user_func(function ($context) { ?>
echo '<time class="friendly">';
echo htmlspecialchars($context);
echo '</time>';
<?php }, '12345');?>
(3)闭包中的参数可以是简单类型的变量,也可以是一个数组或者一个实例。
:: time.friendly ()
"{*[0]}
"{*[1]}
time.friendly = ['12345', '23456']
会被解析为:
<?php call_user_func(function ($context) { ?>
echo '<time class="friendly">';
echo htmlspecialchars($context[0]);
echo htmlspecialchars($context[1]);
echo '</time>';
<?php }, ['12345', '23456']);?>
(1)使用php函数
:external is_array
:if is_array(a)
'hello world
会被解析为:
<?php
if (is_array($data->a)) {
echo 'hello world';
}
?>
(2)引用包涵静态方法的PHP类
:external Category
:if Category.exist(categoryName)
'hello world
会被解析为:
<?php
if (Category::exist($data->categoryName)) {
echo 'hello world';
}
?>
注意: external必须写在文件头部第一行。
# 词法 ## 变量 (1)Jedi变量由字母、数字和'$'组成,不能以数字开头。(2)变量默认被翻译为$data的属性,如'x'会被翻译为'$data->x'。
(3)由:let和:for产生的临时变量不会被翻译为$data的属性。
如:let x='12345' 中, 变量 'x' 会被翻译为 '$x' 。
(4)'$'会被当做一般字符进行翻译,如:
'$x' 会被翻译为 '$data->$x'
"||" , "&&"
"===" , "!==" , "==" , "!=" , "<=>" , ">=" , "<=" , ">" , "<"
"<<<" , ">>>" , "<<" , ">>"
"+" , "-"
"×" , "÷" , "mul" , "div" , "mod"
'!' , "not"
(2)Jedi用if then else 替代 ? :
"{if x then '12345' else '54321'}
会被解析为
(3)实例变量与实例方法用 点 '.' 连接。
实例变量:
"{ad.categoryEnglishName}
会被解析为
实例方法:
"{ad.user()}
会被解析为