mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
369 lines
9.4 KiB
Ruby
369 lines
9.4 KiB
Ruby
require 'io/console'
|
|
require 'timeout'
|
|
require 'reline/version'
|
|
require 'reline/config'
|
|
require 'reline/key_actor'
|
|
require 'reline/key_stroke'
|
|
require 'reline/line_editor'
|
|
|
|
module Reline
|
|
Key = Struct.new('Key', :char, :combined_char, :with_meta)
|
|
|
|
extend self
|
|
FILENAME_COMPLETION_PROC = nil
|
|
USERNAME_COMPLETION_PROC = nil
|
|
|
|
if RUBY_PLATFORM =~ /mswin|mingw/
|
|
IS_WINDOWS = true
|
|
else
|
|
IS_WINDOWS = false
|
|
end
|
|
|
|
CursorPos = Struct.new(:x, :y)
|
|
|
|
@@config = Reline::Config.new
|
|
@@line_editor = Reline::LineEditor.new(@@config)
|
|
@@ambiguous_width = nil
|
|
|
|
HISTORY = Class.new(Array) {
|
|
def to_s
|
|
'HISTORY'
|
|
end
|
|
|
|
def delete_at(index)
|
|
index = check_index(index)
|
|
super(index)
|
|
end
|
|
|
|
def [](index)
|
|
index = check_index(index)
|
|
super(index)
|
|
end
|
|
|
|
def []=(index, val)
|
|
index = check_index(index)
|
|
super(index, String.new(val, encoding: Encoding::default_external))
|
|
end
|
|
|
|
def push(*val)
|
|
super(*(val.map{ |v| String.new(v, encoding: Encoding::default_external) }))
|
|
end
|
|
|
|
def <<(val)
|
|
super(String.new(val, encoding: Encoding::default_external))
|
|
end
|
|
|
|
private def check_index(index)
|
|
index += size if index < 0
|
|
raise RangeError.new("index=<#{index}>") if index < -@@config.history_size or @@config.history_size < index
|
|
raise IndexError.new("index=<#{index}>") if index < 0 or size <= index
|
|
index
|
|
end
|
|
}.new
|
|
|
|
@@completion_append_character = nil
|
|
def self.completion_append_character
|
|
@@completion_append_character
|
|
end
|
|
def self.completion_append_character=(val)
|
|
if val.nil?
|
|
@@completion_append_character = nil
|
|
elsif val.size == 1
|
|
@@completion_append_character = val.encode(Encoding::default_external)
|
|
elsif val.size > 1
|
|
@@completion_append_character = val[0].encode(Encoding::default_external)
|
|
else
|
|
@@completion_append_character = nil
|
|
end
|
|
end
|
|
|
|
@@basic_word_break_characters = " \t\n`><=;|&{("
|
|
def self.basic_word_break_characters
|
|
@@basic_word_break_characters
|
|
end
|
|
def self.basic_word_break_characters=(v)
|
|
@@basic_word_break_characters = v.encode(Encoding::default_external)
|
|
end
|
|
|
|
@@completer_word_break_characters = @@basic_word_break_characters.dup
|
|
def self.completer_word_break_characters
|
|
@@completer_word_break_characters
|
|
end
|
|
def self.completer_word_break_characters=(v)
|
|
@@completer_word_break_characters = v.encode(Encoding::default_external)
|
|
end
|
|
|
|
@@basic_quote_characters = '"\''
|
|
def self.basic_quote_characters
|
|
@@basic_quote_characters
|
|
end
|
|
def self.basic_quote_characters=(v)
|
|
@@basic_quote_characters = v.encode(Encoding::default_external)
|
|
end
|
|
|
|
@@completer_quote_characters = '"\''
|
|
def self.completer_quote_characters
|
|
@@completer_quote_characters
|
|
end
|
|
def self.completer_quote_characters=(v)
|
|
@@completer_quote_characters = v.encode(Encoding::default_external)
|
|
end
|
|
|
|
@@filename_quote_characters = ''
|
|
def self.filename_quote_characters
|
|
@@filename_quote_characters
|
|
end
|
|
def self.filename_quote_characters=(v)
|
|
@@filename_quote_characters = v.encode(Encoding::default_external)
|
|
end
|
|
|
|
@@special_prefixes = ''
|
|
def self.special_prefixes
|
|
@@special_prefixes
|
|
end
|
|
def self.special_prefixes=(v)
|
|
@@special_prefixes = v.encode(Encoding::default_external)
|
|
end
|
|
|
|
@@completion_case_fold = nil
|
|
def self.completion_case_fold
|
|
@@completion_case_fold
|
|
end
|
|
def self.completion_case_fold=(v)
|
|
@@completion_case_fold = v
|
|
end
|
|
|
|
@@completion_proc = nil
|
|
def self.completion_proc
|
|
@@completion_proc
|
|
end
|
|
def self.completion_proc=(p)
|
|
raise ArgumentError unless p.is_a?(Proc)
|
|
@@completion_proc = p
|
|
end
|
|
|
|
@@pre_input_hook = nil
|
|
def self.pre_input_hook
|
|
@@pre_input_hook
|
|
end
|
|
def self.pre_input_hook=(p)
|
|
@@pre_input_hook = p
|
|
end
|
|
|
|
@@dig_perfect_match_proc = nil
|
|
def self.dig_perfect_match_proc
|
|
@@dig_perfect_match_proc
|
|
end
|
|
def self.dig_perfect_match_proc=(p)
|
|
@@dig_perfect_match_proc = p
|
|
end
|
|
|
|
def self.insert_text(text)
|
|
@@line_editor&.insert_text(text)
|
|
self
|
|
end
|
|
|
|
def self.redisplay
|
|
@@line_editor&.rerender
|
|
end
|
|
|
|
def self.line_buffer
|
|
@@line_editor&.line
|
|
end
|
|
|
|
def self.point
|
|
@@line_editor ? @@line_editor.byte_pointer : 0
|
|
end
|
|
|
|
def self.point=(val)
|
|
@@line_editor.byte_pointer = val
|
|
end
|
|
|
|
def self.delete_text(start = nil, length = nil)
|
|
@@line_editor&.delete_text(start, length)
|
|
end
|
|
|
|
private_class_method def self.test_mode
|
|
remove_const('IOGate') if const_defined?('IOGate')
|
|
const_set('IOGate', Reline::GeneralIO)
|
|
end
|
|
|
|
def self.input=(val)
|
|
raise TypeError unless val.respond_to?(:getc) or val.nil?
|
|
if val.respond_to?(:getc)
|
|
if defined?(Reline::ANSI) and IOGate == Reline::ANSI
|
|
Reline::ANSI.input = val
|
|
elsif IOGate == Reline::GeneralIO
|
|
Reline::GeneralIO.input = val
|
|
end
|
|
end
|
|
end
|
|
|
|
@@output = STDOUT
|
|
def self.output=(val)
|
|
raise TypeError unless val.respond_to?(:write) or val.nil?
|
|
@@output = val
|
|
if defined?(Reline::ANSI) and IOGate == Reline::ANSI
|
|
Reline::ANSI.output = val
|
|
end
|
|
end
|
|
|
|
def self.vi_editing_mode
|
|
@@config.editing_mode = :vi_insert
|
|
nil
|
|
end
|
|
|
|
def self.emacs_editing_mode
|
|
@@config.editing_mode = :emacs
|
|
nil
|
|
end
|
|
|
|
def self.vi_editing_mode?
|
|
@@config.editing_mode_is?(:vi_insert, :vi_command)
|
|
end
|
|
|
|
def self.emacs_editing_mode?
|
|
@@config.editing_mode_is?(:emacs)
|
|
end
|
|
|
|
def self.get_screen_size
|
|
Reline::IOGate.get_screen_size
|
|
end
|
|
|
|
def retrieve_completion_block(line, byte_pointer)
|
|
break_regexp = /[#{Regexp.escape(@@basic_word_break_characters)}]/
|
|
before_pointer = line.byteslice(0, byte_pointer)
|
|
break_point = before_pointer.rindex(break_regexp)
|
|
if break_point
|
|
preposing = before_pointer[0..(break_point)]
|
|
block = before_pointer[(break_point + 1)..-1]
|
|
else
|
|
preposing = ''
|
|
block = before_pointer
|
|
end
|
|
postposing = line.byteslice(byte_pointer, line.bytesize)
|
|
[preposing, block, postposing]
|
|
end
|
|
|
|
def readmultiline(prompt = '', add_hist = false, &confirm_multiline_termination)
|
|
unless confirm_multiline_termination
|
|
raise ArgumentError.new('#readmultiline needs block to confirm multiline termination')
|
|
end
|
|
if block_given?
|
|
inner_readline(prompt, add_hist, true, &confirm_multiline_termination)
|
|
else
|
|
inner_readline(prompt, add_hist, true)
|
|
end
|
|
|
|
whole_buffer = @@line_editor.whole_buffer.dup
|
|
whole_buffer.taint
|
|
if add_hist and whole_buffer and whole_buffer.chomp.size > 0
|
|
Reline::HISTORY << whole_buffer
|
|
end
|
|
|
|
@@line_editor.reset_line if @@line_editor.whole_buffer.nil?
|
|
whole_buffer
|
|
end
|
|
|
|
def readline(prompt = '', add_hist = false)
|
|
inner_readline(prompt, add_hist, false)
|
|
|
|
line = @@line_editor.line.dup
|
|
line.taint
|
|
if add_hist and line and line.chomp.size > 0
|
|
Reline::HISTORY << line.chomp
|
|
end
|
|
|
|
@@line_editor.reset_line if @@line_editor.line.nil?
|
|
line
|
|
end
|
|
|
|
def inner_readline(prompt, add_hist, multiline, &confirm_multiline_termination)
|
|
if ENV['RELINE_STDERR_TTY']
|
|
$stderr.reopen(ENV['RELINE_STDERR_TTY'], 'w')
|
|
end
|
|
@@config.read
|
|
otio = Reline::IOGate.prep
|
|
|
|
may_req_ambiguous_char_width
|
|
@@line_editor.reset(prompt)
|
|
if multiline
|
|
@@line_editor.multiline_on
|
|
if block_given?
|
|
@@line_editor.confirm_multiline_termination_proc = confirm_multiline_termination
|
|
end
|
|
else
|
|
@@line_editor.multiline_off
|
|
end
|
|
@@line_editor.output = @@output
|
|
@@line_editor.completion_proc = @@completion_proc
|
|
@@line_editor.dig_perfect_match_proc = @@dig_perfect_match_proc
|
|
@@line_editor.pre_input_hook = @@pre_input_hook
|
|
@@line_editor.retrieve_completion_block = method(:retrieve_completion_block)
|
|
@@line_editor.rerender
|
|
|
|
if IS_WINDOWS
|
|
config = {
|
|
key_mapping: {
|
|
[224, 72] => :ed_prev_history, # ↑
|
|
[224, 80] => :ed_next_history, # ↓
|
|
[224, 77] => :ed_next_char, # →
|
|
[224, 75] => :ed_prev_char, # ←
|
|
[224, 83] => :key_delete # Del
|
|
}
|
|
}
|
|
else
|
|
config = {
|
|
key_mapping: {
|
|
[27, 91, 65] => :ed_prev_history, # ↑
|
|
[27, 91, 66] => :ed_next_history, # ↓
|
|
[27, 91, 67] => :ed_next_char, # →
|
|
[27, 91, 68] => :ed_prev_char, # ←
|
|
[27, 91, 51, 126] => :key_delete # Del
|
|
}
|
|
}
|
|
end
|
|
|
|
key_stroke = Reline::KeyStroke.new(config)
|
|
begin
|
|
loop do
|
|
key_stroke.read_io(@@config.keyseq_timeout) { |inputs|
|
|
inputs.each { |c|
|
|
@@line_editor.input_key(c)
|
|
@@line_editor.rerender
|
|
}
|
|
}
|
|
break if @@line_editor.finished?
|
|
end
|
|
Reline::IOGate.move_cursor_column(0)
|
|
rescue StandardError => e
|
|
Reline::IOGate.deprep(otio)
|
|
raise e
|
|
end
|
|
|
|
Reline::IOGate.deprep(otio)
|
|
end
|
|
|
|
def may_req_ambiguous_char_width
|
|
@@ambiguous_width = 2 if Reline::IOGate == Reline::GeneralIO or STDOUT.is_a?(File)
|
|
return if @@ambiguous_width
|
|
Reline::IOGate.move_cursor_column(0)
|
|
print "\u{25bd}"
|
|
@@ambiguous_width = Reline::IOGate.cursor_pos.x
|
|
Reline::IOGate.move_cursor_column(0)
|
|
Reline::IOGate.erase_after_cursor
|
|
end
|
|
|
|
def self.ambiguous_width
|
|
@@ambiguous_width
|
|
end
|
|
end
|
|
|
|
if Reline::IS_WINDOWS
|
|
require 'reline/windows'
|
|
Reline::IOGate = Reline::Windows
|
|
else
|
|
require 'reline/ansi'
|
|
Reline::IOGate = Reline::ANSI
|
|
end
|
|
require 'reline/general_io'
|