mirror of
				https://github.com/ruby/ruby.git
				synced 2022-11-09 12:17:21 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			473 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			473 lines
		
	
	
	
		
			12 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
 | 
						|
  @@key_stroke = Reline::KeyStroke.new(@@config)
 | 
						|
  @@line_editor = Reline::LineEditor.new(@@config)
 | 
						|
  @@ambiguous_width = nil
 | 
						|
 | 
						|
  HISTORY = Class.new(Array) {
 | 
						|
    def initialize(config)
 | 
						|
      @config = config
 | 
						|
    end
 | 
						|
 | 
						|
    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 concat(*val)
 | 
						|
      val.each do |v|
 | 
						|
        push(*v)
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    def push(*val)
 | 
						|
      diff = size + val.size - @config.history_size
 | 
						|
      if diff > 0
 | 
						|
        if diff <= size
 | 
						|
          shift(diff)
 | 
						|
        else
 | 
						|
          diff -= size
 | 
						|
          clear
 | 
						|
          val.shift(diff)
 | 
						|
        end
 | 
						|
      end
 | 
						|
      super(*(val.map{ |v| String.new(v, encoding: Encoding::default_external) }))
 | 
						|
    end
 | 
						|
 | 
						|
    def <<(val)
 | 
						|
      shift if size + 1 > @config.history_size
 | 
						|
      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
 | 
						|
 | 
						|
    private def set_config(config)
 | 
						|
      @config = config
 | 
						|
    end
 | 
						|
  }.new(@@config)
 | 
						|
 | 
						|
  @@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
 | 
						|
 | 
						|
  @@output_modifier_proc = nil
 | 
						|
  def self.output_modifier_proc
 | 
						|
    @@output_modifier_proc
 | 
						|
  end
 | 
						|
  def self.output_modifier_proc=(p)
 | 
						|
    raise ArgumentError unless p.is_a?(Proc)
 | 
						|
    @@output_modifier_proc = p
 | 
						|
  end
 | 
						|
 | 
						|
  @@prompt_proc = nil
 | 
						|
  def self.prompt_proc
 | 
						|
    @@prompt_proc
 | 
						|
  end
 | 
						|
  def self.prompt_proc=(p)
 | 
						|
    raise ArgumentError unless p.is_a?(Proc)
 | 
						|
    @@prompt_proc = p
 | 
						|
  end
 | 
						|
 | 
						|
  @@auto_indent_proc = nil
 | 
						|
  def self.auto_indent_proc
 | 
						|
    @@auto_indent_proc
 | 
						|
  end
 | 
						|
  def self.auto_indent_proc=(p)
 | 
						|
    raise ArgumentError unless p.is_a?(Proc)
 | 
						|
    @@auto_indent_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)
 | 
						|
    raise ArgumentError unless p.is_a?(Proc)
 | 
						|
    @@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)
 | 
						|
    @@config.instance_variable_set(:@test_mode, true)
 | 
						|
    @@config.reset
 | 
						|
  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 eof?
 | 
						|
    @@line_editor.eof?
 | 
						|
  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
 | 
						|
    inner_readline(prompt, add_hist, true, &confirm_multiline_termination)
 | 
						|
 | 
						|
    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')
 | 
						|
      $stderr.sync = true
 | 
						|
      $stderr.puts "Reline is used by #{Process.pid}"
 | 
						|
    end
 | 
						|
    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.output_modifier_proc = @@output_modifier_proc
 | 
						|
    @@line_editor.prompt_proc = @@prompt_proc
 | 
						|
    @@line_editor.auto_indent_proc = @@auto_indent_proc
 | 
						|
    @@line_editor.dig_perfect_match_proc = @@dig_perfect_match_proc
 | 
						|
    @@line_editor.pre_input_hook = @@pre_input_hook
 | 
						|
    @@line_editor.rerender
 | 
						|
 | 
						|
    unless @@config.test_mode
 | 
						|
      @@config.read
 | 
						|
      @@config.reset_default_key_bindings
 | 
						|
      Reline::IOGate::RAW_KEYSTROKE_CONFIG.each_pair do |key, func|
 | 
						|
        @@config.add_default_key_binding(key, func)
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    begin
 | 
						|
      loop do
 | 
						|
        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
 | 
						|
      @@line_editor.finalize
 | 
						|
      Reline::IOGate.deprep(otio)
 | 
						|
      raise e
 | 
						|
    end
 | 
						|
 | 
						|
    @@line_editor.finalize
 | 
						|
    Reline::IOGate.deprep(otio)
 | 
						|
  end
 | 
						|
 | 
						|
  # Keystrokes of GNU Readline will timeout it with the specification of
 | 
						|
  # "keyseq-timeout" when waiting for the 2nd character after the 1st one.
 | 
						|
  # If the 2nd character comes after 1st ESC without timeout it has a
 | 
						|
  # meta-property of meta-key to discriminate modified key with meta-key
 | 
						|
  # from multibyte characters that come with 8th bit on.
 | 
						|
  #
 | 
						|
  # GNU Readline will wait for the 2nd character with "keyseq-timeout"
 | 
						|
  # milli-seconds but wait forever after 3rd characters.
 | 
						|
  def read_io(keyseq_timeout, &block)
 | 
						|
    buffer = []
 | 
						|
    loop do
 | 
						|
      c = Reline::IOGate.getc
 | 
						|
      buffer << c
 | 
						|
      result = @@key_stroke.match_status(buffer)
 | 
						|
      case result
 | 
						|
      when :matched
 | 
						|
        block.(@@key_stroke.expand(buffer).map{ |c| Reline::Key.new(c, c, false) })
 | 
						|
        break
 | 
						|
      when :matching
 | 
						|
        if buffer.size == 1
 | 
						|
          begin
 | 
						|
            succ_c = nil
 | 
						|
            Timeout.timeout(keyseq_timeout / 1000.0) {
 | 
						|
              succ_c = Reline::IOGate.getc
 | 
						|
            }
 | 
						|
          rescue Timeout::Error # cancel matching only when first byte
 | 
						|
            block.([Reline::Key.new(c, c, false)])
 | 
						|
            break
 | 
						|
          else
 | 
						|
            if @@key_stroke.match_status(buffer.dup.push(succ_c)) == :unmatched
 | 
						|
              if c == "\e".ord
 | 
						|
                block.([Reline::Key.new(succ_c, succ_c | 0b10000000, true)])
 | 
						|
              else
 | 
						|
                block.([Reline::Key.new(c, c, false), Reline::Key.new(succ_c, succ_c, false)])
 | 
						|
              end
 | 
						|
              break
 | 
						|
            else
 | 
						|
              Reline::IOGate.ungetc(succ_c)
 | 
						|
            end
 | 
						|
          end
 | 
						|
        end
 | 
						|
      when :unmatched
 | 
						|
        if buffer.size == 1 and c == "\e".ord
 | 
						|
          read_escaped_key(keyseq_timeout, buffer, block)
 | 
						|
        else
 | 
						|
          block.(buffer.map{ |c| Reline::Key.new(c, c, false) })
 | 
						|
        end
 | 
						|
        break
 | 
						|
      end
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def read_escaped_key(keyseq_timeout, buffer, block)
 | 
						|
    begin
 | 
						|
      escaped_c = nil
 | 
						|
      Timeout.timeout(keyseq_timeout / 1000.0) {
 | 
						|
        escaped_c = Reline::IOGate.getc
 | 
						|
      }
 | 
						|
    rescue Timeout::Error # independent ESC
 | 
						|
      block.([Reline::Key.new(c, c, false)])
 | 
						|
    else
 | 
						|
      if escaped_c.nil?
 | 
						|
        block.([Reline::Key.new(c, c, false)])
 | 
						|
      elsif escaped_c >= 128 # maybe, first byte of multi byte
 | 
						|
        block.([Reline::Key.new(c, c, false), Reline::Key.new(escaped_c, escaped_c, false)])
 | 
						|
      elsif escaped_c == "\e".ord # escape twice
 | 
						|
        block.([Reline::Key.new(c, c, false), Reline::Key.new(c, c, false)])
 | 
						|
      else
 | 
						|
        block.([Reline::Key.new(escaped_c, escaped_c | 0b10000000, true)])
 | 
						|
      end
 | 
						|
    end
 | 
						|
  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'
 |