程序退出后,使curses程序输出在终端回滚历史记录中保持不变

时间:2013-01-21 20:03:51

标签: python ncurses curses

我对诅咒很陌生,所以我在python中尝试了一些不同的东西。

我已初始化窗口并为window对象设置了scrollok。我可以添加字符串,并且滚动工作使得addstr()在窗口的末尾没有任何错误。

我希望能够在程序完成后在终端程序(tmux或KDE Konsole,在本例中)中滚动回程序输出。

在我的代码中,如果我跳过endwin()调用,我至少可以看到输出,但终端需要重置调用才能恢复运行。

此外,即使在程序运行时,在curses窗口向下滚动之后,我也无法在Konsole中向后滚动以查看初始输出。

#!/usr/bin/env python2
import curses
import time
win = curses.initscr()
win.scrollok(True)
(h,w)=win.getmaxyx()
h = h + 10
while h > 0:
    win.addstr("[h=%d] This is a sample string.  After 1 second, it will be lost\n" % h)
    h = h - 1
    win.refresh()
    time.sleep(0.05)
time.sleep(1.0)
curses.endwin()

1 个答案:

答案 0 :(得分:8)

对于此任务,我建议您使用打击垫(http://docs.python.org/2/library/curses.html#curses.newpad):

  

垫子就像一个窗口,除了它不受屏幕尺寸的限制,并且不一定与屏幕的特定部分相关联。 [...]一次只有一部分窗口会出现在屏幕上。 [...]

为了在完成使用curses之后将pad的内容留在控制台上,我会从pad中读回内容,结束curses并将内容写入标准输出。

以下代码实现了您所描述的内容。

#!/usr/bin/env python2

import curses
import time

# Create curses screen
scr = curses.initscr()
scr.keypad(True)
scr.refresh()
curses.noecho()

# Get screen width/height
height,width = scr.getmaxyx()

# Create a curses pad (pad size is height + 10)
mypad_height = height + 10
mypad = curses.newpad(mypad_height, width);
mypad.scrollok(True)
mypad_pos = 0
mypad_refresh = lambda: mypad.refresh(mypad_pos, 0, 0, 0, height-1, width)
mypad_refresh()

# Fill the window with text (note that 5 lines are lost forever)
for i in range(0, height + 15):
    mypad.addstr("{0} This is a sample string...\n".format(i))
    if i > height: mypad_pos = min(i - height, mypad_height - height)
    mypad_refresh()
    time.sleep(0.05)

# Wait for user to scroll or quit
running = True
while running:
    ch = scr.getch()
    if ch == curses.KEY_DOWN and mypad_pos < mypad_height - height:
        mypad_pos += 1
        mypad_refresh()
    elif ch == curses.KEY_UP and mypad_pos > 0:
        mypad_pos -= 1
        mypad_refresh()
    elif ch < 256 and chr(ch) == 'q':
        running = False

# Store the current contents of pad
mypad_contents = []
for i in range(0, mypad_height):
    mypad_contents.append(mypad.instr(i, 0))

# End curses
curses.endwin()

# Write the old contents of pad to console
print '\n'.join(mypad_contents)