2016-05-24 9 views
0

Прежде всего, я новичок в python. Должен был пройти курс обучения в колледже и получил свою эффективность.Почему запрос Windows не отвечает после использования окна python curses?

У меня есть эта липкая проблема, когда приглашение Windows 7 перестает отвечать на запросы после использования окна curses. В Windows 10 он работает хорошо. Обратите внимание, что я использую терминал Win7 с настройками по умолчанию. В моем коде я создаю окно curses, чтобы показать 2 одновременных индикатора выполнения, каждый для загрузки файла. Я реализовал это, передав окно curses в класс FileDownload (один экземпляр класса для каждой загрузки), который обрабатывает свой индикатор выполнения внутри этого окна. Как ни странно, в Windows 7, когда загрузка выполнена, и элемент управления возвращается в приглашение, он перестает отвечать на клавиатуру. Я работал над этим, вызывая curses.endwin() после использования окна, но это заставляет подсказку отображать весь путь вниз по экрану, что скрывает окно проклятий.

Вот мой код. Любые идеи приветствуются. Благодаря!

# Skeleton version for simulations. 
# Downloads 2 files simultaneously and shows a progress bar for each. 
# Each file download is a FileDownload object that interacts with a 
# common curses window passed as an argument. 

import requests, math, threading, curses, datetime 


class FileDownload: 

    def __init__(self, y_pos, window, url): 
     # Y position of the progress bar in the download queue window. 
     self.__bar_pos = int(y_pos) 
     self.__progress_window = window 
     self.__download_url = url 
     # Status of the file download object. 
     self.__status = "queued" 
     t = threading.Thread(target=self.__file_downloader) 
     t.start() 

    # Downloads selected file and handles its progress bar. 
    def __file_downloader(self): 
     file = requests.get(self.__download_url, stream=True) 
     self.__status = "downloading" 
     self.__progress_window.addstr(self.__bar_pos + 1, 1, "0%" + " " * 60 + "100%") 
     size = int(file.headers.get('content-length')) 
     win_prompt = "Downloading " + format(size, ",d") + " Bytes:" 
     self.__progress_window.addstr(self.__bar_pos, 1, win_prompt) 

     file_name = str(datetime.datetime.now().strftime("%Y-%m-%d_%H.%M.%d")) 
     dump = open(file_name, "wb") 

     # Progress bar length. 
     bar_space = 58 
     # Same as an index. 
     current_iteration = 0 
     # Beginning position of the progress bar. 
     progress_position = 4 
     # How many iterations will be needed (in chunks of 1 MB). 
     iterations = math.ceil(size/1024 ** 2) 

     # Downloads the file in 1MB chunks. 
     for block in file.iter_content(1024 ** 2): 
      dump.write(block) 

      # Progress bar controller. 
      current_iteration += 1 
      step = math.floor(bar_space/iterations) 
      if current_iteration > 1: 
       progress_position += step 
      if current_iteration == iterations: 
       step = bar_space - step * (current_iteration - 1) 
      # Updates the progress bar. 
      self.__progress_window.addstr(self.__bar_pos + 1, progress_position, 
              "#" * step) 
     dump.close() 
     self.__status = "downloaded" 

    # Returns the current status of the file download ("queued", "downloading" or 
    # "downloaded"). 
    def get_status(self): 
     return self.__status 


# Instantiates each file download. 
def files_downloader(): 

    # Creates curses window. 
    curses.initscr() 
    win = curses.newwin(8, 70) 
    win.border(0) 
    win.immedok(True) 

    # Download URLs. 
    urls = ["http://ipv4.download.thinkbroadband.com/10MB.zip", 
      "http://ipv4.download.thinkbroadband.com/5MB.zip"] 

    downloads_dct = {} 
    for n in range(len(urls)): 
     # Progress bar position in the window for the file. 
     y_pos = n * 4 + 1 
     downloads_dct[n + 1] = FileDownload(y_pos, win, urls[n]) 

    # Waits for all files to be downloaded before passing control of the terminal 
    # to the user. 
    all_downloaded = False 
    while not all_downloaded: 
     all_downloaded = True 
     for key, file_download in downloads_dct.items(): 
      if file_download.get_status() != "downloaded": 
       all_downloaded = False 

    # Prevents the prompt from returning inside the curses window. 
    win.addstr(7, 1, "-") 

    # This solves the unresponsive prompt issue but hides the curses window if the screen buffer 
    # is higher than the window size. 
    # curses.endwin() 

while input("\nEnter to continue: ") == "": 
    files_downloader() 
+0

стандартной библиотеки Python не реализует ругательства на Windows. Это может помочь, если вы сообщите людям, какая версия проклятий вы используете. – eryksun

+0

Я использую curses 2.2 в python 3.5.1. – R01k

ответ

1

Возможно, вы используете Cygwin (и Ncurses): Ncurses (как и любой другой реализации проклятия) изменения терминал режим ввода/вывода, когда он работает. Изменения, которые вы, вероятно, видя, что

  • входные символы не отображаются
  • вы должны ввести ControlJ до конца входной строки, а не просто Введите
  • выход не промывается автоматически в конце каждой строки

Это делает эти изменения, чтобы позволить ему читать одиночные символы, а также чтобы использовать терминал более эффективно.

Чтобы изменить назад в обычный режим ввода/вывода терминала, вы должны использовать функцию endwin. Функция reset_shell_mode также была бы полезна.

Дальнейшее чтение:

+0

reset_shell_mode сделал трюк. Спасибо! – R01k