]> git.donarmstrong.com Git - lilypond.git/commitdiff
* scm/ly-from-scheme.scm (read-lily-expression): A variable
authorNicolas Sceaux <nicolas.sceaux@free.fr>
Fri, 14 May 2004 08:46:53 +0000 (08:46 +0000)
committerNicolas Sceaux <nicolas.sceaux@free.fr>
Fri, 14 May 2004 08:46:53 +0000 (08:46 +0000)
refering to a music expression can be used in lily-inside-scheme:
#{ $music #}

* lily/my-lily-parser.cc (LY_DEFINE): introduce ly:clone-parser
and ly:parser-define, and change ly:parser-parse-string in order
to make #{ $music #} work.

* scm/new-markup.scm (compile-markup-expression): when an argument
is a string, use `make-simple-markup'.

ChangeLog
lily/my-lily-parser.cc
scm/ly-from-scheme.scm
scm/new-markup.scm

index 222a4fe72137c7123496212cc4c7d9cbdf986561..baf09eb6979abf5d1fd77e1fc55a0c317f62d2ee 100644 (file)
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,16 @@
+2004-05-14  Nicolas Sceaux  <nicolas.sceaux@free.fr>
+
+       * scm/ly-from-scheme.scm (read-lily-expression):  A variable
+       refering to a music expression can be used in lily-inside-scheme:
+       #{ $music #}
+
+       * lily/my-lily-parser.cc (LY_DEFINE): introduce ly:clone-parser
+       and ly:parser-define, and change ly:parser-parse-string in order
+       to make #{ $music #} work.
+
+       * scm/new-markup.scm (compile-markup-expression): when an argument
+       is a string, use `make-simple-markup'.
+
 2004-05-14  Han-Wen Nienhuys   <hanwen@xs4all.nl>
 
        * lily/parser.yy (My_lily_lexer): bugfix; op should be tag.  
index af660e464632e42a20fc504df1786f31f3b8c8ae..0d6aeb34bca4331887d369cf675059932c200eed 100644 (file)
@@ -310,6 +310,27 @@ LY_DEFINE (ly_parse_string, "ly:parse-string",
   return SCM_UNSPECIFIED;
 }
 
+LY_DEFINE (ly_clone_parser, "ly:clone-parser",
+           1, 0, 0, 
+           (SCM parser_smob),
+           "Return a clone of PARSER_SMOB.")
+{
+  My_lily_parser *parser = unsmob_my_lily_parser (parser_smob);
+  My_lily_parser *clone = new My_lily_parser (*parser);
+  return scm_gc_unprotect_object (clone->self_scm ());
+}
+
+LY_DEFINE(ly_parser_define, "ly:parser-define",
+          3, 0, 0, 
+          (SCM parser_smob, SCM symbol, SCM val),
+          "Bind SYMBOL to VAL in PARSER_SMOB's module.")
+{
+  SCM_ASSERT_TYPE (ly_c_symbol_p (symbol), symbol, SCM_ARG1, __FUNCTION__, "symbol");
+  My_lily_parser *parser = unsmob_my_lily_parser (parser_smob);
+  parser->lexer_->set_identifier (scm_symbol_to_string (symbol), val);
+  return SCM_UNSPECIFIED;
+}
+
 LY_DEFINE (ly_parser_parse_string, "ly:parser-parse-string",
           2, 0, 0,
           (SCM parser_smob, SCM ly_code),
@@ -321,7 +342,7 @@ LY_DEFINE (ly_parser_parse_string, "ly:parser-parse-string",
 #endif
   SCM_ASSERT_TYPE (ly_c_string_p (ly_code), ly_code, SCM_ARG1, __FUNCTION__, "string");
 
-#if 0
+#if 1
   My_lily_parser *parser = unsmob_my_lily_parser (parser_smob);
   parser->parse_string (ly_scm2string (ly_code));
 #else
index 6a94976c92601ed705fd637b983fd9ae8ae31be2..f7928db4b3e4965b862f8acb66479418e3aa658c 100644 (file)
@@ -16,7 +16,7 @@
                                                                                             (char->integer #\0)))))
                                                  (string->list (number->string var-idx)))))))))
 
-(define-public (ly:parse-string-result str module)
+(define-public (ly:parse-string-result str parser module)
   "Parse `str', which is supposed to contain a music expression."
   (let ((music-sym (gen-lily-sym)))
     (ly:parser-parse-string
 the scheme music expression. The $ character may be used to introduce
 scheme forms, typically symbols. $$ may be used to simply write a `$'
 character."
-  (let* ((format-args '())
-         (lily-string (with-output-to-string
-                        (lambda ()
+  (let ((bindings '()))
+    (define (create-binding! val)
+      "Create a new symbol, bind it to `val' and return it."
+      (let ((tmp-symbol (gen-lily-sym)))
+        (set! bindings (cons (cons tmp-symbol val) bindings))
+        tmp-symbol))
+    (define (remove-dollars! form)
+      "Generate a form where `$variable' and `$ value' mottos are replaced
+      by new symbols, which are binded to the adequate values."
+      (cond (;; $variable
+             (and (symbol? form)
+                  (string=? (substring (symbol->string form) 0 1) "$")
+                  (not (string=? (substring (symbol->string form) 1 2) "$")))
+             (create-binding! (string->symbol (substring (symbol->string form) 1))))
+            (;; atom
+             (not (pair? form)) form)
+            (;; ($ value ...)
+             (eqv? (car form) '$)
+             (cons (create-binding! (cadr form)) (remove-dollars! (cddr form))))
+            (else ;; (something ...)
+             (cons (remove-dollars! (car form)) (remove-dollars! (cdr form))))))
+    (let ((lily-string (call-with-output-string
+                        (lambda (out)
                           (do ((c (read-char port) (read-char port)))
-                              ((and (char=? c #\#)
-                                    (char=? (peek-char port) #\}))
-                               (read-char port))
-                            (cond ((and (char=? c #\$)
-                                        (not (char=? (peek-char port) #\$)))
-                                   ;; a $variable
-                                   (display "~a")
-                                   (set! format-args (cons (read port) 
-format-args)))
-                                  ((and (char=? c #\$)
-                                        (char=? (peek-char port) #\$))
-                                   ;; just a $ character
-                                   (display (read-char port)))
-                                  (else
-                                   ;; other caracters
-                                   (display c))))))))
-   `(ly:parse-string-result (format #f ,lily-string ,@(reverse! format-args))
-                            (current-module))))
+                             ((and (char=? c #\#)
+                                   (char=? (peek-char port) #\})) ;; we stop when #} is encoutered
+                              (read-char port))
+                           (cond
+                            ;; a $form expression
+                            ((and (char=? c #\$) (not (char=? (peek-char port) #\$)))
+                             (format out "\\~a" (create-binding! (read port))))
+                            ;; just a $ character
+                            ((and (char=? c #\$) (char=? (peek-char port) #\$))
+                             (display (read-char port) out))  ;; pop the second $
+                            ;; a #scheme expression
+                            ((char=? c #\#)
+                             (format out "#~a" (remove-dollars! (read port))))
+                            ;; other caracters
+                            (else
+                             (display c out))))))))
+      `(let ((parser-clone (ly:clone-parser parser)))
+         ,@(map (lambda (binding)
+                  `(ly:parser-define parser-clone ',(car binding) ,(cdr binding)))
+                (reverse bindings))
+         (ly:parse-string-result ,lily-string parser-clone (current-module))))))
 
 (read-hash-extend #\{ read-lily-expression)
index 2d3632cdf86e731d82c751000d8bc8f937922245..7dcf129684903f9b270ce89cf1a09a50d26c0bf0 100644 (file)
@@ -175,8 +175,11 @@ e.g. (make-COMMAND-markup arg1 arg2 ...), and the rest expression."
          ;; expr === ((#:COMMAND arg1 ...) ...)
          (receive (m r) (compile-markup-expression (car expr))
                   (values m (cdr expr))))
+        ((and (pair? expr)
+              (string? (car expr))) ;; expr === ("string" ...)
+         (values `(make-simple-markup ,(car expr)) (cdr expr)))
         (else
-         ;; expr === (symbol ...) or ("string" ...) or ((funcall ...) ...)
+         ;; expr === (symbol ...) or ((funcall ...) ...)
          (values (car expr)
                  (cdr expr)))))