2019-05-30 16:30:41 +02:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
namespace Psalm\Progress;
|
|
|
|
|
2021-12-03 21:07:25 +01:00
|
|
|
use function max;
|
|
|
|
use function microtime;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function str_repeat;
|
|
|
|
use function strlen;
|
|
|
|
|
2019-05-31 00:37:01 +02:00
|
|
|
class DefaultProgress extends LongProgress
|
2019-05-30 16:30:41 +02:00
|
|
|
{
|
2020-09-20 18:54:46 +02:00
|
|
|
private const TOO_MANY_FILES = 1500;
|
2019-05-30 16:30:41 +02:00
|
|
|
|
2019-06-28 20:25:47 +02:00
|
|
|
// Update the progress bar at most once per 0.1 seconds.
|
|
|
|
// This reduces flickering and reduces the amount of time spent writing to STDERR and updating the terminal.
|
2020-09-20 18:54:46 +02:00
|
|
|
private const PROGRESS_BAR_SAMPLE_INTERVAL = 0.1;
|
2019-06-28 20:25:47 +02:00
|
|
|
|
|
|
|
/** @var float the last time when the progress bar UI was updated */
|
|
|
|
private $previous_update_time = 0.0;
|
|
|
|
|
2019-06-03 17:20:42 +02:00
|
|
|
public function taskDone(int $level): void
|
2019-05-30 16:30:41 +02:00
|
|
|
{
|
2019-05-31 00:37:01 +02:00
|
|
|
if ($this->number_of_tasks > self::TOO_MANY_FILES) {
|
|
|
|
++$this->progress;
|
2019-05-30 16:30:41 +02:00
|
|
|
|
2019-06-28 20:25:47 +02:00
|
|
|
// Source for rate limiting:
|
|
|
|
// https://github.com/phan/phan/blob/9a788581ee1a4e1c35bebf89c435fd8a238c1d17/src/Phan/CLI.php
|
2021-12-03 21:07:25 +01:00
|
|
|
$time = microtime(true);
|
2019-06-28 20:25:47 +02:00
|
|
|
|
|
|
|
// If not enough time has elapsed, then don't update the progress bar.
|
|
|
|
// Making the update frequency based on time (instead of the number of files)
|
|
|
|
// prevents the terminal from rapidly flickering while processing small/empty files,
|
|
|
|
// and reduces the time spent writing to stderr.
|
|
|
|
if ($time - $this->previous_update_time < self::PROGRESS_BAR_SAMPLE_INTERVAL) {
|
|
|
|
// Make sure to output the section for 100% completion regardless of limits, to avoid confusion.
|
|
|
|
if ($this->progress !== $this->number_of_tasks) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$this->previous_update_time = $time;
|
|
|
|
|
2019-05-31 01:00:26 +02:00
|
|
|
$inner_progress = self::renderInnerProgressBar(
|
|
|
|
self::NUMBER_OF_COLUMNS,
|
|
|
|
$this->progress / $this->number_of_tasks
|
|
|
|
);
|
2019-05-30 16:30:41 +02:00
|
|
|
|
2019-05-31 01:00:26 +02:00
|
|
|
$this->write($inner_progress . ' ' . $this->getOverview() . "\r");
|
|
|
|
} else {
|
2019-06-03 17:20:42 +02:00
|
|
|
parent::taskDone($level);
|
2019-05-31 01:00:26 +02:00
|
|
|
}
|
|
|
|
}
|
2019-05-30 16:30:41 +02:00
|
|
|
|
2019-05-31 01:00:26 +02:00
|
|
|
/**
|
|
|
|
* Fully stolen from
|
|
|
|
* https://github.com/phan/phan/blob/d61a624b1384ea220f39927d53fd656a65a75fac/src/Phan/CLI.php
|
|
|
|
* Renders a unicode progress bar that goes from light (left) to dark (right)
|
|
|
|
* The length in the console is the positive integer $length
|
2019-07-05 22:24:00 +02:00
|
|
|
*
|
2019-05-31 01:00:26 +02:00
|
|
|
* @see https://en.wikipedia.org/wiki/Block_Elements
|
|
|
|
*/
|
2021-12-05 18:51:26 +01:00
|
|
|
private static function renderInnerProgressBar(int $length, float $p): string
|
2019-05-31 01:00:26 +02:00
|
|
|
{
|
|
|
|
$current_float = $p * $length;
|
|
|
|
$current = (int)$current_float;
|
2021-12-03 21:07:25 +01:00
|
|
|
$rest = max($length - $current, 0);
|
2019-05-30 16:30:41 +02:00
|
|
|
|
2019-05-31 01:00:26 +02:00
|
|
|
if (!self::doesTerminalSupportUtf8()) {
|
|
|
|
// Show a progress bar of "XXXX>------" in Windows when utf-8 is unsupported.
|
2019-07-05 22:24:00 +02:00
|
|
|
$progress_bar = str_repeat('X', $current);
|
2019-05-31 01:00:26 +02:00
|
|
|
$delta = $current_float - $current;
|
|
|
|
if ($delta > 0.5) {
|
2019-07-05 22:24:00 +02:00
|
|
|
$progress_bar .= '>' . str_repeat('-', $rest - 1);
|
2019-05-31 01:00:26 +02:00
|
|
|
} else {
|
2019-07-05 22:24:00 +02:00
|
|
|
$progress_bar .= str_repeat('-', $rest);
|
2019-05-31 01:00:26 +02:00
|
|
|
}
|
2019-07-05 22:24:00 +02:00
|
|
|
|
2019-05-31 01:00:26 +02:00
|
|
|
return $progress_bar;
|
|
|
|
}
|
2019-05-30 16:30:41 +02:00
|
|
|
|
2019-05-31 01:00:26 +02:00
|
|
|
// The left-most characters are "Light shade"
|
|
|
|
$progress_bar = str_repeat("\u{2588}", $current);
|
|
|
|
$delta = $current_float - $current;
|
2019-05-31 17:55:47 +02:00
|
|
|
if ($delta > 3.0 / 4) {
|
|
|
|
$progress_bar .= "\u{258A}" . str_repeat("\u{2591}", $rest - 1);
|
|
|
|
} elseif ($delta > 2.0 / 4) {
|
|
|
|
$progress_bar .= "\u{258C}" . str_repeat("\u{2591}", $rest - 1);
|
|
|
|
} elseif ($delta > 1.0 / 4) {
|
|
|
|
$progress_bar .= "\u{258E}" . str_repeat("\u{2591}", $rest - 1);
|
2019-05-31 00:37:01 +02:00
|
|
|
} else {
|
2019-05-31 01:00:26 +02:00
|
|
|
$progress_bar .= str_repeat("\u{2591}", $rest);
|
2019-05-30 16:30:41 +02:00
|
|
|
}
|
2019-05-31 01:00:26 +02:00
|
|
|
|
|
|
|
return $progress_bar;
|
2019-05-30 16:30:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public function finish(): void
|
|
|
|
{
|
2019-05-31 00:37:01 +02:00
|
|
|
if ($this->number_of_tasks > self::TOO_MANY_FILES) {
|
|
|
|
$this->write(str_repeat(' ', self::NUMBER_OF_COLUMNS + strlen($this->getOverview()) + 1) . "\r");
|
|
|
|
} else {
|
|
|
|
parent::finish();
|
2019-05-30 16:30:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|