123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- <?php
- /*
- * This file is part of the Symfony package.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Symfony\Component\Console\Output;
- use Symfony\Component\Console\Formatter\OutputFormatterInterface;
- use Symfony\Component\Console\Helper\Helper;
- use Symfony\Component\Console\Terminal;
- /**
- * @author Pierre du Plessis <pdples@gmail.com>
- * @author Gabriel Ostrolucký <gabriel.ostrolucky@gmail.com>
- */
- class ConsoleSectionOutput extends StreamOutput
- {
- private array $content = [];
- private int $lines = 0;
- private array $sections;
- private Terminal $terminal;
- private int $maxHeight = 0;
- /**
- * @param resource $stream
- * @param ConsoleSectionOutput[] $sections
- */
- public function __construct($stream, array &$sections, int $verbosity, bool $decorated, OutputFormatterInterface $formatter)
- {
- parent::__construct($stream, $verbosity, $decorated, $formatter);
- array_unshift($sections, $this);
- $this->sections = &$sections;
- $this->terminal = new Terminal();
- }
- /**
- * Defines a maximum number of lines for this section.
- *
- * When more lines are added, the section will automatically scroll to the
- * end (i.e. remove the first lines to comply with the max height).
- */
- public function setMaxHeight(int $maxHeight): void
- {
- // when changing max height, clear output of current section and redraw again with the new height
- $previousMaxHeight = $this->maxHeight;
- $this->maxHeight = $maxHeight;
- $existingContent = $this->popStreamContentUntilCurrentSection($previousMaxHeight ? min($previousMaxHeight, $this->lines) : $this->lines);
- parent::doWrite($this->getVisibleContent(), false);
- parent::doWrite($existingContent, false);
- }
- /**
- * Clears previous output for this section.
- *
- * @param int $lines Number of lines to clear. If null, then the entire output of this section is cleared
- *
- * @return void
- */
- public function clear(?int $lines = null)
- {
- if (empty($this->content) || !$this->isDecorated()) {
- return;
- }
- if ($lines) {
- array_splice($this->content, -$lines);
- } else {
- $lines = $this->lines;
- $this->content = [];
- }
- $this->lines -= $lines;
- parent::doWrite($this->popStreamContentUntilCurrentSection($this->maxHeight ? min($this->maxHeight, $lines) : $lines), false);
- }
- /**
- * Overwrites the previous output with a new message.
- *
- * @return void
- */
- public function overwrite(string|iterable $message)
- {
- $this->clear();
- $this->writeln($message);
- }
- public function getContent(): string
- {
- return implode('', $this->content);
- }
- public function getVisibleContent(): string
- {
- if (0 === $this->maxHeight) {
- return $this->getContent();
- }
- return implode('', \array_slice($this->content, -$this->maxHeight));
- }
- /**
- * @internal
- */
- public function addContent(string $input, bool $newline = true): int
- {
- $width = $this->terminal->getWidth();
- $lines = explode(\PHP_EOL, $input);
- $linesAdded = 0;
- $count = \count($lines) - 1;
- foreach ($lines as $i => $lineContent) {
- // re-add the line break (that has been removed in the above `explode()` for
- // - every line that is not the last line
- // - if $newline is required, also add it to the last line
- if ($i < $count || $newline) {
- $lineContent .= \PHP_EOL;
- }
- // skip line if there is no text (or newline for that matter)
- if ('' === $lineContent) {
- continue;
- }
- // For the first line, check if the previous line (last entry of `$this->content`)
- // needs to be continued (i.e. does not end with a line break).
- if (0 === $i
- && (false !== $lastLine = end($this->content))
- && !str_ends_with($lastLine, \PHP_EOL)
- ) {
- // deduct the line count of the previous line
- $this->lines -= (int) ceil($this->getDisplayLength($lastLine) / $width) ?: 1;
- // concatenate previous and new line
- $lineContent = $lastLine.$lineContent;
- // replace last entry of `$this->content` with the new expanded line
- array_splice($this->content, -1, 1, $lineContent);
- } else {
- // otherwise just add the new content
- $this->content[] = $lineContent;
- }
- $linesAdded += (int) ceil($this->getDisplayLength($lineContent) / $width) ?: 1;
- }
- $this->lines += $linesAdded;
- return $linesAdded;
- }
- /**
- * @internal
- */
- public function addNewLineOfInputSubmit(): void
- {
- $this->content[] = \PHP_EOL;
- ++$this->lines;
- }
- /**
- * @return void
- */
- protected function doWrite(string $message, bool $newline)
- {
- // Simulate newline behavior for consistent output formatting, avoiding extra logic
- if (!$newline && str_ends_with($message, \PHP_EOL)) {
- $message = substr($message, 0, -\strlen(\PHP_EOL));
- $newline = true;
- }
- if (!$this->isDecorated()) {
- parent::doWrite($message, $newline);
- return;
- }
- // Check if the previous line (last entry of `$this->content`) needs to be continued
- // (i.e. does not end with a line break). In which case, it needs to be erased first.
- $linesToClear = $deleteLastLine = ($lastLine = end($this->content) ?: '') && !str_ends_with($lastLine, \PHP_EOL) ? 1 : 0;
- $linesAdded = $this->addContent($message, $newline);
- if ($lineOverflow = $this->maxHeight > 0 && $this->lines > $this->maxHeight) {
- // on overflow, clear the whole section and redraw again (to remove the first lines)
- $linesToClear = $this->maxHeight;
- }
- $erasedContent = $this->popStreamContentUntilCurrentSection($linesToClear);
- if ($lineOverflow) {
- // redraw existing lines of the section
- $previousLinesOfSection = \array_slice($this->content, $this->lines - $this->maxHeight, $this->maxHeight - $linesAdded);
- parent::doWrite(implode('', $previousLinesOfSection), false);
- }
- // if the last line was removed, re-print its content together with the new content.
- // otherwise, just print the new content.
- parent::doWrite($deleteLastLine ? $lastLine.$message : $message, true);
- parent::doWrite($erasedContent, false);
- }
- /**
- * At initial stage, cursor is at the end of stream output. This method makes cursor crawl upwards until it hits
- * current section. Then it erases content it crawled through. Optionally, it erases part of current section too.
- */
- private function popStreamContentUntilCurrentSection(int $numberOfLinesToClearFromCurrentSection = 0): string
- {
- $numberOfLinesToClear = $numberOfLinesToClearFromCurrentSection;
- $erasedContent = [];
- foreach ($this->sections as $section) {
- if ($section === $this) {
- break;
- }
- $numberOfLinesToClear += $section->maxHeight ? min($section->lines, $section->maxHeight) : $section->lines;
- if ('' !== $sectionContent = $section->getVisibleContent()) {
- if (!str_ends_with($sectionContent, \PHP_EOL)) {
- $sectionContent .= \PHP_EOL;
- }
- $erasedContent[] = $sectionContent;
- }
- }
- if ($numberOfLinesToClear > 0) {
- // move cursor up n lines
- parent::doWrite(sprintf("\x1b[%dA", $numberOfLinesToClear), false);
- // erase to end of screen
- parent::doWrite("\x1b[0J", false);
- }
- return implode('', array_reverse($erasedContent));
- }
- private function getDisplayLength(string $text): int
- {
- return Helper::width(Helper::removeDecoration($this->getFormatter(), str_replace("\t", ' ', $text)));
- }
- }
|