diff options
Diffstat (limited to 'stdlib/pic.tl')
-rw-r--r-- | stdlib/pic.tl | 119 |
1 files changed, 119 insertions, 0 deletions
diff --git a/stdlib/pic.tl b/stdlib/pic.tl new file mode 100644 index 00000000..6c2c8048 --- /dev/null +++ b/stdlib/pic.tl @@ -0,0 +1,119 @@ +;; Copyright 2021 +;; Kaz Kylheku <kaz@kylheku.com> +;; Vancouver, Canada +;; All rights reserved. +;; +;; Redistribution and use in source and binary forms, with or without +;; modification, are permitted provided that the following conditions are met: +;; +;; 1. Redistributions of source code must retain the above copyright notice, this +;; list of conditions and the following disclaimer. +;; +;; 2. Redistributions in binary form must reproduce the above copyright notice, +;; this list of conditions and the following disclaimer in the documentation +;; and/or other materials provided with the distribution. +;; +;; THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +;; ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +;; WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +;; DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +;; FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +;; DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +;; SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +;; CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +;; OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +;; OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +(defun expand-pic-num (fmt val) + (let* ((zero (or (starts-with "0" fmt) + (starts-with "+0" fmt) + (starts-with "-0" fmt))) + (plus (eql [fmt 0] #\+)) + (minus (eql [fmt 0] #\-)) + (exc (pos #\! fmt)) + (dot (or exc (pos #\. fmt))) + (fmt (if (and exc (eq #\! [fmt -1])) [fmt 0..-1] fmt)) + (int (if dot [fmt 0..dot] fmt)) + (fra (if dot [fmt (succ dot)..:] ""))) + (let ((code (if (or minus plus (not zero)) + ^(fmt ,`~@(len fmt),@(if plus "+")@(if zero "0")@(len fra)f` + ,val) + ^(fmt ,`~@(len fmt),-0@(len fra)f` + ,val)))) + (if exc + (with-gensyms (str) + ^(let ((,str ,code)) + (if (> (len ,str) ,(len fmt)) + ,(mkstring (len fmt) #\#) + ,str))) + code)))) + +(defun expand-pic-align (chr fmt val) + ^(fmt ,`~@(if chr chr)@(len fmt)a` ,val)) + +(defun pic-join-opt (join-form) + (labels ((et (str) (regsub #/\~/ "~~" str))) + (match-case join-form + ((join @(stringp @s) (fmt `@fmt` . @args) . @rest) + (pic-join-opt ^(join (fmt ,`@(et s)@fmt` ,*args) ,*rest))) + ((join (fmt `@fmt` . @args) @(stringp @s) . @rest) + (pic-join-opt ^(join (fmt ,`@fmt@(et s)` ,*args) ,*rest))) + ((join (fmt `@fmt1` . @args1) (fmt `@fmt2` . @args2) . @rest) + (pic-join-opt ^(join (fmt ,`@fmt1@fmt2` ,*args1 ,*args2) ,*rest))) + ((join @(stringp @s1) @(stringp @s2) . @rest) + (pic-join-opt ^(join ,`@s1@s2` ,*rest))) + ((join "" @item . @rest) + (pic-join-opt ^(join ,item ,*rest))) + ((join @item "" . @rest) + (pic-join-opt ^(join ,item ,*rest))) + ((join @item) item) + (@else else)))) + +(defun expand-pic (f fmt val) + (unless (stringp fmt) + (compile-error f "~s is required to be a format string" fmt)) + (cond + ([m^$ #/\~[~#<>\|\-+0.!]/ fmt] [fmt 1..2]) + ([m^$ #/\~./ fmt] (compile-error f "unrecognized escape sequence ~a" fmt)) + ([m^$ #/\~/ fmt] (compile-error f "incomplete ~~ escape")) + ([m^$ #/[+\-]?0?#+([.!]#+|!)?/ fmt] (expand-pic-num fmt val)) + ([m^$ #/<+/ fmt] (expand-pic-align "<" fmt val)) + ([m^$ #/>+/ fmt] (expand-pic-align nil fmt val)) + ([m^$ #/\|+/ fmt] (expand-pic-align "^" fmt val)) + (t (compile-error f "unrecognized format string ~s" fmt)))) + +(defmacro pic (:form f :env e bigfmt . args) + (let* ((regex #/[+\-]?0?#+([.!]#+|!)?| \ + <+| \ + >+| \ + \|+| \ + \~.|\~/)) + (labels ((pic-compile-string (fmtstr) + (let ((items (collect-each ((piece (tok regex t fmtstr))) + (cond + ((m^$ regex piece) + (cond + ((starts-with "~" piece) + (expand-pic f piece nil)) + (args + (expand-pic f piece (pop args))) + (t (compile-error + f "insufficient arguments for format")))) + (t piece))))) + (pic-join-opt ^(join ,*items))))) + (match-case bigfmt + (@(stringp @s) + (let ((out (pic-compile-string s))) + (if args + (compile-warning f "excess arguments")) + out)) + ((@(or sys:quasi) . @qargs) + (let ((nqargs (build (each ((q qargs)) + (if (stringp q) + (add (pic-compile-string q)) + (add q)))))) + (if args + (compile-warning f "excess arguments")) + ^(sys:quasi ,*nqargs))) + (@else (compile-error + f "~s is required to be a string or quasiliteral" else)))))) |