lib/ting/tones/accents.rb

Summary

Maintainability
A
0 mins
Test Coverage
# coding: utf-8

module Ting
  module Tones
    class Accents < Tone
      class << self

      UNICODE_TONE_GLYPHS={
        :a=>[97, 257, 225, 462, 224],
        :e=>[101, 275, 233, 283, 232],
        :i=>[105, 299, 237, 464, 236],
        :o=>[111, 333, 243, 466, 242],
        :u=>[117, 363, 250, 468, 249],
        :v=>[252, 470, 472, 474, 476]
      }

      def tone_glyph(letter,tone)
        if (u=UNICODE_TONE_GLYPHS[letter.to_sym][tone%MAX_TONE])
          [u].pack('U')
        end
      end

      def add_tone(syll, tone)
        syll = syll.sub('ü','v')
        tone = tone % MAX_TONE
        case syll
        when /a/
          syll.sub(/a/, tone_glyph(:a,tone))
        when /e/
          syll.sub(/e/, tone_glyph(:e,tone)).sub('v', 'ü')
        when /o/
          syll.sub(/o/, tone_glyph(:o,tone))
        when /(i|u|v)\z/
          syll.sub($1, tone_glyph($1,tone)).sub('v', 'ü')
        when /(i|u|v)/
          syll.sub($1, tone_glyph($1,tone)).sub('v', 'ü')
        else
          syll
        end
      end

      def peek_tone(syll)
        candidates = each_tone_glyph.map do |vowel, tones|
          tone_glyph = syll.codepoints.find {|t| tones.include?(t)}
          normalize( tones.index(tone_glyph) ) if tone_glyph
        end.compact
        candidates.reject {|tone| tone == 5}.first || candidates.first
      end

      # returns [ tone number, syllable without tone ]
      # e.g. ni3 => [ 3, 'ni' ]
      def pop_tone(syll)
        tone = peek_tone(syll)
        unpacked = syll.codepoints.to_a
        each_tone_glyph do |vowel, tone_glyph_codepoints|
          tone_glyph_cp = tone_glyph_codepoints[tone % MAX_TONE]
          if unpacked.include? tone_glyph_cp
            unpacked[unpacked.index(tone_glyph_cp)] = vowel.to_s.unpack('U').first
            return [normalize(tone_glyph_codepoints.index(tone_glyph_cp)), unpacked.pack('U*').sub('v', 'ü')]
          end
        end
      end

      private
        def each_tone_glyph
          return to_enum(__method__) unless block_given?
          [:a,:e,:i,:o,:u,:v].each do |v|  #Order is significant
            vowel, tones = v, UNICODE_TONE_GLYPHS[v]
            yield vowel,tones
          end
        end

      end
    end
  end
end