Skip to content

Commit e5e0e3b

Browse files
committed
snippet wip
1 parent 0729ede commit e5e0e3b

File tree

4 files changed

+104
-9
lines changed

4 files changed

+104
-9
lines changed

src/Application/UI/Control.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -146,7 +146,7 @@ public function isControlInvalid(?string $snippet = null): bool
146146
if ($snippet !== null) {
147147
return $this->invalidSnippets[$snippet] ?? isset($this->invalidSnippets["\0"]);
148148

149-
} elseif (count($this->invalidSnippets) > 0) {
149+
} elseif (array_keys($this->invalidSnippets, true, true)) {
150150
return true;
151151
}
152152

src/Bridges/ApplicationLatte/Nodes/ControlNode.php

+6-2
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ class ControlNode extends StatementNode
3131
public ?ExpressionNode $method = null;
3232
public ArrayNode $args;
3333
public ?bool $escape = null;
34+
public bool $inSnippetArea = false;
3435

3536

3637
public static function create(Tag $tag): static
@@ -40,6 +41,7 @@ public static function create(Tag $tag): static
4041
$stream = $tag->parser->stream;
4142
$node = new static;
4243
$node->name = $tag->parser->parseUnquotedStringOrExpression(colon: false);
44+
$node->inSnippetArea = $tag->closestTag(['snippetArea']) && !$tag->closestTag(['snippet']);
4345
if ($stream->tryConsume(':')) {
4446
$node->method = $tag->parser->parseExpression();
4547
}
@@ -94,11 +96,14 @@ public function print(PrintContext $context): string
9496
$this->name,
9597
);
9698

99+
$fetchCode .= $this->inSnippetArea
100+
? 'if ($ʟ_tmp instanceof Nette\Application\UI\Renderable) $ʟ_tmp->snippetMode = $this->global->uiControl->snippetMode;'
101+
: 'if ($ʟ_tmp instanceof Nette\Application\UI\Renderable) $ʟ_tmp->redrawControl(null, false);';
102+
97103
if ($this->escape) {
98104
return $context->format(
99105
<<<'XX'
100106
%raw
101-
if ($ʟ_tmp instanceof Nette\Application\UI\Renderable) $ʟ_tmp->redrawControl(null, false);
102107
ob_start(fn() => '');
103108
$ʟ_tmp->%raw(%args) %line;
104109
$ʟ_fi = new LR\FilterInfo(%dump); echo %modifyContent(ob_get_clean());
@@ -117,7 +122,6 @@ public function print(PrintContext $context): string
117122
return $context->format(
118123
<<<'XX'
119124
%raw
120-
if ($ʟ_tmp instanceof Nette\Application\UI\Renderable) $ʟ_tmp->redrawControl(null, false);
121125
$ʟ_tmp->%raw(%args) %line;
122126

123127

src/Bridges/ApplicationLatte/SnippetRuntime.php

+6-6
Original file line numberDiff line numberDiff line change
@@ -64,10 +64,11 @@ public function enter(string $name, string $type): void
6464
$this->nestingLevel++;
6565
}
6666

67-
$this->stack[] = [$name, $obStarted];
68-
if ($name !== '') {
69-
$this->control->redrawControl($name, false);
67+
$this->stack[] = [$name, $obStarted, $this->control->snippetMode];
68+
if ($type !== self::TypeArea) {
69+
$this->control->snippetMode = false;
7070
}
71+
$this->control->redrawControl($name, false);
7172
}
7273

7374

@@ -78,7 +79,7 @@ public function leave(): void
7879
return;
7980
}
8081

81-
[$name, $obStarted] = array_pop($this->stack);
82+
[$name, $obStarted, $this->control->snippetMode] = array_pop($this->stack);
8283
if ($this->nestingLevel > 0 && --$this->nestingLevel === 0) {
8384
$content = ob_get_clean();
8485
$this->payload ??= $this->control->getPresenter()->getPayload();
@@ -107,7 +108,6 @@ public function renderSnippets(array $blocks, array $params): bool
107108
}
108109

109110
$this->renderingSnippets = true;
110-
$this->control->snippetMode = false;
111111
foreach ($blocks as $name => $block) {
112112
if (!$this->control->isControlInvalid($name)) {
113113
continue;
@@ -116,8 +116,8 @@ public function renderSnippets(array $blocks, array $params): bool
116116
$function = reset($block->functions);
117117
$function($params);
118118
}
119+
$this->renderingSnippets = false;
119120

120-
$this->control->snippetMode = true;
121121
$this->renderChildren();
122122
return true;
123123
}
+91
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
<?php
2+
3+
/**
4+
* Test: renderSnippets and control wrapped in a snippet
5+
* @phpVersion 8.0
6+
*/
7+
8+
declare(strict_types=1);
9+
10+
use Nette\Http;
11+
use Tester\Assert;
12+
13+
require __DIR__ . '/../bootstrap.php';
14+
15+
if (version_compare(Latte\Engine::VERSION, '3', '<')) {
16+
Tester\Environment::skip('Test for Latte 3');
17+
}
18+
19+
20+
class TestControl extends Nette\Application\UI\Control
21+
{
22+
public int $counter = 0;
23+
24+
25+
public function render(int $arg = 0)
26+
{
27+
$this->counter++;
28+
$latte = new Latte\Engine;
29+
$latte->setLoader(new Latte\Loaders\StringLoader);
30+
$latte->addExtension(new Nette\Bridges\ApplicationLatte\UIExtension($this));
31+
$latte->render('{snippet foo}hello {$arg}{/snippet}', ['arg' => $arg]);
32+
}
33+
}
34+
35+
class TestPresenter extends Nette\Application\UI\Presenter
36+
{
37+
public function createComponentTest()
38+
{
39+
return new TestControl;
40+
}
41+
42+
43+
public function render()
44+
{
45+
$latte = new Latte\Engine;
46+
$latte->setLoader(new Latte\Loaders\StringLoader);
47+
$latte->addExtension(new Nette\Bridges\ApplicationLatte\UIExtension($this));
48+
$latte->render('{snippetArea foo}{control test 123}{/snippetArea}');
49+
}
50+
}
51+
52+
53+
$presenter = new TestPresenter;
54+
$presenter->injectPrimary(null, null, null, new Http\Request(new Http\UrlScript('/')), new Http\Response);
55+
$presenter->snippetMode = true;
56+
$presenter->redrawControl('foo');
57+
$presenter['test']->redrawControl('foo');
58+
$presenter->render();
59+
Assert::same([
60+
'snippets' => [
61+
'snippet-test-foo' => 'hello 123',
62+
],
63+
], (array) $presenter->payload);
64+
Assert::same(1, $presenter['test']->counter);
65+
66+
67+
$presenter = new TestPresenter;
68+
$presenter->injectPrimary(null, null, null, new Http\Request(new Http\UrlScript('/')), new Http\Response);
69+
$presenter->snippetMode = true;
70+
$presenter->redrawControl('foo');
71+
$presenter['test']->redrawControl();
72+
$presenter->render();
73+
Assert::same([
74+
'snippets' => [
75+
'snippet-test-foo' => 'hello 123',
76+
],
77+
], (array) $presenter->payload);
78+
Assert::same(2, $presenter['test']->counter);
79+
80+
81+
$presenter = new TestPresenter;
82+
$presenter->injectPrimary(null, null, null, new Http\Request(new Http\UrlScript('/')), new Http\Response);
83+
$presenter->snippetMode = true;
84+
$presenter['test']->redrawControl('foo');
85+
$presenter->render();
86+
Assert::same([
87+
'snippets' => [
88+
'snippet-test-foo' => 'hello 0',
89+
],
90+
], (array) $presenter->payload);
91+
Assert::same(1, $presenter['test']->counter);

0 commit comments

Comments
 (0)