2019-04-27 08:53:09 +03:00
|
|
|
class Reline::KeyStroke
|
|
|
|
using Module.new {
|
|
|
|
refine Array do
|
|
|
|
def start_with?(other)
|
|
|
|
other.size <= size && other == self.take(other.size)
|
|
|
|
end
|
|
|
|
|
|
|
|
def bytes
|
|
|
|
self
|
|
|
|
end
|
|
|
|
end
|
|
|
|
}
|
|
|
|
|
|
|
|
def initialize(config)
|
|
|
|
@config = config
|
|
|
|
end
|
|
|
|
|
|
|
|
def match_status(input)
|
|
|
|
key_mapping.keys.select { |lhs|
|
|
|
|
lhs.start_with? input
|
|
|
|
}.tap { |it|
|
|
|
|
return :matched if it.size == 1 && (it.max_by(&:size)&.size&.== input.size)
|
|
|
|
return :matching if it.size == 1 && (it.max_by(&:size)&.size&.!= input.size)
|
|
|
|
return :matched if it.max_by(&:size)&.size&.< input.size
|
|
|
|
return :matching if it.size > 1
|
|
|
|
}
|
|
|
|
key_mapping.keys.select { |lhs|
|
|
|
|
input.start_with? lhs
|
|
|
|
}.tap { |it|
|
|
|
|
return it.size > 0 ? :matched : :unmatched
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def expand(input)
|
2019-11-08 10:17:53 +03:00
|
|
|
lhs = key_mapping.keys.select { |item| input.start_with? item }.sort_by(&:size).reverse.first
|
2019-04-27 08:53:09 +03:00
|
|
|
return input unless lhs
|
|
|
|
rhs = key_mapping[lhs]
|
|
|
|
|
|
|
|
case rhs
|
|
|
|
when String
|
|
|
|
rhs_bytes = rhs.bytes
|
|
|
|
expand(expand(rhs_bytes) + expand(input.drop(lhs.size)))
|
|
|
|
when Symbol
|
|
|
|
[rhs] + expand(input.drop(lhs.size))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-06-05 05:29:41 +03:00
|
|
|
private
|
|
|
|
|
2019-04-27 08:53:09 +03:00
|
|
|
def key_mapping
|
2019-06-01 03:05:58 +03:00
|
|
|
@config.key_bindings
|
2019-04-27 08:53:09 +03:00
|
|
|
end
|
|
|
|
end
|