You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
devdocs/lib/docs/core/entry_index.rb

86 lines
1.9 KiB

require 'yajl/json_gem'
module Docs
class EntryIndex
attr_reader :entries, :types
def initialize
@entries = []
@index = Set.new
@types = Hash.new { |hash, key| hash[key] = Type.new key }
end
def add(entry)
if entry.is_a? Array
entry.each(&method(:add))
else
add_entry(entry) unless entry.root?
end
end
def empty?
@entries.empty?
end
alias_method :blank?, :empty?
def length
@entries.length
end
def as_json
{ entries: entries_as_json, types: types_as_json }
end
def to_json
JSON.generate(as_json)
end
private
def add_entry(entry)
if @index.add?(entry.as_json.to_s)
@entries << entry.dup
@types[entry.type].count += 1 if entry.type
end
end
def entries_as_json
@entries.sort! { |a, b| sort_fn(a.name, b.name) }.map(&:as_json)
end
def types_as_json
@types.values.sort! { |a, b| sort_fn(a.name, b.name) }.map(&:as_json)
end
SPLIT_INTS = /(?<=\d)\.(?=[\s\d])/.freeze
def sort_fn(a, b)
if (a.getbyte(0) >= 49 && a.getbyte(0) <= 57) || (b.getbyte(0) >= 49 && b.getbyte(0) <= 57)
a_split = a.split(SPLIT_INTS)
b_split = b.split(SPLIT_INTS)
a_length = a_split.length
b_length = b_split.length
return a.casecmp(b) if a_length == 1 && b_length == 1
return 1 if a_length == 1
return -1 if b_length == 1
a_split.each_with_index { |s, i| a_split[i] = s.to_i unless i == a_length - 1 }
b_split.each_with_index { |s, i| b_split[i] = s.to_i unless i == b_length - 1 }
if b_length > a_length
(b_length - a_length).times { a_split.insert(-2, 0) }
elsif a_length > b_length
(a_length - b_length).times { b_split.insert(-2, 0) }
end
a_split <=> b_split
else
a.casecmp(b)
end
end
end
end