mxhold/lifelike

View on GitHub
lib/lifelike/lifelike_cellular_automaton/world_string_analyzer.rb

Summary

Maintainability
A
0 mins
Test Coverage
module Lifelike
  class LifelikeCellularAutomaton
    class WorldStringAnalyzer
      def initialize(string, default_dead_char: ' ', default_alive_char: 'X')
        @world_string = string
        @default_dead_char = default_dead_char
        @default_alive_char = default_alive_char
      end

      def dead_char
        case valid_chars.size
        when 0
          fail InsufficientValidCharacterError.new(allowed_chars_by_aliveness)
        when 1
          deadlike_char || @default_dead_char
        else
          least_alive_valid_char
        end
      end

      def alive_char
        case valid_chars.size
        when 0
          fail InsufficientValidCharacterError.new(allowed_chars_by_aliveness)
        when 1
          lifelike_char || @default_alive_char
        else
          most_alive_valid_char
        end
      end

      private

      def deadlike_char
        valid_char if deadlike.include?(valid_char)
      end

      def lifelike_char
        valid_char if lifelike.include?(valid_char)
      end

      def valid_char
        valid_chars.first
      end

      def least_alive_valid_char
        valid_chars_by_aliveness.first
      end

      def most_alive_valid_char
        valid_chars_by_aliveness.last
      end

      def valid_chars_by_aliveness
        valid_chars.take(2).sort_by { |char| aliveness(char) }
      end

      def valid_chars
        @world_string.chars.uniq.select do |char|
          allowed_chars_by_aliveness.include?(char)
        end
      end

      # Roughly in order from most dead-like to most alive-like
      def allowed_chars_by_aliveness
        deadlike + lifelike
      end

      def deadlike
        [' ', '_', '.', ',', 'o', 'O', '0']
      end

      def lifelike
        ['1', 'x', '*', 'X', '#', '@']
      end

      def aliveness(char)
        allowed_chars_by_aliveness.index(char)
      end
    end
  end
end