-
Notifications
You must be signed in to change notification settings - Fork 118
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Display
show_cmds
's output in a pager when in TTY environment
This can: - Make it easier to scroll up and down the commands list - Avoid pushing up users' previous output - Allow users to do basic search with `/<word>`
- Loading branch information
Showing
4 changed files
with
89 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
# frozen_string_literal: true | ||
|
||
module IRB | ||
# The implementation of this class is borrowed from RDoc's lib/rdoc/ri/driver.rb. | ||
# Please do NOT use this class directly outside of IRB. | ||
class Pager | ||
PAGE_COMMANDS = [ENV['RI_PAGER'], ENV['PAGER'], 'pager', 'less -r', 'more -r'].compact.uniq | ||
|
||
class << self | ||
def page | ||
if STDIN.tty? && pager = setup_pager | ||
begin | ||
pid = pager.pid | ||
yield pager | ||
ensure | ||
pager.close | ||
end | ||
else | ||
yield $stdout | ||
end | ||
# When user presses Ctrl-C, IRB would raise `IRB::Abort` | ||
# But since Pager is implemented by running paging commands like `less` in another process with `IO.popen`, | ||
# the `IRB::Abort` exception only interrupts IRB's execution but doesn't affect the pager | ||
# So to properly terminate the pager with Ctrl-C, we need to catch `IRB::Abort` and kill the pager process | ||
rescue IRB::Abort | ||
Process.kill("TERM", pid) if pid | ||
nil | ||
rescue Errno::EPIPE | ||
end | ||
|
||
private | ||
|
||
def setup_pager | ||
require 'shellwords' | ||
|
||
PAGE_COMMANDS.each do |pager| | ||
pager = Shellwords.split(pager) | ||
next if pager.empty? | ||
|
||
begin | ||
io = IO.popen(pager, 'w') | ||
rescue | ||
next | ||
end | ||
|
||
if $? && $?.pid == io.pid && $?.exited? # pager didn't work | ||
next | ||
end | ||
|
||
return io | ||
end | ||
|
||
nil | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters