GitHub Access Token became invalid

It seems like the GitHub access token used for retrieving details about this repository from GitHub became invalid. This might prevent certain types of inspections from being run (in particular, everything related to pull requests).
Please ask an admin of your repository to re-new the access token on this website.
Completed
Push — master ( 7b3670...ed4dc1 )
by Zordius
04:42
created

Runtime::wi()   B

Complexity

Conditions 8
Paths 8

Size

Total Lines 17

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 11
CRAP Score 8

Importance

Changes 0
Metric Value
dl 0
loc 17
ccs 11
cts 11
cp 1
rs 8.4444
c 0
b 0
f 0
cc 8
nc 8
nop 6
crap 8
1
<?php
2
/*
3
4
MIT License
5
Copyright 2013-2019 Zordius Chen. All Rights Reserved.
6
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
7
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
8
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
9
10
Origin: https://github.com/zordius/lightncandy
11
*/
12
13
/**
14
 * file to support LightnCandy compiled PHP runtime
15
 *
16
 * @package    LightnCandy
17
 * @author     Zordius <[email protected]>
18
 */
19
20
namespace LightnCandy;
21
22
use \LightnCandy\Encoder;
23
24
/**
25
 * LightnCandy class for Object property access on a string.
26
 */
27
class StringObject
28
{
29
    protected $string;
30
31 2
    public function __construct($string)
32
    {
33 2
        $this->string = $string;
34 2
    }
35
36 2
    public function __toString()
37
    {
38 2
        return strval($this->string);
39
    }
40
}
41
42
/**
43
 * LightnCandy class for compiled PHP runtime.
44
 */
45
class Runtime extends Encoder
46
{
47
    const DEBUG_ERROR_LOG = 1;
48
    const DEBUG_ERROR_EXCEPTION = 2;
49
    const DEBUG_TAGS = 4;
50
    const DEBUG_TAGS_ANSI = 12;
51
    const DEBUG_TAGS_HTML = 20;
52
53
    /**
54
     * Output debug info.
55
     *
56
     * @param string $v expression
57
     * @param string $f runtime function name
58
     * @param array<string,array|string|integer> $cx render time context for lightncandy
59
     *
60
     * @expect '{{123}}' when input '123', 'miss', array('flags' => array('debug' => Runtime::DEBUG_TAGS), 'runtime' => 'LightnCandy\\Runtime'), ''
61
     * @expect '<!--MISSED((-->{{#123}}<!--))--><!--SKIPPED--><!--MISSED((-->{{/123}}<!--))-->' when input '123', 'wi', array('flags' => array('debug' => Runtime::DEBUG_TAGS_HTML), 'runtime' => 'LightnCandy\\Runtime'), false, null, false, function () {return 'A';}
62
     */
63 9
    public static function debug($v, $f, $cx)
64
    {
65
        // Build array of reference for call_user_func_array
66 9
        $P = func_get_args();
67 9
        $params = array();
68 9
        for ($i=2;$i<count($P);$i++) {
0 ignored issues
show
Performance Best Practice introduced by
It seems like you are calling the size function count() as part of the test condition. You might want to compute the size beforehand, and not on each iteration.

If the size of the collection does not change during the iteration, it is generally a good practice to compute it beforehand, and not on each iteration:

for ($i=0; $i<count($array); $i++) { // calls count() on each iteration
}

// Better
for ($i=0, $c=count($array); $i<$c; $i++) { // calls count() just once
}
Loading history...
69 9
            $params[] = &$P[$i];
70
        }
71 9
        $r = call_user_func_array((isset($cx['funcs'][$f]) ? $cx['funcs'][$f] : "{$cx['runtime']}::$f"), $params);
72
73 8
        if ($cx['flags']['debug'] & static::DEBUG_TAGS) {
74 5
            $ansi = $cx['flags']['debug'] & (static::DEBUG_TAGS_ANSI - static::DEBUG_TAGS);
75 5
            $html = $cx['flags']['debug'] & (static::DEBUG_TAGS_HTML - static::DEBUG_TAGS);
76 5
            $cs = ($html ? (($r !== '') ? '<!!--OK((-->' : '<!--MISSED((-->') : '')
77 5
                  . ($ansi ? (($r !== '') ? "\033[0;32m" : "\033[0;31m") : '');
78 5
            $ce = ($html ? '<!--))-->' : '')
79 5
                  . ($ansi ? "\033[0m" : '');
80 5
            switch ($f) {
81 5
                case 'sec':
82 3
                case 'wi':
83 3
                    if ($r == '') {
84 3
                        if ($ansi) {
85 1
                            $r = "\033[0;33mSKIPPED\033[0m";
86
                        }
87 3
                        if ($html) {
88 2
                            $r = '<!--SKIPPED-->';
89
                        }
90
                    }
91 3
                    return "$cs{{#{$v}}}$ce{$r}$cs{{/{$v}}}$ce";
92
                default:
93 3
                    return "$cs{{{$v}}}$ce";
94
            }
95
        } else {
96 3
            return $r;
97
        }
98
    }
99
100
    /**
101
     * Handle error by error_log or throw exception.
102
     *
103
     * @param array<string,array|string|integer> $cx render time context for lightncandy
104
     * @param string $err error message
105
     *
106
     * @throws \Exception
107
     */
108 18
    public static function err($cx, $err)
109
    {
110 18
        if ($cx['flags']['debug'] & static::DEBUG_ERROR_LOG) {
111 5
            error_log($err);
112 5
            return;
113
        }
114 13
        if ($cx['flags']['debug'] & static::DEBUG_ERROR_EXCEPTION) {
115 9
            throw new \Exception($err);
116
        }
117 4
    }
118
119
    /**
120
     * Handle missing data error.
121
     *
122
     * @param array<string,array|string|integer> $cx render time context for lightncandy
123
     * @param string $v expression
124
     */
125 5
    public static function miss($cx, $v)
126
    {
127 5
        static::err($cx, "Runtime: $v does not exist");
128 4
    }
129
130
    /**
131
     * For {{log}} .
132
     *
133
     * @param array<string,array|string|integer> $cx render time context for lightncandy
134
     * @param string $v expression
135
     */
136
    public static function lo($cx, $v)
137
    {
138
        error_log(var_export($v[0], true));
139
        return '';
140
    }
141
142
    /**
143
     * Resursive lookup variable and helpers. This is slow and will only be used for instance property or method detection or lambdas.
144
     *
145
     * @param array<string,array|string|integer> $cx render time context for lightncandy
146
     * @param array|string|boolean|integer|double|null $in current context
147
     * @param array<array|string|integer> $base current variable context
148
     * @param array<string|integer> $path array of names for path
149
     * @param array|null $args extra arguments for lambda
150
     *
151
     * @return null|string Return the value or null when not found
152
     *
153
     * @expect null when input array('scopes' => array(), 'flags' => array('prop' => 0, 'method' => 0, 'mustlok' => 0)), null, 0, array('a', 'b')
154
     * @expect 3 when input array('scopes' => array(), 'flags' => array('prop' => 0, 'method' => 0), 'mustlok' => 0), null, array('a' => array('b' => 3)), array('a', 'b')
155
     * @expect null when input array('scopes' => array(), 'flags' => array('prop' => 0, 'method' => 0, 'mustlok' => 0)), null, (Object) array('a' => array('b' => 3)), array('a', 'b')
156
     * @expect 3 when input array('scopes' => array(), 'flags' => array('prop' => 1, 'method' => 0, 'mustlok' => 0)), null, (Object) array('a' => array('b' => 3)), array('a', 'b')
157
     */
158 388
    public static function v($cx, $in, $base, $path, $args = null)
159
    {
160 388
        $count = count($cx['scopes']);
161 388
        $plen = count($path);
162 388
        while ($base) {
163 352
            $v = $base;
164 352
            foreach ($path as $i => $name) {
165 352
                if (is_array($v)) {
166 350
                    if (isset($v[$name])) {
167 346
                        $v = $v[$name];
168 346
                        continue;
169
                    }
170 31
                    if (($i === $plen - 1) && ($name === 'length')) {
171 1
                        return count($v);
172
                    }
173
                }
174 37
                if (is_object($v)) {
175 6
                    if ($cx['flags']['prop'] && !($v instanceof \Closure) && isset($v->$name)) {
176 4
                        $v = $v->$name;
177 4
                        continue;
178
                    }
179 5
                    if ($cx['flags']['method'] && is_callable(array($v, $name))) {
180 4
                        $v = $v->$name();
181 4
                        continue;
182
                    }
183
                }
184 34
                if ($cx['flags']['mustlok']) {
185 31
                    unset($v);
186 31
                    break;
187
                }
188 3
                return null;
189
            }
190 351
            if (isset($v)) {
191 340
                if ($v instanceof \Closure) {
192 29
                    if ($cx['flags']['mustlam'] || $cx['flags']['lambda']) {
193 29
                        if (!$cx['flags']['knohlp'] && ($args || ($args === 0))) {
194 20
                            $A = $args ? $args[0] : array();
195 20
                            $A[] = array('hash' => $args[1], '_this' => $in);
196
                        } else {
197 10
                            $A = array($in);
198
                        }
199 29
                        $v = call_user_func_array($v, $A);
200
                    }
201
                }
202 340
                return $v;
203
            }
204 31
            $count--;
205 31
            switch ($count) {
206
                case -1:
207 30
                    $base = $cx['sp_vars']['root'];
208 30
                    break;
209
                case -2:
210 29
                    return null;
211
                default:
212 6
                    $base = $cx['scopes'][$count];
213
            }
214
        }
215 40
        if ($args) {
216 3
            static::err($cx, 'Can not find helper or lambda: "' . implode('.', $path) . '" !');
217
        }
218 40
    }
219
220
    /**
221
     * For {{#if}} .
222
     *
223
     * @param array<string,array|string|integer> $cx render time context for lightncandy
224
     * @param array<array|string|integer>|string|integer|null $v value to be tested
225
     * @param boolean $zero include zero as true
226
     *
227
     * @return boolean Return true when the value is not null nor false.
228
     *
229
     * @expect false when input array(), null, false
230
     * @expect false when input array(), 0, false
231
     * @expect true when input array(), 0, true
232
     * @expect false when input array(), false, false
233
     * @expect true when input array(), true, false
234
     * @expect true when input array(), 1, false
235
     * @expect false when input array(), '', false
236
     * @expect false when input array(), array(), false
237
     * @expect true when input array(), array(''), false
238
     * @expect true when input array(), array(0), false
239
     */
240 74
    public static function ifvar($cx, $v, $zero)
241
    {
242 74
        return ($v !== null) && ($v !== false) && ($zero || ($v !== 0) && ($v !== 0.0)) && ($v !== '') && (is_array($v) ? (count($v) > 0) : true);
243
    }
244
245
    /**
246
     * For {{^var}} .
247
     *
248
     * @param array<string,array|string|integer> $cx render time context for lightncandy
249
     * @param array<array|string|integer>|string|integer|null $v value to be tested
250
     *
251
     * @return boolean Return true when the value is not null nor false.
252
     *
253
     * @expect true when input array(), null
254
     * @expect false when input array(), 0
255
     * @expect true when input array(), false
256
     * @expect false when input array(), 'false'
257
     * @expect true when input array(), array()
258
     * @expect false when input array(), array('1')
259
     */
260 37
    public static function isec($cx, $v)
261
    {
262 37
        return ($v === null) || ($v === false) || (is_array($v) && (count($v) === 0));
263
    }
264
265
    /**
266
     * For {{var}} .
267
     *
268
     * @param array<string,array|string|integer> $cx render time context for lightncandy
269
     * @param array<array|string|integer>|string|integer|null $var value to be htmlencoded
270
     *
271
     * @return string The htmlencoded value of the specified variable
272
     *
273
     * @expect 'a' when input array('flags' => array('mustlam' => 0, 'lambda' => 0)), 'a'
274
     * @expect 'a&amp;b' when input array('flags' => array('mustlam' => 0, 'lambda' => 0)), 'a&b'
275
     * @expect 'a&#039;b' when input array('flags' => array('mustlam' => 0, 'lambda' => 0)), 'a\'b'
276
     * @expect 'a&b' when input null, new \LightnCandy\SafeString('a&b')
277
     */
278 46
    public static function enc($cx, $var)
279
    {
280 46
        if ($var instanceof \LightnCandy\SafeString) {
281 1
            return (string)$var;
282
        }
283
284 46
        return htmlspecialchars(static::raw($cx, $var), ENT_QUOTES, 'UTF-8');
285
    }
286
287
    /**
288
     * For {{var}} , do html encode just like handlebars.js .
289
     *
290
     * @param array<string,array|string|integer> $cx render time context for lightncandy
291
     * @param array<array|string|integer>|string|integer|null $var value to be htmlencoded
292
     *
293
     * @return string The htmlencoded value of the specified variable
294
     *
295
     * @expect 'a' when input array('flags' => array('mustlam' => 0, 'lambda' => 0)), 'a'
296
     * @expect 'a&amp;b' when input array('flags' => array('mustlam' => 0, 'lambda' => 0)), 'a&b'
297
     * @expect 'a&#x27;b' when input array('flags' => array('mustlam' => 0, 'lambda' => 0)), 'a\'b'
298
     * @expect '&#x60;a&#x27;b' when input array('flags' => array('mustlam' => 0, 'lambda' => 0)), '`a\'b'
299
     */
300 341
    public static function encq($cx, $var)
301
    {
302 341
        if ($var instanceof \LightnCandy\SafeString) {
303 3
            return (string)$var;
304
        }
305
306 338
        return str_replace(array('=', '`', '&#039;'), array('&#x3D;', '&#x60;', '&#x27;'), htmlspecialchars(static::raw($cx, $var), ENT_QUOTES, 'UTF-8'));
307
    }
308
309
    /**
310
     * For {{#var}} or {{#each}} .
311
     *
312
     * @param array<string,array|string|integer> $cx render time context for lightncandy
313
     * @param array<array|string|integer>|string|integer|null $v value for the section
314
     * @param array<string>|null $bp block parameters
315
     * @param array<array|string|integer>|string|integer|null $in input data with current scope
316
     * @param boolean $each true when rendering #each
317
     * @param Closure $cb callback function to render child context
318
     * @param Closure|null $else callback function to render child context when {{else}}
319
     *
320
     * @return string The rendered string of the section
321
     *
322
     * @expect '' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), false, null, false, false, function () {return 'A';}
323
     * @expect '' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), null, null, null, false, function () {return 'A';}
324
     * @expect 'A' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), true, null, true, false, function () {return 'A';}
325
     * @expect 'A' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), 0, null, 0, false, function () {return 'A';}
326
     * @expect '-a=' when input array('scopes' => array(), 'flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), array('a'), null, array('a'), false, function ($c, $i) {return "-$i=";}
327
     * @expect '-a=-b=' when input array('scopes' => array(), 'flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), array('a','b'), null, array('a','b'), false, function ($c, $i) {return "-$i=";}
328
     * @expect '' when input array('scopes' => array(), 'flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), 'abc', null, 'abc', true, function ($c, $i) {return "-$i=";}
329
     * @expect '-b=' when input array('scopes' => array(), 'flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), array('a' => 'b'), null, array('a' => 'b'), true, function ($c, $i) {return "-$i=";}
330
     * @expect 'b' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), 'b', null, 'b', false, function ($c, $i) {return print_r($i, true);}
331
     * @expect '1' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), 1, null, 1, false, function ($c, $i) {return print_r($i, true);}
332
     * @expect '0' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), 0, null, 0, false, function ($c, $i) {return print_r($i, true);}
333
     * @expect '{"b":"c"}' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), array('b' => 'c'), null, array('b' => 'c'), false, function ($c, $i) {return json_encode($i);}
334
     * @expect 'inv' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), array(), null, 0, true, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
335
     * @expect 'inv' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), array(), null, 0, false, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
336
     * @expect 'inv' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), false, null, 0, true, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
337
     * @expect 'inv' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), false, null, 0, false, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
338
     * @expect 'inv' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), '', null, 0, true, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
339
     * @expect 'cb' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), '', null, 0, false, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
340
     * @expect 'inv' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), 0, null, 0, true, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
341
     * @expect 'cb' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), 0, null, 0, false, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
342
     * @expect 'inv' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'lambda' => 0)), new stdClass, null, 0, true, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
343
     * @expect 'cb' when input array('flags' => array('spvar' => 0, 'mustlam' => 0, 'mustsec' => 0, 'lambda' => 0)), new stdClass, null, 0, false, function ($c, $i) {return 'cb';}, function ($c, $i) {return 'inv';}
344
     * @expect '268' when input array('scopes' => array(), 'flags' => array('spvar' => 1, 'mustlam' => 0, 'lambda' => 0), 'sp_vars'=>array('root' => 0)), array(1,3,4), null, 0, false, function ($c, $i) {return $i * 2;}
345
     * @expect '038' when input array('scopes' => array(), 'flags' => array('spvar' => 1, 'mustlam' => 0, 'lambda' => 0), 'sp_vars'=>array('root' => 0)), array(1,3,'a'=>4), null, 0, true, function ($c, $i) {return $i * $c['sp_vars']['index'];}
346
     */
347 170
    public static function sec($cx, $v, $bp, $in, $each, $cb, $else = null)
348
    {
349 170
        $push = ($in !== $v) || $each;
350
351 170
        $isAry = is_array($v) || ($v instanceof \ArrayObject);
352 170
        $isTrav = $v instanceof \Traversable;
353 170
        $loop = $each;
354 170
        $keys = null;
355 170
        $last = null;
356 170
        $isObj = false;
357
358 170
        if ($isAry && $else !== null && count($v) === 0) {
359 3
            $ret = $else($cx, $in);
360 3
            return $ret;
361
        }
362
363
        // #var, detect input type is object or not
364 168
        if (!$loop && $isAry) {
365 66
            $keys = array_keys($v);
366 66
            $loop = (count(array_diff_key($v, array_keys($keys))) == 0);
367 66
            $isObj = !$loop;
368
        }
369
370 168
        if (($loop && $isAry) || $isTrav) {
371 106
            if ($each && !$isTrav) {
372
                // Detect input type is object or not when never done once
373 50
                if ($keys == null) {
374 50
                    $keys = array_keys($v);
375 50
                    $isObj = (count(array_diff_key($v, array_keys($keys))) > 0);
376
                }
377
            }
378 106
            $ret = array();
379 106
            if ($push) {
380 104
                $cx['scopes'][] = $in;
381
            }
382 106
            $i = 0;
383 106
            if ($cx['flags']['spvar']) {
384 95
                $old_spvar = $cx['sp_vars'];
385 95
                $cx['sp_vars'] = array_merge(array('root' => $old_spvar['root']), $old_spvar, array('_parent' => $old_spvar));
386 95
                if (!$isTrav) {
387 94
                    $last = count($keys) - 1;
388
                }
389
            }
390
391 106
            $isSparceArray = $isObj && (count(array_filter(array_keys($v), 'is_string')) == 0);
392 106
            foreach ($v as $index => $raw) {
393 99
                if ($cx['flags']['spvar']) {
394 89
                    $cx['sp_vars']['first'] = ($i === 0);
395 89
                    $cx['sp_vars']['last'] = ($i == $last);
396 89
                    $cx['sp_vars']['key'] = $index;
397 89
                    $cx['sp_vars']['index'] = $isSparceArray ? $index : $i;
398 89
                    $i++;
399
                }
400 99
                if (isset($bp[0])) {
401 4
                    $raw = static::m($cx, $raw, array($bp[0] => $raw));
402
                }
403 99
                if (isset($bp[1])) {
404 3
                    $raw = static::m($cx, $raw, array($bp[1] => $index));
405
                }
406 99
                $ret[] = $cb($cx, $raw);
407
            }
408 105
            if ($cx['flags']['spvar']) {
409 94
                if ($isObj) {
410 16
                    unset($cx['sp_vars']['key']);
411
                } else {
412 80
                    unset($cx['sp_vars']['last']);
413
                }
414 94
                unset($cx['sp_vars']['index']);
415 94
                unset($cx['sp_vars']['first']);
416 94
                $cx['sp_vars'] = $old_spvar;
417
            }
418 105
            if ($push) {
419 103
                array_pop($cx['scopes']);
420
            }
421 105
            return join('', $ret);
422
        }
423 66
        if ($each) {
424 2
            if ($else !== null) {
425 1
                $ret = $else($cx, $v);
426 1
                return $ret;
427
            }
428 2
            return '';
429
        }
430 65
        if ($isAry) {
431 11
            if ($push) {
432 10
                $cx['scopes'][] = $in;
433
            }
434 11
            $ret = $cb($cx, $v);
435 11
            if ($push) {
436 10
                array_pop($cx['scopes']);
437
            }
438 11
            return $ret;
439
        }
440
441 55
        if ($cx['flags']['mustsec']) {
442 20
            return $v ? $cb($cx, $in) : '';
443
        }
444
445 35
        if ($v === true) {
446 9
            return $cb($cx, $in);
447
        }
448
449 28
        if (($v !== null) && ($v !== false)) {
450 9
            return $cb($cx, $v);
451
        }
452
453 21
        if ($else !== null) {
454 10
            $ret = $else($cx, $in);
455 10
            return $ret;
456
        }
457
458 12
        return '';
459
    }
460
461
    /**
462
     * For {{#with}} .
463
     *
464
     * @param array<string,array|string|integer> $cx render time context for lightncandy
465
     * @param array<array|string|integer>|string|integer|null $v value to be the new context
466
     * @param array<array|string|integer>|string|integer|null $in input data with current scope
467
     * @param array<string>|null $bp block parameters
468
     * @param Closure $cb callback function to render child context
469
     * @param Closure|null $else callback function to render child context when {{else}}
470
     *
471
     * @return string The rendered string of the token
472
     *
473
     * @expect '' when input array(), false, null, false, function () {return 'A';}
474
     * @expect '' when input array(), null, null, null, function () {return 'A';}
475
     * @expect '{"a":"b"}' when input array(), array('a'=>'b'), null, array('a'=>'c'), function ($c, $i) {return json_encode($i);}
476
     * @expect '-b=' when input array(), 'b', null, array('a'=>'b'), function ($c, $i) {return "-$i=";}
477
     */
478 30
    public static function wi($cx, $v, $bp, $in, $cb, $else = null)
479
    {
480 30
        if (isset($bp[0])) {
481 4
            $v = static::m($cx, $v, array($bp[0] => $v));
482
        }
483 30
        if (($v === false) || ($v === null) || (is_array($v) && (count($v) === 0))) {
484 5
            return $else ? $else($cx, $in) : '';
485
        }
486 26
        if ($v === $in) {
487 5
            $ret = $cb($cx, $v);
488
        } else {
489 21
            $cx['scopes'][] = $in;
490 21
            $ret = $cb($cx, $v);
491 21
            array_pop($cx['scopes']);
492
        }
493 26
        return $ret;
494
    }
495
496
    /**
497
     * Get merged context.
498
     *
499
     * @param array<string,array|string|integer> $cx render time context for lightncandy
500
     * @param array<array|string|integer>|string|integer|null $a the context to be merged
501
     * @param array<array|string|integer>|string|integer|null $b the new context to overwrite
502
     *
503
     * @return array<array|string|integer>|string|integer the merged context object
0 ignored issues
show
Documentation introduced by
Should the return type not be array|string|integer|null|StringObject? Also, consider making the array more specific, something like array<String>, or String[].

This check compares the return type specified in the @return annotation of a function or method doc comment with the types returned by the function and raises an issue if they mismatch.

If the return type contains the type array, this check recommends the use of a more specific type like String[] or array<String>.

Loading history...
504
     *
505
     */
506 103
    public static function m($cx, $a, $b)
507
    {
508 103
        if (is_array($b)) {
509 103
            if ($a === null) {
510 14
                return $b;
511 92
            } elseif (is_array($a)) {
512 82
                return array_merge($a, $b);
513 10
            } elseif ($cx['flags']['method'] || $cx['flags']['prop']) {
514 6
                if (!is_object($a)) {
515 2
                    $a = new StringObject($a);
516
                }
517 6
                foreach ($b as $i => $v) {
518 2
                    $a->$i = $v;
519
                }
520
            }
521
        }
522 10
        return $a;
523
    }
524
525
    /**
526
     * For {{> partial}} .
527
     *
528
     * @param array<string,array|string|integer> $cx render time context for lightncandy
529
     * @param string $p partial name
530
     * @param array<array|string|integer>|string|integer|null $v value to be the new context
531
     *
532
     * @return string The rendered string of the partial
533
     *
534
     */
535 100
    public static function p($cx, $p, $v, $pid, $sp = '')
536
    {
537 100
        $pp = ($p === '@partial-block') ? "$p" . ($pid > 0 ? $pid : $cx['partialid']) : $p;
538
539 100
        if (!isset($cx['partials'][$pp])) {
540 8
            static::err($cx, "Can not find partial named as '$p' !!");
541 4
            return '';
542
        }
543
544 94
        $cx['partialid'] = ($p === '@partial-block') ? (($pid > 0) ? $pid : (($cx['partialid'] > 0) ? $cx['partialid'] - 1 : 0)) : $pid;
545
546 94
        return call_user_func($cx['partials'][$pp], $cx, static::m($cx, $v[0][0], $v[1]), $sp);
547
    }
548
549
    /**
550
     * For {{#* inlinepartial}} .
551
     *
552
     * @param array<string,array|string|integer> $cx render time context for lightncandy
553
     * @param string $p partial name
554
     * @param Closure $code the compiled partial code
555
     *
556
     */
557 9
    public static function in(&$cx, $p, $code)
558
    {
559 9
        $cx['partials'][$p] = $code;
560 9
    }
561
562
    /* For single custom helpers.
563
     *
564
     * @param array<string,array|string|integer> $cx render time context for lightncandy
565
     * @param string $ch the name of custom helper to be executed
566
     * @param array<array|string|integer>|string|integer|null $vars variables for the helper
567
     * @param string $op the name of variable resolver. should be one of: 'raw', 'enc', or 'encq'.
568
     * @param array<string,array|string|integer> $_this current rendering context for the helper
569
     *
570
     * @return string The rendered string of the token
571
     */
572 120
    public static function hbch($cx, $ch, $vars, $op, &$_this)
573
    {
574 120
        if (isset($cx['blparam'][0][$ch])) {
575 1
            return $cx['blparam'][0][$ch];
576
        }
577
578
        $options = array(
579 120
            'name' => $ch,
580 120
            'hash' => $vars[1],
581 120
            'contexts' => count($cx['scopes']) ? $cx['scopes'] : array(null),
582 120
            'fn.blockParams' => 0,
583 120
            '_this' => &$_this
584
        );
585
586 120
        if ($cx['flags']['spvar']) {
587 115
            $options['data'] = $cx['sp_vars'];
588
        }
589
590 120
        return static::exch($cx, $ch, $vars, $options);
591
    }
592
593
    /**
594
     * For block custom helpers.
595
     *
596
     * @param array<string,array|string|integer> $cx render time context for lightncandy
597
     * @param string $ch the name of custom helper to be executed
598
     * @param array<array|string|integer>|string|integer|null $vars variables for the helper
599
     * @param array<string,array|string|integer> $_this current rendering context for the helper
600
     * @param boolean $inverted the logic will be inverted
601
     * @param Closure|null $cb callback function to render child context
602
     * @param Closure|null $else callback function to render child context when {{else}}
603
     *
604
     * @return string The rendered string of the token
605
     */
606 57
    public static function hbbch($cx, $ch, $vars, &$_this, $inverted, $cb, $else = null)
607
    {
608
        $options = array(
609 57
            'name' => $ch,
610 57
            'hash' => $vars[1],
611 57
            'contexts' => count($cx['scopes']) ? $cx['scopes'] : array(null),
612 57
            'fn.blockParams' => 0,
613 57
            '_this' => &$_this,
614
        );
615
616 57
        if ($cx['flags']['spvar']) {
617 53
            $options['data'] = $cx['sp_vars'];
618
        }
619
620 57
        if (isset($vars[2])) {
621 3
            $options['fn.blockParams'] = count($vars[2]);
622
        }
623
624
        // $invert the logic
625 57
        if ($inverted) {
626
            $tmp = $else;
627
            $else = $cb;
628
            $cb = $tmp;
629
        }
630
631
        $options['fn'] = function ($context = '_NO_INPUT_HERE_', $data = null) use ($cx, &$_this, $cb, $options, $vars) {
632 46
            if ($cx['flags']['echo']) {
633
                ob_start();
634
            }
635 46
            if (isset($data['data'])) {
636 6
                $old_spvar = $cx['sp_vars'];
637 6
                $cx['sp_vars'] = array_merge(array('root' => $old_spvar['root']), $data['data'], array('_parent' => $old_spvar));
638
            }
639 46
            $ex = false;
640 46
            if (isset($data['blockParams']) && isset($vars[2])) {
641 3
                $ex = array_combine($vars[2], array_slice($data['blockParams'], 0, count($vars[2])));
642 3
                array_unshift($cx['blparam'], $ex);
643 44
            } elseif (isset($cx['blparam'][0])) {
644 1
                $ex = $cx['blparam'][0];
645
            }
646 46
            if (($context === '_NO_INPUT_HERE_') || ($context === $_this)) {
647 29
                $ret = $cb($cx, is_array($ex) ? static::m($cx, $_this, $ex) : $_this);
648
            } else {
649 20
                $cx['scopes'][] = $_this;
650 20
                $ret = $cb($cx, is_array($ex) ? static::m($cx, $context, $ex) : $context);
651 20
                array_pop($cx['scopes']);
652
            }
653 46
            if (isset($data['data'])) {
654 6
                $cx['sp_vars'] = $old_spvar;
655
            }
656 46
            return $cx['flags']['echo'] ? ob_get_clean() : $ret;
657
        };
658
659 57
        if ($else) {
660
            $options['inverse'] = function ($context = '_NO_INPUT_HERE_') use ($cx, $_this, $else) {
661 6
                if ($cx['flags']['echo']) {
662
                    ob_start();
663
                }
664 6
                if ($context === '_NO_INPUT_HERE_') {
665 1
                    $ret = $else($cx, $_this);
666
                } else {
667 5
                    $cx['scopes'][] = $_this;
668 5
                    $ret = $else($cx, $context);
669 5
                    array_pop($cx['scopes']);
670
                }
671 6
                return $cx['flags']['echo'] ? ob_get_clean() : $ret;
672 10
            };
673
        } else {
674
            $options['inverse'] = function () {
675
                return '';
676
            };
677
        }
678
679 57
        return static::exch($cx, $ch, $vars, $options);
680
    }
681
682
    /**
683
     * Execute custom helper with prepared options
684
     *
685
     * @param array<string,array|string|integer> $cx render time context for lightncandy
686
     * @param string $ch the name of custom helper to be executed
687
     * @param array<array|string|integer>|string|integer|null $vars variables for the helper
688
     * @param array<string,array|string|integer> $options the options object
689
     *
690
     * @return string The rendered string of the token
691
     */
692 167
    public static function exch($cx, $ch, $vars, &$options)
693
    {
694 167
        $args = $vars[0];
695 167
        $args[] = $options;
696 167
        $r = true;
697
698
        try {
699 167
            $r = call_user_func_array($cx['helpers'][$ch], $args);
700 2
        } catch (\Exception $E) {
701 2
            static::err($cx, "Runtime: call custom helper '$ch' error: " . $E->getMessage());
702
        }
703
704 166
        return $r;
705
    }
706
}
707