;;; -*- mode: Lisp; Syntax: Common-Lisp; -*- ;;; ;;; Copyright (c) 2009 by the authors. ;;; ;;; See LICENCE for details. (in-package :hu.dwim.web-server) ;;;;;; ;;; Variables (def (special-variable e) *dojo-skin-name*) (def (special-variable e) *dojo-file-name*) (def (special-variable e) *dojo-directory-name*) ;;;;;; ;;; application-with-dojo-support (def (class* ea) application-with-dojo-support (application) ((dojo-skin-name "tundra") (dojo-file-name "dojo.js") ;; FIXME ? this way it gets captured at build-time, instead of start-time (dojo-directory-name (or (find-latest-js-library "dojo") "dojotoolkit/")))) (def method startup-broker :after ((self application-with-dojo-support)) (unless (dojo-directory-name-of self) (warn "The ~S slot of application ~A is not initialized by the time the server was started! Please refer to the install guide (e.g. on http://dwim.hu) or the sources for details on how to build dojo." 'dojo-directory-name self))) (def method call-in-application-environment :around ((application application-with-dojo-support) session thunk) (bind ((*dojo-skin-name* (or (dojo-skin-name-of application) *dojo-skin-name*)) (*dojo-file-name* (or (dojo-file-name-of application) *dojo-file-name*)) (*dojo-directory-name* (or (dojo-directory-name-of application) *dojo-directory-name*))) (call-next-method))) ;;;;;; ;;; Rendering (def special-variable *dojo-widgets*) (macrolet ((x (&body entries) `(progn ,@(iter (for (name dojo-name) :on entries :by #'cddr) (collect `(def (constant e) ,name ,dojo-name)))))) (x +dijit/accordion-container+ "dijit.layout.AccordionContainer" +dijit/border-container+ "dijit.layout.BorderContainer" +dijit/content-pane+ "dijit.layout.ContentPane" +dijit/split-container+ "dijit.layout.SplitContainer" +dijit/tab-container+ "dijit.layout.TabContainer" +dijit/button+ "dijit.form.Button" +dijit/drop-down-button+ "dijit.form.DropDownButton" +dijit/date-text-box+ "dijit.form.DateTextBox" +dijit/simple-text-area+ "dijit.form.SimpleTextarea" +dijit/text-box+ "dijit.form.TextBox" +dijit/time-text-box+ "dijit.form.TimeTextBox" +dijit/combo-box+ "dijit.form.ComboBox" +dijit/filtering-select+ "dijit.form.FilteringSelect" +dijit/number-text-box+ "dijit.form.NumberTextBox" +dijit/dialog+ "dijit.Dialog" +dijit/tooltip-dialog+ "dijit.TooltipDialog" +dijit/editor+ "dijit.Editor" +dijit/menu+ "dijit.Menu" +dijit/menu-bar+ "dijit.MenuBar" +dijit/menu-item+ "dijit.MenuItem" +dijit/menu-bar-item+ "dijit.MenuBarItem" +dijit/menu-separator+ "dijit.MenuSeparator" +dijit/popup-menu-item+ "dijit.PopupMenuItem" +dijit/popup-menu-bar-item+ "dijit.PopupMenuBarItem" +dijit/inline-edit-box+ "dijit.InlineEditBox" )) (pushnew 'dojo-widget-collector/wrapper *xhtml-body-environment-wrappers*) (def function dojo-widget-collector/wrapper (thunk) (bind ((*dojo-widgets* '())) (multiple-value-prog1 (funcall thunk) (when *dojo-widgets* ;; NOTE: instantiation must happen before any other js code tinkers with the dojo widgets, ;; therefore we wrap here again with the js script collapser to emit us before the parent ;; WITH-XHTML-BODY-ENVIRONMENT emits the rest of the js stuff. `xml,@(with-xhtml-body-environment (:wrappers '(js-script-collapser/wrapper)) ;; NOTE don't use `js-onload here (comes later in *xhtml-body-environment-wrappers*, and it's better to have a visually more standalone entry in the output for this) `js(on-load (hdws.io.instantiate-dojo-widgets (array ,@(iter (for entry :in *dojo-widgets*) (bind (((id dojo-type &rest dojo-properties) entry)) (collect (if dojo-properties `js-piece(create :node ,id :type ,dojo-type :inherited (create ,@(iter (for (name value) :on dojo-properties :by #'cddr) ;; we would render a js null here otherwise, and there's no (easy?) way to differentiate ;; between the two situations, so just drop the whole thing... (when value (collect (if (keywordp name) (string-downcase (symbol-name name)) name)) (collect value))))) `js-piece(create :node ,id :type ,dojo-type))))))))))))) (def (with-macro* e) render-dojo-widget (dojo-type &optional (dojo-properties '()) &key (id (generate-unique-string "_w"))) (multiple-value-prog1 (-with-macro/body- (id '-id-)) (push (list* id dojo-type dojo-properties) *dojo-widgets*))) #+() ; TODO decide about this (def (function e) render-dojo-widget* (dojo-type &optional (dojo-properties '()) &key (id (generate-unique-string "_w"))) (render-dojo-widget (dojo-type dojo-properties :id id)
))