(defpackage #:hsx/element (:use #:cl) (:export #:create-element #:element-type #:element-props #:element-children #:expand-component)) (in-package #:hsx/element) ;;;; class definitions (defclass element () ((type :reader element-type :initarg :type) (props :reader element-props :initarg :props) (children :reader element-children :initarg :children))) (defclass tag (element) ()) (defclass html-tag (tag) ()) (defclass fragment (element) ()) (defclass component (element) ()) ;;;; factory (defun create-element (type props &rest children) (let ((elm (make-instance (cond ((functionp type) 'component) ((eq type :<>) 'fragment) ((eq type :html) 'html-tag) ((keywordp type) 'tag) (t (error "element-type must be either a keyword or a function."))) :type type :props props :children (flatten children)))) (create-element-hook elm) elm)) (defun flatten (x) (labels ((rec (x acc) (cond ((null x) acc) ((atom x) (cons x acc)) (t (rec (car x) (rec (cdr x) acc)))))) (rec x nil))) (defmethod create-element-hook ((elm element))) (defmethod create-element-hook ((elm fragment)) (when (element-props elm) (error "Cannot pass props to fragment."))) (defmethod create-element-hook ((elm component)) ;dry-run to validate props (expand-component elm)) ;;;; methods (defmethod print-object ((elm tag) stream) (with-accessors ((type element-type) (props element-props) (children element-children)) elm (let ((type-str (string-downcase type))) (if children (format stream (if (rest children) "~@<<~a~a>~2I~:@_~<~@{~a~^~:@_~}~:>~0I~:@_~:>" "~@<<~a~a>~2I~:_~<~a~^~:@_~:>~0I~_~:>") type-str (props->string props) children type-str) (format stream "<~a~a>" type-str (props->string props) type-str))))) (defun props->string (props) (with-output-to-string (stream) (loop :for (key value) :on props :by #'cddr :do (let ((key-str (string-downcase key))) (if (typep value 'boolean) (format stream "~@[ ~a~]" (and value key-str)) (format stream " ~a=\"~a\"" key-str value)))))) (defmethod print-object ((elm html-tag) stream) (format stream "~%") (call-next-method)) (defmethod print-object ((elm fragment) stream) (with-accessors ((children element-children)) elm (if children (format stream (if (rest children) "~<~@{~a~^~:@_~}~:>" "~<~a~:>") children)))) (defmethod print-object ((elm component) stream) (print-object (expand-component elm) stream)) (defmethod expand-component ((elm component)) (with-accessors ((type element-type) (props element-props) (children element-children)) elm (apply type (merge-children-into-props props children)))) (defun merge-children-into-props (props children) (append props (and children (list :children children))))