Skip to content

Commit 576b6ce

Browse files
committed
Add the ability to append content to sections.
Closes #135.
1 parent 5874ba8 commit 576b6ce

File tree

5 files changed

+133
-8
lines changed

5 files changed

+133
-8
lines changed

docs/templates/sections.md

+14-2
Original file line numberDiff line numberDiff line change
@@ -7,11 +7,11 @@ title: Sections
77
Sections
88
========
99

10-
The `start()` and `stop()` functions allow you to build sections (or blocks) of content within your template, and instead of them being rendered directly, they are saved for use elsewhere. For example, in your [layout](/templates/layouts/) template.
10+
The `start()` and `stop` functions allow you to build sections (or blocks) of content within your template, and instead of them being rendered directly, they are saved for use elsewhere. For example, in your [layout](/templates/layouts/) template.
1111

1212
## Creating sections
1313

14-
You define the name of the section in the `start()` function, and end the section with the `stop()` function.
14+
You define the name of the section with the `start()` function. To end a section call the `stop()` function.
1515

1616
~~~ php
1717
<?php $this->start('welcome') ?>
@@ -22,6 +22,18 @@ You define the name of the section in the `start()` function, and end the sectio
2222
<?php $this->stop() ?>
2323
~~~
2424

25+
## Stacking section content
26+
27+
By default, when you render a section its content will overwrite any existing content for that section. However, it's possible to append (or stack) the content instead using the `push()` method. This can be useful for specifying any JavaScript libraries required by your child views.
28+
29+
~~~ php
30+
<?php $this->push('scripts') ?>
31+
<script src="example.js"></script>
32+
<?php $this->end() ?>
33+
~~~
34+
35+
<p class="message-notice">The <code>end()</code> function is simply an alias of <code>stop()</code>. These functions can be used interchangeably.</p>
36+
2537
## Accessing section content
2638

2739
Access rendered section content using the name you assigned in the `start()` method. This variable can be accessed from the current template and layout templates using the `section()` function.

example/templates/layout.php

+2
Original file line numberDiff line numberDiff line change
@@ -6,5 +6,7 @@
66

77
<?=$this->section('content')?>
88

9+
<?=$this->section('scripts')?>
10+
911
</body>
1012
</html>

example/templates/profile.php

+7-1
Original file line numberDiff line numberDiff line change
@@ -3,4 +3,10 @@
33
<h1>User Profile</h1>
44
<p>Hello, <?=$this->e($name)?>!</p>
55

6-
<?php $this->insert('sidebar') ?>
6+
<?php $this->insert('sidebar') ?>
7+
8+
<?php $this->push('scripts') ?>
9+
<script>
10+
// Some JavaScript
11+
</script>
12+
<?php $this->end() ?>

src/Template/Template.php

+46-5
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,18 @@ class Template
3636
*/
3737
protected $sections = array();
3838

39+
/**
40+
* The name of the section currently being rendered.
41+
* @var string
42+
*/
43+
protected $sectionName;
44+
45+
/**
46+
* Whether the section should be appended or not.
47+
* @var boolean
48+
*/
49+
protected $appendSection;
50+
3951
/**
4052
* The name of the template layout.
4153
* @var string
@@ -178,7 +190,7 @@ public function layout($name, array $data = array())
178190

179191
/**
180192
* Start a new section block.
181-
* @param string $name
193+
* @param string $name
182194
* @return null
183195
*/
184196
public function start($name)
@@ -189,26 +201,55 @@ public function start($name)
189201
);
190202
}
191203

192-
$this->sections[$name] = '';
204+
if ($this->sectionName) {
205+
throw new LogicException('You cannot nest sections within other sections.');
206+
}
207+
208+
$this->sectionName = $name;
193209

194210
ob_start();
195211
}
196212

213+
/**
214+
* Start a new append section block.
215+
* @param string $name
216+
* @return null
217+
*/
218+
public function push($name)
219+
{
220+
$this->appendSection = true;
221+
222+
$this->start($name);
223+
}
224+
197225
/**
198226
* Stop the current section block.
199227
* @return null
200228
*/
201229
public function stop()
202230
{
203-
if (empty($this->sections)) {
231+
if (is_null($this->sectionName)) {
204232
throw new LogicException(
205233
'You must start a section before you can stop it.'
206234
);
207235
}
208236

209-
end($this->sections);
237+
if (!isset($this->sections[$this->sectionName])) {
238+
$this->sections[$this->sectionName] = '';
239+
}
240+
241+
$this->sections[$this->sectionName] = $this->appendSection ? $this->sections[$this->sectionName] . ob_get_clean() : ob_get_clean();
242+
$this->sectionName = null;
243+
$this->appendSection = false;
244+
}
210245

211-
$this->sections[key($this->sections)] = ob_get_clean();
246+
/**
247+
* Alias of stop().
248+
* @return null
249+
*/
250+
public function end()
251+
{
252+
$this->stop();
212253
}
213254

214255
/**

tests/Template/TemplateTest.php

+64
Original file line numberDiff line numberDiff line change
@@ -151,6 +151,21 @@ public function testSection()
151151
$this->assertEquals($this->template->render(), 'Hello World');
152152
}
153153

154+
public function testReplaceSection()
155+
{
156+
vfsStream::create(
157+
array(
158+
'template.php' => implode('\n', array(
159+
'<?php $this->layout("layout")?><?php $this->start("test") ?>Hello World<?php $this->stop() ?>',
160+
'<?php $this->layout("layout")?><?php $this->start("test") ?>See this instead!<?php $this->stop() ?>',
161+
)),
162+
'layout.php' => '<?php echo $this->section("test") ?>',
163+
)
164+
);
165+
166+
$this->assertEquals($this->template->render(), 'See this instead!');
167+
}
168+
154169
public function testStartSectionWithInvalidName()
155170
{
156171
$this->setExpectedException('LogicException', 'The section name "content" is reserved.');
@@ -164,6 +179,19 @@ public function testStartSectionWithInvalidName()
164179
$this->template->render();
165180
}
166181

182+
public function testNestSectionWithinAnotherSection()
183+
{
184+
$this->setExpectedException('LogicException', 'You cannot nest sections within other sections.');
185+
186+
vfsStream::create(
187+
array(
188+
'template.php' => '<?php $this->start("section1") ?><?php $this->start("section2") ?>',
189+
)
190+
);
191+
192+
$this->template->render();
193+
}
194+
167195
public function testStopSectionBeforeStarting()
168196
{
169197
$this->setExpectedException('LogicException', 'You must start a section before you can stop it.');
@@ -198,6 +226,42 @@ public function testNullSection()
198226
$this->assertEquals($this->template->render(), 'NULL');
199227
}
200228

229+
public function testPushSection()
230+
{
231+
vfsStream::create(
232+
array(
233+
'template.php' => implode('\n', array(
234+
'<?php $this->layout("layout")?>',
235+
'<?php $this->push("scripts") ?><script src="example1.js"></script><?php $this->end() ?>',
236+
'<?php $this->push("scripts") ?><script src="example2.js"></script><?php $this->end() ?>',
237+
)),
238+
'layout.php' => '<?php echo $this->section("scripts") ?>',
239+
)
240+
);
241+
242+
$this->assertEquals($this->template->render(), '<script src="example1.js"></script><script src="example2.js"></script>');
243+
}
244+
245+
public function testPushWithMultipleSections()
246+
{
247+
vfsStream::create(
248+
array(
249+
'template.php' => implode('\n', array(
250+
'<?php $this->layout("layout")?>',
251+
'<?php $this->push("scripts") ?><script src="example1.js"></script><?php $this->end() ?>',
252+
'<?php $this->start("test") ?>test<?php $this->stop() ?>',
253+
'<?php $this->push("scripts") ?><script src="example2.js"></script><?php $this->end() ?>',
254+
)),
255+
'layout.php' => implode('\n', array(
256+
'<?php echo $this->section("test") ?>',
257+
'<?php echo $this->section("scripts") ?>',
258+
)),
259+
)
260+
);
261+
262+
$this->assertEquals($this->template->render(), 'test\n<script src="example1.js"></script><script src="example2.js"></script>');
263+
}
264+
201265
public function testFetchFunction()
202266
{
203267
vfsStream::create(

0 commit comments

Comments
 (0)