1.3.1 Première approche (exemple)

  1. 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 }
    

    [image of music]

  2. Les solutions envisageables
    1. créer un nouveau stencil
    2. redimensionner le stencil par défaut sur l’axe des X
  3. 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.

  4. 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 }
      

      [image of music]

      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 }
      

      [image of music]

      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 }
      

      [image of music]

      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 }
      

      [image of music]

      • − 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
      }
      

      [image of music]

      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
      }
      

      [image of music]

      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 }
      

      [image of music]

      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.

  5. 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 }
    

    [image of music]

    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 respectivement ly:slur::print et ly: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'] }
    

    [image of music]


GNU LilyPond – Extension des fonctionnalités v2.25.22 (development-branch).