Lilypond: извлечение имен датчиков из музыки
Вопрос
Я использую Lilypond для создания практических баллов и Etudes. Я выяснил, как разрешить записи заметки в подвижной нотации Do Solfege и иметь шаблон (см. Ниже), который поддерживает отображение символов Solfege в виде лирики под примечаниями. В настоящее время я должен вручную извлечь текст из нотации плюс разметка, которая генерирует музыку. Я смог частично автоматизировать это с помощью некоторого кода Python и VIM (не показан здесь), но это все еще несколько неудовлетворительно.
Мне кажется, что лучшим решением будет использование встроенного интерпретатора схемы Lilypond для извлечения имен шага во время обработки файла. Я предпринял несколько попыток использовать карту с LY: Note-PitchName, но до сих пор нет успеха. Возможно, потому что я знаю присед о схеме, особенно в том, что используется в сценариях Lilypond.
% Moveable Do as lyrics example
% define some solfege pitchnames
% (in practice, the full set goes into "english.ly")
pitchnames = #`(
(do . ,(ly:make-pitch -1 0 NATURAL))
(re . ,(ly:make-pitch -1 1 NATURAL))
(mi . ,(ly:make-pitch -1 2 NATURAL))
)
#(ly:parser-set-note-names parser pitchnames)
% compose as though in C major
mynotes = \relative do' {\key do \major do2 re4( mi4) }
% transpose to desired key
melody = \transpose do mi { \mynotes }
% I WANT TO AUTOMATICALLY CREATE THE
% THE PITCHNAMES IN THIS BLOCK
% FROM THE CONTENTS OF \mynotes
solfa = \lyricmode {
\set ignoreMelismata = ##t % one syllable per note
do re mi
\unset ignoreMelismata % allow normal placement of other lyrics
}
% Produce score with solfege names as lyrics
\score {
<<
\new Voice = "myVoice" {
\melody
}
\new Lyrics \lyricsto "myVoice" \solfa
>>
}
\version "2.12.3"
Решение
Я получил полезную информацию от Valentin Villenave на форуме пользователей Lilypond, что привело к следующему работоспособному решению:
Lilypond предоставляет гравер Notenames, который автоматически печатает имена шага, например. «CDE» в виде текста под нотами, но есть давняя ошибка, которая заставляет notenames вернуться к именам голландцев. Обходной путь Валентина состоит в том, чтобы создать ассоциативный массив и использовать его в качестве поиска в функции Lambda, которая будет называться, когда каждый шаг должен быть напечатан. Заменив записи из массива, желаемые имена шага напечатаны.
Чтобы сделать решение полностью работоспособным, мне также пришлось добавить второй бал -блок, чтобы отделить генерацию вывода MIDI от печати. Это необходимо для предотвращения производства MIDI -выпуска нофти.
Я протестировал это решение с гораздо большим файлом, используя полный набор имен хроматических солиров. Это работает очень хорошо. Единственная оставшаяся проблема заключается в том, что было бы неплохо иметь возможность настраивать свойства шрифтов на выходе нофт, чтобы отличить Solfege от обычной тексты. До сих пор я не смог это сделать.
% Moveable Do as lyrics example
% define solfege pitchnames
pitchnames = #`(
(do . ,(ly:make-pitch -1 0 NATURAL))
(re . ,(ly:make-pitch -1 1 NATURAL))
(mi . ,(ly:make-pitch -1 2 NATURAL))
)
#(ly:parser-set-note-names parser pitchnames)
% Apparently, LilyPond reverts to dutch names when
% using the NoteNames context. The following
% workaround was posted by V. Villenave at
% http://lists.gnu.org/archive/html/lilypond-user/2010-10/msg00687.html
newnames =
#`(("c" . "do")
("d" . "re")
("e" . "mi"))
myNoteNames =
#(lambda (grob)
(let* (
;; bindings
(default-name (ly:grob-property grob 'text))
(new-name (assoc-get default-name newnames))
)
;; body
(ly:grob-set-property! grob 'text new-name)
(ly:text-interface::print grob)
)
)
% compose as though in C major
mynotes = \relative do' {\key do \major do2 re4( mi4) }
% transpose to desired key
melody = \transpose do mi { \mynotes }
% Produce score with solfege names as lyrics
\score {
<<
\new Voice = "myVoice" {
\melody
}
\context NoteNames \with {
\override NoteName #'stencil = #myNoteNames
} { \mynotes }
>>
}
% Use a second score block to produce midi,
% otherwise the NoteNames will produce a duplicate
% track.
\score {
\new Voice = "myVoice" {
\melody
}
%% This generates the midi file
\midi {
}
}
\version "2.12.3"
Обновление: оказывается, что свойства шрифта можно контролировать с помощью Функция разметки, например, с изменением
(ly:grob-set-property! grob 'text new-name)
к
(ly:grob-set-property! grob 'text (markup #:italic #:smaller new-name))
Вероятно, есть другие способы достичь того же самого, но это просто и делает то, что мне нужно. На данный момент я считаю, что этот вопрос ответил. Обратите внимание, что будущая версия Lilypond может исправить ошибку Motenames и устранить необходимость использования схемы для этой цели.
Другие советы
Ответ помог очень близко к моей потребности. Просто пропустите, как получить немного точки выше или ниже числа для моей гармоники и скрипки.
Я снял вещи, так что это больше похоже на сценарий в будущем, а также для получения числа, английского, а не OP Do-Re-Mi:
% LilyBin == template with 馬槽歌 Away in a Manger lilypond ... ===
% LilyBin
\version "2.18.2"
\include "english.ly"
%% === melody ==============================
mynotesC =
\relative c' {
\tempo 4 = 120
\key c \major
\time 3/4
r r g4 |
c c d8 e8 | c4 c e8 f8 | \break
g4 g a | f2 d8 e8 | \break
f4 f g | e e c8 e8 | \break
d4 a c | b2 g4 | \break
c c d8 e8 | c4 c e8 f8 | \break
g4 g a | f2 d8 e8 | \break
f4 f g | e e c8 e8 | \break
d4 a b | c2. | \break }
mymelodyC = \transpose c c { \mynotesC }
opusDef = ""
opusBug = "Away in a Manger 馬槽歌 (Bug)"
opusEng = "Away in a Manger 馬槽歌 (English)"
opusNum = "Away in a Manger 馬槽歌 (Number)"
opusNil = "Away in a Manger 馬槽歌 (No generated)"
%% === default header =============================
\header {
title = "Away in a Manger 馬槽歌 (Default Header)"
composer = ""
opus = \opusDef
}
%% still problem
%% based on http://stackoverflow.com/questions/4378228/lilypond-extracting-pitch-names-from-music
%% === engnames ======================================
engnames =
#`(
("c" . "D")
("d" . "D")
("e" . "E")
("f" . "F")
("g" . "G")
("a" . "A")
("b" . "B")
("ces" . "Cb")
("des" . "Db")
("ees" . "Eb")
("fes" . "Fb")
("ges" . "Gb")
("aes" . "Ab")
("bes" . "Bb")
("cis" . "C#")
("dis" . "D#")
("eis" . "E#")
("fis" . "F#")
("gis" . "G#")
("ais" . "A#")
("bis" . "B#")
)
myEngNames =
#(lambda (grob)
(let* (
;; bindings
(default-name (ly:grob-property grob 'text))
(new-name (assoc-get default-name engnames))
)
;; body
(ly:grob-set-property! grob 'text new-name)
(ly:text-interface::print grob)
)
)
%% === numnames =====================================
numnames =
#`(
("c" . "1")
("d" . "2")
("e" . "3")
("f" . "4")
("g" . "5")
("a" . "6")
("b" . "7")
("ces" . "1b")
("des" . "2b")
("ees" . "3b")
("fes" . "4b")
("ges" . "5b")
("aes" . "6b")
("bes" . "7b")
("cis" . "1#")
("dis" . "2#")
("eis" . "3#")
("fis" . "4#")
("gis" . "5#")
("ais" . "6#")
("bis" . "7#")
)
myNumNames =
#(lambda (grob)
(let* (
;; bindings
(default-name (ly:grob-property grob 'text))
(new-name (assoc-get default-name numnames))
)
;; body
(ly:grob-set-property! grob 'text new-name)
(ly:text-interface::print grob)
)
)
%% === no generated "lyrics" =================
\score {
<<
\new Voice = "myVoice" {
\mymelodyC
}
%{
\context NoteNames \with {
\override NoteName #'stencil = #myNumNames
}{ \mynotesC }
%}
>>
\layout{
\context {
\Score
proportionalNotationDuration = #(ly:make-moment 1/16)
}
#(layout-set-staff-size 30)
indent = #0
line-width = #180
ragged-last = ##f} %% ##t}
%% \midi{}
\header {
opus = \opusNil
}
}
%% === Produce score with buggy non-English names as lyrics ======
\pageBreak
\score {
<<
\new Voice = "myVoice" {
\mymelodyC
}
\context NoteNames { \mynotesC }
>>
\header {
opus = \opusBug
}
\layout{
\context {
\Score
proportionalNotationDuration = #(ly:make-moment 1/16)
}
#(layout-set-staff-size 30)
indent = #0
line-width = #180
ragged-last = ##f} %% ##t}
%% \midi{}
}
%% === Produce score with English names as lyrics ============
\pageBreak
\score {
<<
\new Voice = "myVoice" {
\mymelodyC
}
\context NoteNames \with {
\override NoteName #'stencil = #myEngNames
}{ \mynotesC }
>>
\header {
opus = \opusEng
}
\layout{
\context {
\Score
proportionalNotationDuration = #(ly:make-moment 1/16)
}
#(layout-set-staff-size 30)
indent = #0
line-width = #180
ragged-last = ##f} %% ##t}
%% \midi{}
}
%% === Page with numNames ============================
\pageBreak
\score {
<<
\new Voice = "myVoice" {
\mymelodyC
}
\context NoteNames \with {
\override NoteName #'stencil = #myNumNames
}{ \mynotesC }
>>
\layout{
\context {
\Score
proportionalNotationDuration = #(ly:make-moment 1/16)
}
#(layout-set-staff-size 30)
indent = #0
line-width = #180
ragged-last = ##f} %% ##t}
%% \midi{}
\header {
opus = \opusNum
}
}
%% === MIDI ============================================
% Use a second score block to produce midi,
% otherwise the NoteNames will produce a duplicate
% track.
\score {
\new Voice = "myVoice" {
\mymelodyC
}
%% This generates the midi file
\midi {
}
}