ebc88989782f

Add direction weights and proper mutations
[view raw] [browse files]
author Steve Losh <steve@stevelosh.com>
date Mon, 08 Aug 2016 16:15:24 +0000 (2016-08-08)
parents 13cffbf17353
children c863300abd60
branches/tags (none)
files silt.lisp

Changes

--- a/silt.lisp	Mon Aug 08 15:06:30 2016 +0000
+++ b/silt.lisp	Mon Aug 08 16:15:24 2016 +0000
@@ -92,6 +92,26 @@
   (mapcar #'cdr ticklist))
 
 
+;;;; Weightlists ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
+(defstruct (weightlist (:constructor %make-weightlist))
+  weights sums items total)
+
+(defun make-weightlist (items weights)
+  "Make a weightlist of the given items and weights."
+  (%make-weightlist
+    :items items
+    :weights weights
+    :sums (prefix-sums weights)
+    :total (apply #'+ weights)))
+
+(defun weightlist-random (weightlist)
+  (iterate
+    (with n = (random (weightlist-total weightlist)))
+    (for item :in (weightlist-items weightlist))
+    (for weight :in (weightlist-sums weightlist))
+    (finding item :such-that (< n weight))))
+
+
 ;;;; Utils ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
 (defun manage-screen ()
   (multiple-value-bind (w h)
@@ -749,7 +769,8 @@
 ;;; Fauna
 (define-entity creature
     (coords visible sentient flavor metabolizing aging inspectable)
-  (name :accessor creature-name :initarg :name))
+  (name :accessor creature-name :initarg :name)
+  (directions :accessor creature-directions :initarg :directions))
 
 (define-entity corpse
     (coords visible flavor decomposing))
@@ -758,37 +779,77 @@
 (defparameter *directions*
   (iterate dirs (for dx :from -1 :to 1)
            (iterate (for dy :from -1 :to 1)
-                    (in dirs (collect (cons dx dy) :result-type 'vector)))))
+                    (in dirs (collect (cons dx dy))))))
+
+(defparameter *default-creature-directions*
+  (make-weightlist *directions*
+                   (iterate (repeat (length *directions*))
+                            (collect 10))))
+
+(defparameter *creature-colors*
+  (vector +color-white-black+
+          +color-blue-black+
+          +color-cyan-black+
+          +color-yellow-black+
+          +color-green-black+
+          +color-pink-black+))
+
+(defparameter *creature-glyphs*
+  (vector "@" "$" "?" "!" "&" "+"))
+
 
+(defun creature-mutate-glyph (c)
+  (setf (visible/glyph c) (random-elt *creature-glyphs*)))
+
+(defun creature-mutate-color (c)
+  (setf (visible/color c) (random-elt *creature-colors*)))
+
+(defun creature-mutate-directions (c)
+  (let ((old (creature-directions c)))
+    (setf (creature-directions c)
+          (make-weightlist (weightlist-items old)
+                           (mapcar (lambda (w)
+                                     (max 0 (random-around w 2)))
+                                   (weightlist-weights old))))))
+
+(defun creature-mutate-appearance (c)
+  (if (randomp)
+    (creature-mutate-color c)
+    (creature-mutate-glyph c)))
 
 (defun creature-mutate (c)
   (let ((v (random 1.0)))
-    (declare (ignore v))
     (cond
-      (t (setf (visible/glyph c)
-               (random-elt #("@" "$" "?" "!" "&" "+")))))))
+      ((< v 0.90) (creature-mutate-directions c))
+      ((< v 0.99) (creature-mutate-color c))
+      ((< v 1.00) (creature-mutate-glyph c)))))
+
 
 (defun creature-should-reproduce-p (c)
   (and (> (metabolizing/energy c) 1000)
        (< (random 1.0) 0.01)))
 
 (defun creature-should-mutate-p ()
-  (< (random 1.0) 0.01))
+  (< (random 1.0) 0.1))
 
 (defun creature-reproduce (parent)
-  (let ((energy (floor (metabolizing/energy parent) 2))
-        (child (make-creature (coords/x parent) (coords/y parent))))
-    (setf (metabolizing/energy parent) energy
-          (metabolizing/energy child) energy)
+  (let* ((energy (floor (metabolizing/energy parent) 2))
+         (child (make-creature (coords/x parent) (coords/y parent)
+                               :color (visible/color parent)
+                               :glyph (visible/glyph parent)
+                               :energy energy
+                               :directions (creature-directions parent))))
+    (setf (metabolizing/energy parent) energy)
     (when (creature-should-mutate-p)
       (creature-mutate child))
     (log-message "~A begets ~A." (creature-name parent) (creature-name child))))
 
+
 (defun creature-move (c)
   (let ((x (coords/x c))
         (y (coords/y c)))
     (destructuring-bind (dx . dy)
-        (random-elt *directions*)
+        (weightlist-random (creature-directions c))
       (coords-move-entity c (+ x dx) (+ y dy)))))
 
 (defun creature-eat (c food)
@@ -804,28 +865,33 @@
       (t (creature-move c)))))
 
 
-(defun make-creature (x y)
+(defun make-creature (x y &key
+                      (directions *default-creature-directions*)
+                      (color +color-white-black+)
+                      (glyph "@")
+                      (energy 2000))
   (let ((name (random-name)))
     (create-entity
       'creature
       :name name
+      :directions directions
       :coords/x x
       :coords/y y
-      :visible/color +color-white-black+
-      :visible/glyph "@"
-      :metabolizing/energy 2000
+      :visible/color color
+      :visible/glyph glyph
+      :metabolizing/energy energy
       :metabolizing/insulation 1
       :sentient/function 'creature-act
-      :inspectable/slots '(metabolizing/energy aging/birthtick aging/age)
+      :inspectable/slots '(name directions metabolizing/energy aging/birthtick aging/age)
       :flavor/text (list (format nil "A creature named ~:(~A~) is here." name)
                          "It likes food."))))
 
-(defun make-corpse (x y name)
+(defun make-corpse (x y color name)
   (create-entity
     'corpse
     :coords/x x
     :coords/y y
-    :visible/color +color-white-black+
+    :visible/color color
     :visible/glyph "%"
     :decomposing/rate 0.001
     :flavor/text (list (format nil "The corpse of ~:(~A~) lies here." name))))
@@ -837,6 +903,7 @@
                (aging/age c))
   (make-corpse (coords/x c)
                (coords/y c)
+               (visible/color c)
                (creature-name c)))
 
 
@@ -880,7 +947,7 @@
     (iterate
       (for creature :in (remove-if-not #'creature? (nearby f)))
       (unless (member creature (ticklist-contents recent))
-        (creature-mutate creature)
+        (creature-mutate-appearance creature)
         (ticklist-push recent creature 1000)
         (log-message "~A drinks from the fountain and... changes."
                      (creature-name creature))))))
@@ -1081,8 +1148,9 @@
                                      (mapcar (compose #'length #'symbol-name)
                                              slots)))
                 (for slot :in slots)
-                (collect (format nil "~vA ~A"
-                                 width slot (slot-value entity slot)))))
+                (collect (let ((*print-pretty* nil))
+                           (format nil "~vA ~A"
+                                   width slot (slot-value entity slot))))))
             :into text))
 
         (collecting "" :into text))