2012-08-11 20:22:29 -04:00
|
|
|
class Pry
|
2012-12-25 16:35:17 -05:00
|
|
|
class Command::AmendLine < Pry::ClassCommand
|
2013-03-02 19:01:55 -05:00
|
|
|
match(/amend-line(?: (-?\d+)(?:\.\.(-?\d+))?)?/)
|
2012-08-11 21:26:59 -04:00
|
|
|
group 'Editing'
|
2012-12-25 16:35:17 -05:00
|
|
|
description 'Amend a line of input in multi-line mode.'
|
|
|
|
command_options :interpolate => false, :listing => 'amend-line'
|
2012-08-11 20:22:29 -04:00
|
|
|
|
|
|
|
banner <<-'BANNER'
|
2013-01-09 15:23:19 -05:00
|
|
|
Amend a line of input in multi-line mode. `amend-line N`, where the N represents
|
|
|
|
line to replace. Can also specify a range of lines using `amend-line N..M`
|
|
|
|
syntax. Passing "!" as replacement content deletes the line(s) instead.
|
2012-08-11 20:22:29 -04:00
|
|
|
|
2013-01-09 15:23:19 -05:00
|
|
|
amend-line 1 puts 'new' # replace line 1
|
|
|
|
amend-line 1..4 ! # delete lines 1..4
|
|
|
|
amend-line 3 >puts 'bye' # insert before line 3
|
|
|
|
amend-line puts 'appended' # no line number modifies immediately preceding line
|
2012-08-11 20:22:29 -04:00
|
|
|
BANNER
|
|
|
|
|
|
|
|
def process
|
2013-01-02 11:53:01 -05:00
|
|
|
raise CommandError, "No input to amend." if eval_string.empty?
|
2012-08-11 20:22:29 -04:00
|
|
|
|
2013-01-02 11:53:01 -05:00
|
|
|
eval_string.replace amended_input(eval_string)
|
2013-01-12 16:15:51 -05:00
|
|
|
run "fix-indent"
|
2013-01-02 11:53:01 -05:00
|
|
|
run "show-input"
|
|
|
|
end
|
2012-08-11 20:22:29 -04:00
|
|
|
|
2013-01-02 11:53:01 -05:00
|
|
|
private
|
2012-08-11 20:22:29 -04:00
|
|
|
|
2013-01-02 11:53:01 -05:00
|
|
|
# @param [String] string The string to amend.
|
|
|
|
# @return [String] A new string with the amendments applied to it.
|
|
|
|
def amended_input(string)
|
|
|
|
input_array = eval_string.each_line.to_a
|
2012-08-11 20:22:29 -04:00
|
|
|
|
|
|
|
if arg_string == "!"
|
2013-01-02 11:53:01 -05:00
|
|
|
delete_from_array(input_array, line_range)
|
2012-08-11 20:22:29 -04:00
|
|
|
elsif arg_string.start_with?(">")
|
2013-01-02 11:53:01 -05:00
|
|
|
insert_into_array(input_array, line_range)
|
2012-08-11 20:22:29 -04:00
|
|
|
else
|
2013-01-02 11:53:01 -05:00
|
|
|
replace_in_array(input_array, line_range)
|
|
|
|
end
|
|
|
|
|
|
|
|
input_array.join
|
|
|
|
end
|
|
|
|
|
|
|
|
def delete_from_array(array, range)
|
|
|
|
array.slice!(range)
|
|
|
|
end
|
|
|
|
|
|
|
|
def insert_into_array(array, range)
|
|
|
|
insert_slot = Array(range).first
|
2013-11-12 01:54:31 -05:00
|
|
|
array.insert(insert_slot, arg_string[1..-1] << "\n")
|
2013-01-02 11:53:01 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def replace_in_array(array, range)
|
2013-11-24 18:16:28 -05:00
|
|
|
array[range] = arg_string + "\n"
|
2013-01-02 11:53:01 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
# @return [Fixnum] The number of lines currently in `eval_string` (the input buffer).
|
|
|
|
def line_count
|
|
|
|
eval_string.lines.count
|
|
|
|
end
|
|
|
|
|
|
|
|
# Returns the (one-indexed) start and end lines given by the user.
|
|
|
|
# The lines in this range will be affected by the `amend-line`.
|
|
|
|
# Returns `nil` if no lines were specified by the user.
|
|
|
|
# @return [Array<Fixnum>, nil]
|
|
|
|
def start_and_end_line_number
|
|
|
|
start_line_number, end_line_number = args
|
|
|
|
end_line_number ||= start_line_number.to_i
|
|
|
|
|
|
|
|
[start_line_number.to_i, end_line_number.to_i] if start_line_number
|
|
|
|
end
|
|
|
|
|
|
|
|
# Takes two numbers that are 1-indexed, and returns a range (or
|
|
|
|
# number) that is 0-indexed. 1-indexed means the first element is
|
|
|
|
# indentified by 1 rather than by 0 (as is the case for Ruby arrays).
|
|
|
|
# @param [Fixnum] start_line_number One-indexed number.
|
|
|
|
# @param [Fixnum] end_line_number One-indexed number.
|
|
|
|
# @return [Range] The zero-indexed range.
|
|
|
|
def zero_indexed_range_from_one_indexed_numbers(start_line_number, end_line_number)
|
|
|
|
# FIXME: one_index_number is a horrible name for this method
|
|
|
|
one_index_number(start_line_number)..one_index_number(end_line_number)
|
|
|
|
end
|
|
|
|
|
|
|
|
# The lines (or line) that will be modified by the `amend-line`.
|
|
|
|
# @return [Range, Fixnum] The lines or line.
|
|
|
|
def line_range
|
|
|
|
start_line_number, end_line_number = start_and_end_line_number
|
|
|
|
if start_line_number
|
|
|
|
zero_indexed_range_from_one_indexed_numbers(start_line_number,
|
|
|
|
end_line_number)
|
|
|
|
else
|
|
|
|
line_count - 1
|
2012-08-11 20:22:29 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2012-12-25 16:35:17 -05:00
|
|
|
|
|
|
|
Pry::Commands.add_command(Pry::Command::AmendLine)
|
2012-08-11 20:22:29 -04:00
|
|
|
end
|