# encoding: UTF-8
require 'psd/layer_info'

class PSD
  # Parses and provides information about text areas within layers in
  # the document.
  class TypeTool < LayerInfo
    def self.should_parse?(key)
      key == 'TySh'
    end
    
    # Parse all of the text data in the layer.
    def parse
      version = @file.read_short
      parse_transform_info

      text_version = @file.read_short
      descriptor_version = @file.read_int

      @data[:text] = Descriptor.new(@file).parse

      @data[:engine_data] = nil
      begin
        parser.parse!
        @data[:engine_data] = parser.result
      rescue Exception => e
        PSD.logger.error e.message
      end

      warpVersion = @file.read_short
      descriptor_version = @file.read_int

      @data[:warp] = Descriptor.new(@file).parse
      [:left, :top, :right, :bottom].each do |pos|
        @data[pos] = @file.read_int
      end

      return self
    end

    # Extracts the text within the text area. In the event that psd-enginedata fails
    # for some reason, we attempt to extract the text using some rough regex.
    def text_value
      @data[:text]['Txt ']
    end
    alias :to_s :text_value

    # Gets all of the basic font information for this text area. This assumes that
    # the first font is the only one you want.
    def font
      {
        name: fonts.first,
        sizes: sizes,
        colors: colors,
        css: to_css
      }
    end

    # Returns all fonts listed for this layer, since fonts are defined on a 
    # per-character basis.
    def fonts
      return [] if engine_data.nil?
      engine_data.ResourceDict.FontSet.map(&:Name)
    end

    # Return all font sizes for this layer.
    def sizes
      return [] if engine_data.nil? || !styles.has_key?('FontSize')
      styles['FontSize'].uniq
    end

    # Return all colors used for text in this layer. The colors are returned in RGBA
    # format as an array of arrays.
    #
    # => [[255, 0, 0, 255], [0, 0, 255, 255]]
    def colors
      # If the color is opaque black, this field is sometimes omitted.
      return [[0, 0, 0, 255]] if engine_data.nil? || !styles.has_key?('FillColor')
      styles['FillColor'].map { |s|
        values = s['Values'].map { |v| (v * 255).round }
        values << values.shift # Change ARGB -> RGBA for consistency
      }.uniq
    end

    def engine_data
      @data[:engine_data]
    end

    def styles
      return {} if engine_data.nil?

      @styles ||= (
        data = engine_data.EngineDict.StyleRun.RunArray.map do |r|
          r.StyleSheet.StyleSheetData
        end

        Hash[data.reduce({}) { |m, o|
          o.each do |k, v|
            (m[k] ||= []) << v
          end

          m
        }.map { |k, v|
          [k, v.uniq]
        }]
      )
    end

    def parser
      @parser ||= PSD::EngineData.new(@data[:text]['EngineData'])
    end

    # Creates the CSS string and returns it. Each property is newline separated
    # and not all properties may be present depending on the document.
    #
    # Colors are returned in rgba() format and fonts may include some internal
    # Photoshop fonts.
    def to_css      
      definition = {
        'font-family' => fonts.join(', '),
        'font-size' => "#{sizes.first}pt",
        'color' => "rgba(#{colors.first.join(', ')})"
      }

      css = []
      definition.each do |k, v|
        css << "#{k}: #{v};"
      end

      css.join("\n")
    end

    def to_hash
      {
        value:      text_value,
        font:       font,
        left:       left,
        top:        top,
        right:      right,
        bottom:     bottom,
        transform:  transform
      }
    end

    def method_missing(method, *args, &block)
      return @data[method] if @data.has_key?(method)
      return super
    end

    private

    def parse_transform_info
      @data[:transform] = {}
      [:xx, :xy, :yx, :yy, :tx, :ty].each do |t|
        @data[:transform][t] = @file.read_double
      end
    end
  end
end