# frozen_string_literal: true # A clef assigns pitches to the lines and spaces of a staff. class HeadMusic::Clef include HeadMusic::NamedRudiment CLEFS = [ { pitch: 'G4', line: 2, names: %w[treble G-clef], modern: true, unicode: '𝄞', html_entity: '𝄞', }, { pitch: 'G4', line: 1, names: ['French', 'French violin'], unicode: '𝄞', html_entity: '𝄞', }, { pitch: 'G3', line: 2, names: ['choral tenor', 'tenor', 'tenor G-clef'], modern: true, unicode: '𝄠', html_entity: '𝄠', }, { pitch: 'F3', line: 3, names: ['baritone'], unicode: '𝄢', html_entity: '𝄢', }, { pitch: 'F3', line: 4, names: %w[bass F-clef], modern: true, unicode: '𝄢', html_entity: '𝄢', }, { pitch: 'F3', line: 5, names: ['sub-bass'], unicode: '𝄢', html_entity: '𝄢', }, { pitch: 'C4', line: 1, names: ['soprano'], unicode: '𝄡', html_entity: '𝄡', }, { pitch: 'C4', line: 2, names: ['mezzo-soprano'], unicode: '𝄡', html_entity: '𝄡', }, { pitch: 'C4', line: 3, names: %w[alto viola counter-tenor countertenor C-clef], modern: true, unicode: '𝄡', html_entity: '𝄡', }, { pitch: 'C4', line: 4, names: ['tenor', 'tenor C-clef'], modern: true, unicode: '𝄡', html_entity: '𝄡', }, { pitch: 'C4', line: 5, names: ['baritone', 'baritone C-clef'], unicode: '𝄡', html_entity: '𝄡', }, { pitch: nil, line: 3, names: %w[neutral percussion], modern: true, unicode: '𝄥', html_entity: '𝄥', }, ].freeze def self.get(name) get_by_name(name) end attr_reader :pitch, :line, :musical_symbol delegate :ascii, :html_entity, :unicode, to: :musical_symbol def initialize(name) @name = name.to_s clef_data = CLEFS.detect { |clef| clef[:names].map(&:downcase).include?(name.downcase) } @pitch = HeadMusic::Pitch.get(clef_data[:pitch]) @line = clef_data[:line] @modern = clef_data[:modern] @musical_symbol = HeadMusic::MusicalSymbol.new(clef_data.slice(:ascii, :html_entity, :unicode)) end def clef_type "#{pitch.letter_name}-clef" end def pitch_for_line(line_number) @line_pitches ||= {} @line_pitches[line_number] ||= begin steps = (line_number - line) * 2 pitch.natural_steps(steps) end end def pitch_for_space(space_number) @space_pitches ||= {} @space_pitches[space_number] ||= begin steps = (space_number - line) * 2 + 1 pitch.natural_steps(steps) end end def modern? @modern end def ==(other) to_s == other.to_s end end