Как анимировать командную строку?
-
09-06-2019 - |
Вопрос
Мне всегда было интересно, как люди обновляют предыдущую строку в командной строке.отличным примером этого является использование команды wget в Linux.Он создает своего рода полосу загрузки ASCII, которая выглядит следующим образом:
[======> ] 37%
и конечно полоса загрузки перемещается и проценты меняются, но новую строку не делает.Я не могу понять, как это сделать.Может ли кто-нибудь указать мне в правильном направлении?
Решение
Я знаю два способа сделать это:
- Используйте escape-символ возврата ('\b'), чтобы стереть строку.
- Использовать
curses
пакет, если выбранный вами язык программирования имеет для него привязки.
И Google показал Escape-коды ANSI, что кажется хорошим способом.Для справки, вот функция на C++, позволяющая сделать это:
void DrawProgressBar(int len, double percent) {
cout << "\x1B[2K"; // Erase the entire current line.
cout << "\x1B[0E"; // Move to the beginning of the current line.
string progress;
for (int i = 0; i < len; ++i) {
if (i < static_cast<int>(len * percent)) {
progress += "=";
} else {
progress += " ";
}
}
cout << "[" << progress << "] " << (static_cast<int>(100 * percent)) << "%";
flush(cout); // Required.
}
Другие советы
Один из способов сделать это — неоднократно обновлять строку текста с учетом текущего прогресса.Например:
def status(percent):
sys.stdout.write("%3d%%\r" % percent)
sys.stdout.flush()
Обратите внимание, что я использовал sys.stdout.write
вместо print
(это Python), потому что print
автоматически печатает «
» (новая строка с возвратом каретки) в конце каждой строки.Мне просто нужен возврат каретки, который возвращает курсор в начало строки.Так же flush()
необходимо, потому что по умолчанию sys.stdout
сбрасывает вывод только после новой строки (или после заполнения буфера).
Секрет в том, чтобы печатать только вместо или в начале и строки.
называется возвратом каретки и перемещает курсор в начало строки.
n называется линейной подачей и перемещает курсор на следующей линии в консоли.Если вы используете только , вы перезаписываете ранее написанную строку.Итак, сначала напишите строку, подобную следующей:
[ ]
затем добавьте знак для каждой галочки
\r[= ]
\r[== ]
...
\r[==========]
и так далее.Вы можете использовать 10 символов, каждый из которых представляет 10%.Кроме того, если вы хотите отобразить сообщение после завершения, не забудьте также добавить достаточно белых символов, чтобы перезаписать ранее написанные знаки равенства, например:
\r[done ]
ниже мой ответ, используйте Windows APIКонсоли(Windows), кодирование C.
/*
* file: ProgressBarConsole.cpp
* description: a console progress bar Demo
* author: lijian <hustlijian@gmail.com>
* version: 1.0
* date: 2012-12-06
*/
#include <stdio.h>
#include <windows.h>
HANDLE hOut;
CONSOLE_SCREEN_BUFFER_INFO bInfo;
char charProgress[80] =
{"================================================================"};
char spaceProgress = ' ';
/*
* show a progress in the [row] line
* row start from 0 to the end
*/
int ProgressBar(char *task, int row, int progress)
{
char str[100];
int len, barLen,progressLen;
COORD crStart, crCurr;
GetConsoleScreenBufferInfo(hOut, &bInfo);
crCurr = bInfo.dwCursorPosition; //the old position
len = bInfo.dwMaximumWindowSize.X;
barLen = len - 17;//minus the extra char
progressLen = (int)((progress/100.0)*barLen);
crStart.X = 0;
crStart.Y = row;
sprintf(str,"%-10s[%-.*s>%*c]%3d%%", task,progressLen,charProgress, barLen-progressLen,spaceProgress,50);
#if 0 //use stdand libary
SetConsoleCursorPosition(hOut, crStart);
printf("%s\n", str);
#else
WriteConsoleOutputCharacter(hOut, str, len,crStart,NULL);
#endif
SetConsoleCursorPosition(hOut, crCurr);
return 0;
}
int main(int argc, char* argv[])
{
int i;
hOut = GetStdHandle(STD_OUTPUT_HANDLE);
GetConsoleScreenBufferInfo(hOut, &bInfo);
for (i=0;i<100;i++)
{
ProgressBar("test", 0, i);
Sleep(50);
}
return 0;
}
В PowerShell есть командлет Write-Progress, который создает в консоли индикатор выполнения, который можно обновлять и изменять по мере выполнения сценария.
Вот ответ на ваш вопрос...(питон)
def disp_status(timelapse, timeout):
if timelapse and timeout:
percent = 100 * (float(timelapse)/float(timeout))
sys.stdout.write("progress : ["+"*"*int(percent)+" "*(100-int(percent-1))+"]"+str(percent)+" %")
sys.stdout.flush()
stdout.write("\r \r")
В качестве продолжения Ответ Грега, вот расширенная версия его функции, позволяющая отображать многострочные сообщения;просто передайте список или кортеж строк, которые вы хотите отобразить/обновить.
def status(msgs):
assert isinstance(msgs, (list, tuple))
sys.stdout.write(''.join(msg + '\n' for msg in msgs[:-1]) + msgs[-1] + ('\x1b[A' * (len(msgs) - 1)) + '\r')
sys.stdout.flush()
Примечание:Я тестировал это только с помощью терминала Linux, поэтому ваши результаты могут отличаться в системах на базе Windows.
Если вы используете язык сценариев, вы можете использовать команду «tput cup», чтобы сделать это...P.S.Насколько я знаю, это касается только Linux/Unix...