1.3.1 Première approche (exemple)
- La problématique
Partant du code suivant, l’indication « espressivo » (les deux petits soufflets) devrait être plus large, mais pas gagner en hauteur.
{ f'\espressivo }
- Les solutions envisageables
- créer un nouveau stencil
- redimensionner le stencil par défaut sur l’axe des X
- Réflexions sur l’Thoughts about previous item
La solution A est réalisable, mais demandera plus d’effort de codage. Quant à la solution B, bien que moins exigeante, elle a le désavantage de ne pas donner de résultats acceptables dans tous les cas – mais on ne le saura pas avant d’avoir essayé.
⇒ opter pour la solution B.
- Codage
- Si ’lon veut manipuler le stencil par défaut, il faut s’informer à son
sujet.
L’indication d’espressivo est un objet graphique
Script
. En consultant la référence des propriétés internes, on lit‘stencil’ (stencil): ‘ly:script-interface::print’
Voyons voir si tout cela marche. Pour être assurés d’affecter le bon grob, nous utiliserons des couleurs. Il existe une fonction prédéfinie à cet effet – voir Scheme functions.
-- Fonction: stencil-with-color stencil couleur Renvoie une version modifiée du stencil donné, teint dans la couleur donnée. Voir ‘normalize-color’ pour les formats de couleur possibles.
-- Fonction: normalize-color couleur Convertit une couleur donnée dans l'un des formats pris en charge en une liste de 4 nombres : R, G, B, A. Les formats possibles sont : une liste de 4 nombres ; une liste de 3 nombres (transparence par défaut à 1.0) ; une chaîne CSS (nom de couleur, ou “#RRGGBB”, “#RRGGBBAA”, “#RGB” ou “#RGBA”).
Utilisant un nom de couleur (‘red’), nous obtenons le code suivant :
colorDefaultStencil = \once \override Script.stencil = % `lambda` starts a procedure; its argument is `grob`. #(lambda (grob) ;; With `let`, local variables are defined. (let ((stil (ly:script-interface::print grob))) ;; The procedure returns the colored default stencil. (stencil-with-color stil red))) { \colorDefaultStencil f'\espressivo }
Parfait, cela fonctionne.
- Ce type d’opération – prendre un stencil ou une propriété par défaut
et en retourner une version modifiée – est très courante, et LilyPond
dispose d’une fonction dédiée à cet effet :
grob-transformer
.-- Function: grob-transformer property func Create an override value good for applying FUNC to either pure or unpure values. FUNC is called with the respective grob as first argument and the default value (after resolving all callbacks) as the second.
Laissons-nous tenter. Les concepts de pur ou impur seront vus plus avant, dans Conteneurs requalifiants ; ignorons-les pour l’instant.
colorDefaultStencil = \once \override Script.stencil = #(grob-transformer 'stencil (lambda (grob orig) (stencil-with-color orig red))) { \colorDefaultStencil f'\espressivo }
Excellent !
- Pour obtenir un stencil redimensionné, il faut utiliser
-- Function: ly:stencil-scale stil x y Scale stencil STIL using the horizontal and vertical scaling factors X and optional Y (defaulting to X). Negative values flip or mirror STIL without changing its origin; this may result in collisions unless it is repositioned.
à partir du même chapitre de la référence des propriétés internes.
Résultat :
scaleColorDefaultStencil = \once \override Script.stencil = #(grob-transformer 'stencil (lambda (grob orig) (stencil-with-color (ly:stencil-scale orig ;; 1 is the neutral element with ly:stencil-scale, ;; i.e., scaling with '1 1' (for x- and y-axis) ;; returns a visibly unchanged stencil. 2 1) red))) { \scaleColorDefaultStencil f'\espressivo }
Mis à part la couleur, cela ressemble fort à ce que l’on voulait.
- Certains problèmes subsistent cependant, comme le montre l’exemple
suivant :
{ \scaleColorDefaultStencil f'_\espressivo ^\fermata }
- − Le point d’orgue est redimensionné lui aussi.
- − Devoir taper la commande avant les notes et
\espressivo
après ces notes n’est pas pratique.
Solution aux deux inconvénients : utilisons
\tweak
!{ f'-\tweak stencil #(grob-transformer 'stencil (lambda (grob orig) (stencil-with-color (ly:stencil-scale orig 2 1) red))) _\espressivo ^\fermata }
Cela fait ce qui est demandé, mais autant de saisie n’est pas la panacée ; pourquoi ne pas définir une variable que l’on utiliserait dans le tweak ? Après un peu de copier coller, nous obtenons
#(define longer-script (grob-transformer 'stencil (lambda (grob orig) (stencil-with-color (ly:stencil-scale orig 2 1) red)))) { f'-\tweak stencil #longer-script _\espressivo ^\fermata }
Cela fonctionne correctement, bien que les valeurs soient codées en dur et que taper
-\tweak stencil #longer-script
sans arrêt soit rébarbatif.Pour régler cela, commençons par introduire des variables dans
longer-script
pour les valeurs de redimensionnement x et y.#(define (longer-script x y) (grob-transformer 'stencil (lambda (grob orig) (stencil-with-color (ly:stencil-scale orig x y) red))))
Ensuite, et pour réduire la frappe, nous définissons une fonction événementielle – voir Fonctions événementielles. Les éléments du premier groupe entre parenthèses –
(x-val y-val)
– sont les variables de la fonction, et les éléments du second groupe entre parenthèses sont les types respectifs de variable (on cherche deux fois un nombre – voir Types de prédicat prédéfinis pour les possibilités de test).scaleEspr = #(define-event-function (x-val y-val) (number? number?) #{ \tweak stencil #(longer-script x-val y-val) \espressivo #}) { f'_\scaleEspr 2 1 ^\fermata }
Cela fonctionne à merveille ! Notez l’utilisation de
#{ … #}
: dans cette construction Scheme, il est tout à fait possible d’utiliser la syntaxe de LilyPond, et les arguments de la fonction, tel que x-val, sont correctement expansés – voir Blocs de code LilyPond.
- Si ’lon veut manipuler le stencil par défaut, il faut s’informer à son
sujet.
- Le code finalisé
Pour terminer, faisons un peu de ménage.
- − Nous souhaitons redimensionner uniquement dans l’horizontalité ; nous pouvons donc forcer le redimensionnement vertical dans la fonction événementielle.
- − Supprimons la couleur.
Ceci nous amène au code suivant.
#(define (longer-script x y) (grob-transformer 'stencil (lambda (grob orig) (ly:stencil-scale orig x y)))) longEspressivo = #(define-event-function (x-val) (number?) #{ \tweak stencil #(longer-script x-val 1) \espressivo #}) { f'^\longEspressivo #2 _\fermata }
Voilà !
Le principal bénéfice à l’utilisation de
grob-transformer
réside dans le fait que nul n’est besoin de connaître le nom exact de la fonction de stencil. Dans l’exemple qui suit, nous pivotons une liaison et une ligature de 90 degrés – que les fonction de stencil s’appellent respectivemently:slur::print
etly:beam::print
n’est d’aucun intérêt en l’occurrence.#(define rotate-90 (grob-transformer 'stencil (lambda (grob orig) (ly:stencil-rotate orig 90 0 0)))) { f'8( g') f'8-\tweak stencil #rotate-90 ( g') f'8[ g'] f'8-\tweak stencil #rotate-90 [ g'] }