pry 0.10.4-java → 0.11.0-java
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +52 -18
- data/LICENSE +1 -1
- data/README.md +32 -31
- data/bin/pry +3 -7
- data/lib/pry.rb +4 -4
- data/lib/pry/basic_object.rb +6 -0
- data/lib/pry/cli.rb +39 -34
- data/lib/pry/code.rb +6 -1
- data/lib/pry/code/code_file.rb +8 -2
- data/lib/pry/code_object.rb +23 -0
- data/lib/pry/color_printer.rb +20 -11
- data/lib/pry/command.rb +40 -16
- data/lib/pry/command_set.rb +9 -2
- data/lib/pry/commands/cat/exception_formatter.rb +11 -10
- data/lib/pry/commands/cat/file_formatter.rb +7 -3
- data/lib/pry/commands/code_collector.rb +16 -14
- data/lib/pry/commands/easter_eggs.rb +9 -9
- data/lib/pry/commands/edit.rb +7 -3
- data/lib/pry/commands/edit/file_and_line_locator.rb +1 -1
- data/lib/pry/commands/find_method.rb +1 -1
- data/lib/pry/commands/gem_open.rb +1 -1
- data/lib/pry/commands/gem_readme.rb +25 -0
- data/lib/pry/commands/gem_search.rb +40 -0
- data/lib/pry/commands/hist.rb +2 -2
- data/lib/pry/commands/jump_to.rb +7 -7
- data/lib/pry/commands/ls.rb +3 -1
- data/lib/pry/commands/ls/constants.rb +12 -1
- data/lib/pry/commands/ls/formatter.rb +1 -0
- data/lib/pry/commands/ls/jruby_hacks.rb +2 -2
- data/lib/pry/commands/ls/self_methods.rb +2 -0
- data/lib/pry/commands/play.rb +2 -2
- data/lib/pry/commands/reload_code.rb +2 -2
- data/lib/pry/commands/ri.rb +4 -0
- data/lib/pry/commands/shell_command.rb +34 -8
- data/lib/pry/commands/show_info.rb +10 -2
- data/lib/pry/commands/watch_expression/expression.rb +1 -1
- data/lib/pry/commands/whereami.rb +7 -6
- data/lib/pry/config.rb +3 -16
- data/lib/pry/config/behavior.rb +140 -49
- data/lib/pry/config/default.rb +21 -33
- data/lib/pry/config/memoization.rb +44 -0
- data/lib/pry/core_extensions.rb +12 -2
- data/lib/pry/editor.rb +1 -1
- data/lib/pry/exceptions.rb +1 -1
- data/lib/pry/forwardable.rb +23 -0
- data/lib/pry/helpers/base_helpers.rb +6 -10
- data/lib/pry/helpers/documentation_helpers.rb +1 -0
- data/lib/pry/helpers/options_helpers.rb +1 -1
- data/lib/pry/helpers/text.rb +69 -75
- data/lib/pry/history.rb +22 -1
- data/lib/pry/history_array.rb +1 -1
- data/lib/pry/hooks.rb +48 -107
- data/lib/pry/indent.rb +6 -2
- data/lib/pry/input_completer.rb +138 -120
- data/lib/pry/last_exception.rb +2 -2
- data/lib/pry/method.rb +15 -15
- data/lib/pry/method/disowned.rb +1 -0
- data/lib/pry/method/patcher.rb +0 -3
- data/lib/pry/output.rb +37 -38
- data/lib/pry/pager.rb +11 -8
- data/lib/pry/plugins.rb +20 -5
- data/lib/pry/pry_class.rb +30 -4
- data/lib/pry/pry_instance.rb +8 -6
- data/lib/pry/repl.rb +38 -8
- data/lib/pry/repl_file_loader.rb +1 -1
- data/lib/pry/rubygem.rb +3 -1
- data/lib/pry/slop.rb +661 -0
- data/lib/pry/slop/LICENSE +20 -0
- data/lib/pry/slop/commands.rb +196 -0
- data/lib/pry/slop/option.rb +208 -0
- data/lib/pry/terminal.rb +16 -5
- data/lib/pry/test/helper.rb +12 -3
- data/lib/pry/version.rb +1 -1
- data/lib/pry/wrapped_module.rb +7 -7
- data/lib/pry/{module_candidate.rb → wrapped_module/candidate.rb} +7 -13
- metadata +14 -19
data/lib/pry/indent.rb
CHANGED
@@ -66,8 +66,10 @@ class Pry
|
|
66
66
|
#
|
67
67
|
# :reserved and :keywords are the CodeRay 0.9.8 and 1.0.0 respectively
|
68
68
|
# classifications of "super", "next", "return", etc.
|
69
|
-
STATEMENT_END_TOKENS = IGNORE_TOKENS + [:regexp, :integer, :float,
|
70
|
-
:delimiter, :reserved
|
69
|
+
STATEMENT_END_TOKENS = IGNORE_TOKENS + [:regexp, :integer, :float,
|
70
|
+
:keyword, :delimiter, :reserved,
|
71
|
+
:instance_variable,
|
72
|
+
:class_variable, :global_variable]
|
71
73
|
|
72
74
|
# Collection of tokens that should appear dedented even though they
|
73
75
|
# don't affect the surrounding code.
|
@@ -223,6 +225,8 @@ class Pry
|
|
223
225
|
|
224
226
|
seen_for_at << add_after if OPTIONAL_DO_TOKENS.include?(token)
|
225
227
|
|
228
|
+
next if is_singleline_if
|
229
|
+
|
226
230
|
if kind == :delimiter
|
227
231
|
track_delimiter(token)
|
228
232
|
elsif OPEN_TOKENS.keys.include?(token) && !is_optional_do && !is_singleline_if
|
data/lib/pry/input_completer.rb
CHANGED
@@ -106,137 +106,155 @@ class Pry::InputCompleter
|
|
106
106
|
begin
|
107
107
|
context = target.eval("self")
|
108
108
|
context = context.class unless context.respond_to? :constants
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
109
|
+
candidates = context.constants.collect(&:to_s)
|
110
|
+
rescue
|
111
|
+
candidates = []
|
112
|
+
end
|
113
|
+
candidates = candidates.grep(/^#{message}/).collect(&path)
|
114
|
+
when CONSTANT_OR_METHOD_REGEXP # Constant or class methods
|
115
|
+
receiver = $1
|
116
|
+
message = Regexp.quote($2)
|
117
|
+
begin
|
118
|
+
candidates = eval("#{receiver}.constants.collect(&:to_s)", bind)
|
119
|
+
candidates |= eval("#{receiver}.methods.collect(&:to_s)", bind)
|
120
|
+
rescue Pry::RescuableException
|
121
|
+
candidates = []
|
122
|
+
end
|
123
|
+
candidates.grep(/^#{message}/).collect{|e| receiver + "::" + e}
|
124
|
+
when SYMBOL_METHOD_CALL_REGEXP # method call on a Symbol
|
125
|
+
receiver = $1
|
126
|
+
message = Regexp.quote($2)
|
127
|
+
candidates = Symbol.instance_methods.collect(&:to_s)
|
128
|
+
select_message(path, receiver, message, candidates)
|
129
|
+
when NUMERIC_REGEXP
|
130
|
+
# Numeric
|
131
|
+
receiver = $1
|
132
|
+
message = Regexp.quote($5)
|
133
|
+
begin
|
134
|
+
candidates = eval(receiver, bind).methods.collect(&:to_s)
|
135
|
+
rescue Pry::RescuableException
|
136
|
+
candidates = []
|
137
|
+
end
|
138
|
+
select_message(path, receiver, message, candidates)
|
139
|
+
when HEX_REGEXP
|
140
|
+
# Numeric(0xFFFF)
|
141
|
+
receiver = $1
|
142
|
+
message = Regexp.quote($2)
|
143
|
+
begin
|
144
|
+
candidates = eval(receiver, bind).methods.collect(&:to_s)
|
145
|
+
rescue Pry::RescuableException
|
146
|
+
candidates = []
|
147
|
+
end
|
148
|
+
select_message(path, receiver, message, candidates)
|
149
|
+
when GLOBALVARIABLE_REGEXP # global
|
150
|
+
regmessage = Regexp.new(Regexp.quote($1))
|
151
|
+
candidates = global_variables.collect(&:to_s).grep(regmessage)
|
152
|
+
when VARIABLE_REGEXP # variable
|
153
|
+
receiver = $1
|
154
|
+
message = Regexp.quote($2)
|
155
|
+
|
156
|
+
gv = eval("global_variables", bind).collect(&:to_s)
|
157
|
+
lv = eval("local_variables", bind).collect(&:to_s)
|
158
|
+
cv = eval("self.class.constants", bind).collect(&:to_s)
|
159
|
+
|
160
|
+
if (gv | lv | cv).include?(receiver) or /^[A-Z]/ =~ receiver && /\./ !~ receiver
|
161
|
+
# foo.func and foo is local var. OR
|
162
|
+
# Foo::Bar.func
|
143
163
|
begin
|
144
|
-
candidates = eval(receiver, bind).
|
164
|
+
candidates = eval("#{receiver}.methods", bind).collect(&:to_s)
|
145
165
|
rescue Pry::RescuableException
|
146
166
|
candidates = []
|
147
167
|
end
|
148
|
-
select_message(path, receiver, message, candidates)
|
149
|
-
when GLOBALVARIABLE_REGEXP # global
|
150
|
-
regmessage = Regexp.new(Regexp.quote($1))
|
151
|
-
candidates = global_variables.collect(&:to_s).grep(regmessage)
|
152
|
-
when VARIABLE_REGEXP # variable
|
153
|
-
receiver = $1
|
154
|
-
message = Regexp.quote($2)
|
155
|
-
|
156
|
-
gv = eval("global_variables", bind).collect(&:to_s)
|
157
|
-
lv = eval("local_variables", bind).collect(&:to_s)
|
158
|
-
cv = eval("self.class.constants", bind).collect(&:to_s)
|
159
|
-
|
160
|
-
if (gv | lv | cv).include?(receiver) or /^[A-Z]/ =~ receiver && /\./ !~ receiver
|
161
|
-
# foo.func and foo is local var. OR
|
162
|
-
# Foo::Bar.func
|
163
|
-
begin
|
164
|
-
candidates = eval("#{receiver}.methods", bind).collect(&:to_s)
|
165
|
-
rescue Pry::RescuableException
|
166
|
-
candidates = []
|
167
|
-
end
|
168
|
-
else
|
169
|
-
# func1.func2
|
170
|
-
candidates = []
|
171
|
-
ObjectSpace.each_object(Module){|m|
|
172
|
-
begin
|
173
|
-
name = m.name.to_s
|
174
|
-
rescue Pry::RescuableException
|
175
|
-
name = ""
|
176
|
-
end
|
177
|
-
next if name != "IRB::Context" and
|
178
|
-
/^(IRB|SLex|RubyLex|RubyToken)/ =~ name
|
179
|
-
|
180
|
-
# jruby doesn't always provide #instance_methods() on each
|
181
|
-
# object.
|
182
|
-
if m.respond_to?(:instance_methods)
|
183
|
-
candidates.concat m.instance_methods(false).collect(&:to_s)
|
184
|
-
end
|
185
|
-
}
|
186
|
-
candidates.sort!
|
187
|
-
candidates.uniq!
|
188
|
-
end
|
189
|
-
select_message(path, receiver, message, candidates)
|
190
|
-
when /^\.([^.]*)$/
|
191
|
-
# Unknown(maybe String)
|
192
|
-
receiver = ""
|
193
|
-
message = Regexp.quote($1)
|
194
|
-
candidates = String.instance_methods(true).collect(&:to_s)
|
195
|
-
select_message(path, receiver, message, candidates)
|
196
168
|
else
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
169
|
+
# func1.func2
|
170
|
+
candidates = Set.new
|
171
|
+
to_ignore = ignored_modules
|
172
|
+
ObjectSpace.each_object(Module){|m|
|
173
|
+
next if (to_ignore.include?(m) rescue true)
|
174
|
+
# jruby doesn't always provide #instance_methods() on each
|
175
|
+
# object.
|
176
|
+
if m.respond_to?(:instance_methods)
|
177
|
+
candidates.merge m.instance_methods(false).collect(&:to_s)
|
178
|
+
end
|
179
|
+
}
|
180
|
+
end
|
181
|
+
select_message(path, receiver, message, candidates.sort)
|
182
|
+
when /^\.([^.]*)$/
|
183
|
+
# Unknown(maybe String)
|
184
|
+
receiver = ""
|
185
|
+
message = Regexp.quote($1)
|
186
|
+
candidates = String.instance_methods(true).collect(&:to_s)
|
187
|
+
select_message(path, receiver, message, candidates)
|
188
|
+
else
|
189
|
+
candidates = eval(
|
190
|
+
"methods | private_methods | local_variables | " \
|
191
|
+
"self.class.constants | instance_variables",
|
192
|
+
bind
|
193
|
+
).collect(&:to_s)
|
194
|
+
|
195
|
+
if eval("respond_to?(:class_variables)", bind)
|
196
|
+
candidates += eval("class_variables", bind).collect(&:to_s)
|
208
197
|
end
|
209
|
-
|
210
|
-
|
198
|
+
candidates = (candidates|ReservedWords|custom_completions).grep(/^#{Regexp.quote(input)}/)
|
199
|
+
candidates.collect(&path)
|
211
200
|
end
|
201
|
+
rescue Pry::RescuableException
|
202
|
+
[]
|
212
203
|
end
|
204
|
+
end
|
213
205
|
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
206
|
+
def select_message(path, receiver, message, candidates)
|
207
|
+
candidates.grep(/^#{message}/).collect { |e|
|
208
|
+
case e
|
209
|
+
when /^[a-zA-Z_]/
|
210
|
+
path.call(receiver + "." << e)
|
211
|
+
when /^[0-9]/
|
212
|
+
when *Operators
|
213
|
+
#receiver + " " << e
|
214
|
+
end
|
215
|
+
}.compact
|
216
|
+
end
|
217
|
+
|
218
|
+
# build_path seperates the input into two parts: path and input.
|
219
|
+
# input is the partial string that should be completed
|
220
|
+
# path is a proc that takes an input and builds a full path.
|
221
|
+
def build_path(input)
|
222
|
+
# check to see if the input is a regex
|
223
|
+
return proc {|i| i.to_s }, input if input[/\/\./]
|
224
|
+
trailing_slash = input.end_with?('/')
|
225
|
+
contexts = input.chomp('/').split(/\//)
|
226
|
+
input = contexts[-1]
|
227
|
+
path = proc do |i|
|
228
|
+
p = contexts[0..-2].push(i).join('/')
|
229
|
+
p += '/' if trailing_slash && !i.nil?
|
230
|
+
p
|
224
231
|
end
|
232
|
+
return path, input
|
233
|
+
end
|
234
|
+
|
235
|
+
def ignored_modules
|
236
|
+
# We could cache the result, but IRB is not loaded by default.
|
237
|
+
# And this is very fast anyway.
|
238
|
+
# By using this approach, we avoid Module#name calls, which are
|
239
|
+
# relatively slow when there are a lot of anonymous modules defined.
|
240
|
+
s = Set.new
|
225
241
|
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
trailing_slash = input.end_with?('/')
|
233
|
-
contexts = input.chomp('/').split(/\//)
|
234
|
-
input = contexts[-1]
|
235
|
-
path = proc do |i|
|
236
|
-
p = contexts[0..-2].push(i).join('/')
|
237
|
-
p += '/' if trailing_slash && !i.nil?
|
238
|
-
p
|
242
|
+
scanner = lambda do |m|
|
243
|
+
next if s.include?(m) # IRB::ExtendCommandBundle::EXCB recurses.
|
244
|
+
s << m
|
245
|
+
m.constants(false).each do |c|
|
246
|
+
value = m.const_get(c)
|
247
|
+
scanner.call(value) if value.is_a?(Module)
|
239
248
|
end
|
240
|
-
return path, input
|
241
249
|
end
|
250
|
+
|
251
|
+
# FIXME: Add Pry here as well?
|
252
|
+
[:IRB, :SLex, :RubyLex, :RubyToken].each do |module_name|
|
253
|
+
next unless Object.const_defined?(module_name)
|
254
|
+
scanner.call(Object.const_get(module_name))
|
255
|
+
end
|
256
|
+
|
257
|
+
s.delete(IRB::Context) if defined?(IRB::Context)
|
258
|
+
s
|
259
|
+
end
|
242
260
|
end
|
data/lib/pry/last_exception.rb
CHANGED
data/lib/pry/method.rb
CHANGED
@@ -140,11 +140,11 @@ class Pry
|
|
140
140
|
# @param [Boolean] include_super Whether to include methods from ancestors.
|
141
141
|
# @return [Array[Pry::Method]]
|
142
142
|
def all_from_class(klass, include_super=true)
|
143
|
-
%w(public protected private).
|
143
|
+
%w(public protected private).flat_map do |visibility|
|
144
144
|
safe_send(klass, :"#{visibility}_instance_methods", include_super).map do |method_name|
|
145
145
|
new(safe_send(klass, :instance_method, method_name), :visibility => visibility.to_sym)
|
146
146
|
end
|
147
|
-
end
|
147
|
+
end
|
148
148
|
end
|
149
149
|
|
150
150
|
#
|
@@ -163,7 +163,7 @@ class Pry
|
|
163
163
|
|
164
164
|
#
|
165
165
|
# @deprecated
|
166
|
-
# please use {
|
166
|
+
# please use {all_from_obj} instead.
|
167
167
|
# the `method_type` argument is ignored.
|
168
168
|
#
|
169
169
|
def all_from_common(obj, method_type = nil, include_super=true)
|
@@ -212,9 +212,9 @@ class Pry
|
|
212
212
|
# the lowest copy will be returned.
|
213
213
|
def singleton_class_resolution_order(klass)
|
214
214
|
ancestors = Pry::Method.safe_send(klass, :ancestors)
|
215
|
-
resolution_order = ancestors.grep(Class).
|
215
|
+
resolution_order = ancestors.grep(Class).flat_map do |anc|
|
216
216
|
[singleton_class_of(anc), *singleton_class_of(anc).included_modules]
|
217
|
-
end
|
217
|
+
end
|
218
218
|
|
219
219
|
resolution_order.reverse.uniq.reverse - Class.included_modules
|
220
220
|
end
|
@@ -359,13 +359,13 @@ class Pry
|
|
359
359
|
# Paraphrased from `awesome_print` gem.
|
360
360
|
def signature
|
361
361
|
if respond_to?(:parameters)
|
362
|
-
args = parameters.inject([]) do |arr, (
|
363
|
-
|
364
|
-
arr << case
|
365
|
-
when :req then
|
366
|
-
when :opt then "#{
|
367
|
-
when :rest then "*#{
|
368
|
-
when :block then "&#{
|
362
|
+
args = parameters.inject([]) do |arr, (typ, nam)|
|
363
|
+
nam ||= (typ == :block ? 'block' : "arg#{arr.size + 1}")
|
364
|
+
arr << case typ
|
365
|
+
when :req then nam.to_s
|
366
|
+
when :opt then "#{nam}=?"
|
367
|
+
when :rest then "*#{nam}"
|
368
|
+
when :block then "&#{nam}"
|
369
369
|
else '?'
|
370
370
|
end
|
371
371
|
end
|
@@ -461,8 +461,8 @@ class Pry
|
|
461
461
|
|
462
462
|
# @param [String, Symbol] method_name
|
463
463
|
# @return [Boolean]
|
464
|
-
def respond_to?(method_name)
|
465
|
-
super or @method.respond_to?(method_name)
|
464
|
+
def respond_to?(method_name, include_all=false)
|
465
|
+
super or @method.respond_to?(method_name, include_all)
|
466
466
|
end
|
467
467
|
|
468
468
|
# Delegate any unknown calls to the wrapped method.
|
@@ -484,7 +484,7 @@ class Pry
|
|
484
484
|
else
|
485
485
|
fail_msg = "Cannot locate this method: #{name}."
|
486
486
|
if mri?
|
487
|
-
fail_msg +=
|
487
|
+
fail_msg += " Invoke the 'gem-install pry-doc' Pry command to get access to Ruby Core documentation.\n"
|
488
488
|
end
|
489
489
|
raise CommandError, fail_msg
|
490
490
|
end
|
data/lib/pry/method/disowned.rb
CHANGED
data/lib/pry/method/patcher.rb
CHANGED
@@ -42,9 +42,6 @@ class Pry
|
|
42
42
|
# transation we make that not happen, which means that alias_method_chains, etc.
|
43
43
|
# continue to work.
|
44
44
|
#
|
45
|
-
# @param [String] meth_name The method name before aliasing
|
46
|
-
# @param [Module] target The owner of the method
|
47
|
-
|
48
45
|
def with_method_transaction
|
49
46
|
temp_name = "__pry_#{method.original_name}__"
|
50
47
|
method = self.method
|
data/lib/pry/output.rb
CHANGED
@@ -1,50 +1,49 @@
|
|
1
|
-
class Pry
|
2
|
-
|
3
|
-
attr_reader :_pry_
|
1
|
+
class Pry::Output
|
2
|
+
attr_reader :_pry_
|
4
3
|
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
def puts(*objs)
|
10
|
-
return print "\n" if objs.empty?
|
4
|
+
def initialize(_pry_)
|
5
|
+
@_pry_ = _pry_
|
6
|
+
@boxed_io = _pry_.config.output
|
7
|
+
end
|
11
8
|
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
9
|
+
def puts(*objs)
|
10
|
+
return print "\n" if objs.empty?
|
11
|
+
objs.each do |obj|
|
12
|
+
if ary = Array.try_convert(obj)
|
13
|
+
puts(*ary)
|
14
|
+
else
|
15
|
+
print "#{obj.to_s.chomp}\n"
|
18
16
|
end
|
19
|
-
|
20
|
-
nil
|
21
17
|
end
|
18
|
+
nil
|
19
|
+
end
|
22
20
|
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
end
|
27
|
-
|
28
|
-
nil
|
21
|
+
def print(*objs)
|
22
|
+
objs.each do |obj|
|
23
|
+
@boxed_io.print decolorize_maybe(obj.to_s)
|
29
24
|
end
|
30
|
-
|
31
|
-
|
25
|
+
nil
|
26
|
+
end
|
27
|
+
alias << print
|
28
|
+
alias write print
|
32
29
|
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
str
|
37
|
-
else
|
38
|
-
Helpers::Text.strip_color str
|
39
|
-
end
|
40
|
-
end
|
30
|
+
def tty?
|
31
|
+
@boxed_io.respond_to?(:tty?) and @boxed_io.tty?
|
32
|
+
end
|
41
33
|
|
42
|
-
|
43
|
-
|
44
|
-
|
34
|
+
def method_missing(name, *args, &block)
|
35
|
+
@boxed_io.__send__(name, *args, &block)
|
36
|
+
end
|
37
|
+
|
38
|
+
def respond_to_missing?(m, include_all=false)
|
39
|
+
@boxed_io.respond_to?(m, include_all)
|
40
|
+
end
|
45
41
|
|
46
|
-
|
47
|
-
|
42
|
+
def decolorize_maybe(str)
|
43
|
+
if _pry_.config.color
|
44
|
+
str
|
45
|
+
else
|
46
|
+
Pry::Helpers::Text.strip_color str
|
48
47
|
end
|
49
48
|
end
|
50
49
|
end
|