diff --git a/extras/body.html b/extras/body.html --- a/extras/body.html +++ b/extras/body.html @@ -9,6 +9,10 @@
 
 
+
+ + +
diff --git a/extras/default.css b/extras/default.css --- a/extras/default.css +++ b/extras/default.css @@ -29,6 +29,10 @@ flex: 3 3 30px; } +.qsp-col3 { + flex: 0 0 40px; +} + #qsp-main { flex: 6 6 60px; } @@ -58,7 +62,7 @@ outline: #9E9E9E outset 3px } -// Dropdown +/* Dropdown */ #qsp-dropdown { display: none; @@ -82,3 +86,18 @@ #qsp-dropdown a:hover { background-color: #ddd; } + +/* Buttons */ + +.qsp-col3 a, .qsp-col3 img { + width: 50px; + height: 50px; +} + +#qsp-btn-save { + background: url(''); +} + +#qsp-btn-load { + background: url(''); +} diff --git a/src/api.ps b/src/api.ps --- a/src/api.ps +++ b/src/api.ps @@ -36,6 +36,8 @@ (defm (root api base64-to-state) (data) (setf (root state-stash) (decode-u-r-i-component (atob data))) (let ((data (*j-s-o-n.parse (root state-stash)))) + (api-call clear-id :qsp-main) + (api-call clear-id :qsp-stat) (api-call clear-act) (setf (root vars) (ps:@ data vars)) (setf (root objs) (ps:@ data objs)) @@ -198,3 +200,11 @@ :do (incf i) :do (incf elt.inner-h-t-m-l (api-call make-menu-item-html i item.text item.icon item.loc))) (setf elt.style.display "block"))) + +;;; Content + +(defm (root api clean-audio) () + (loop :for k :in (*object.keys (root playing)) + :for v := (ps:getprop (root playing) k) + :do (when (ps:@ v ended) + (ps:delete (ps:@ (root playing) k))))) diff --git a/src/intrinsic-macros.lisp b/src/intrinsic-macros.lisp --- a/src/intrinsic-macros.lisp +++ b/src/intrinsic-macros.lisp @@ -128,6 +128,9 @@ ;;; 17sound +(ps:defpsmacro isplay (filename) + `(funcall (root playing includes) ,filename)) + ;;; 18img ;;; 19input diff --git a/src/intrinsics.ps b/src/intrinsics.ps --- a/src/intrinsics.ps +++ b/src/intrinsics.ps @@ -197,13 +197,13 @@ (defm (root lib delobj) (name) (let ((index (ps:chain (root objs) (index-of name)))) (when (> index -1) - (funcall (root lib killobj) index))) + (funcall (root lib killobj) (1+ index)))) (values)) -(defm (root lib killobj) (&optional num) - (if num - (ps:chain (root objs) (splice (1+ num) 1)) - (setf (root objs) (list))) +(defm (root lib killobj) (&optional (num nil)) + (if (eq nil num) + (setf (root objs) (list)) + (ps:chain (root objs) (splice (1- num) 1))) (api-call update-objs) (values)) @@ -232,13 +232,21 @@ ;;; 17sound -(defm (root lib play) ()) - -(defm (root lib isplay) ()) +(defm (root lib play) (filename &optional (volume 100)) + (let ((audio (ps:new (*audio filename)))) + (setf (ps:getprop (root playing) filename) audio) + (setf (ps:@ audio volume) (* volume 0.01)) + (ps:chain audio (play)))) -(defm (root lib close) ()) +(defm (root lib close) (filename) + (funcall (root playing filename) stop) + (ps:delete (root playing filename))) -(defm (root lib closeall) ()) +(defm (root lib closeall) () + (loop :for k :in (*object.keys (root playing)) + :for v := (ps:getprop (root playing) k) + :do (funcall v stop)) + (setf (root playing) (ps:create))) ;;; 18img diff --git a/src/main.ps b/src/main.ps --- a/src/main.ps +++ b/src/main.ps @@ -2,11 +2,22 @@ (in-package sugar-qsp) (setf (root) - (ps:create vars (ps:create) - objs (list) - state-stash (ps:create) - acts (ps:create) - locs (ps:create))) + (ps:create + ;;; Game session state + ;; Variables + vars (ps:create) + ;; Inventory (objects) + objs (list) + ;;; Transient state + ;; Savegame data + state-stash (ps:create) + ;; List of audio files being played + playing (ps:create) + ;;; Game data + ;; ACTions + acts (ps:create) + ;; Locations + locs (ps:create))) ;; Launch the game from the first location (setf window.onload