_active)) { throw new Exception(sprintf("A view block with the name '%s' is already/still open.", $name)); } $this->_active[$name] = $mode; ob_start(); } /** * End a capturing block. The compliment to ViewBlock::start() * * @return void * @see \Cake\View\ViewBlock::start() */ public function end(): void { if ($this->_discardActiveBufferOnEnd) { $this->_discardActiveBufferOnEnd = false; ob_end_clean(); return; } if ($this->_active) { $mode = end($this->_active); $active = key($this->_active); $content = ob_get_clean(); if ($mode === ViewBlock::OVERRIDE) { $this->_blocks[$active] = $content; } else { $this->concat($active, $content, $mode); } array_pop($this->_active); } } /** * Concat content to an existing or new block. * Concating to a new block will create the block. * * Calling concat() without a value will create a new capturing * block that needs to be finished with View::end(). The content * of the new capturing context will be added to the existing block context. * * @param string $name Name of the block * @param mixed $value The content for the block. Value will be type cast * to string. * @param string $mode If ViewBlock::APPEND content will be appended to existing content. * If ViewBlock::PREPEND it will be prepended. * @return void */ public function concat(string $name, $value = null, $mode = ViewBlock::APPEND): void { if ($value === null) { $this->start($name, $mode); return; } if (!isset($this->_blocks[$name])) { $this->_blocks[$name] = ''; } if ($mode === ViewBlock::PREPEND) { $this->_blocks[$name] = $value . $this->_blocks[$name]; } else { $this->_blocks[$name] .= $value; } } /** * Set the content for a block. This will overwrite any * existing content. * * @param string $name Name of the block * @param mixed $value The content for the block. Value will be type cast * to string. * @return void */ public function set(string $name, $value): void { $this->_blocks[$name] = (string)$value; } /** * Get the content for a block. * * @param string $name Name of the block * @param string $default Default string * @return string The block content or $default if the block does not exist. */ public function get(string $name, string $default = ''): string { if (!isset($this->_blocks[$name])) { return $default; } return $this->_blocks[$name]; } /** * Check if a block exists * * @param string $name Name of the block * @return bool */ public function exists(string $name): bool { return isset($this->_blocks[$name]); } /** * Get the names of all the existing blocks. * * @return string[] An array containing the blocks. */ public function keys(): array { return array_keys($this->_blocks); } /** * Get the name of the currently open block. * * @return string|null Either null or the name of the last open block. */ public function active(): ?string { end($this->_active); return key($this->_active); } /** * Get the unclosed/active blocks. Key is name, value is mode. * * @return string[] An array of unclosed blocks. */ public function unclosed(): array { return $this->_active; } }