2023-06-26 11:12:06 -05:00
;;;; Copyright © 2023, Jaidyn Ann <jadedctrl@posteo.at>
2023-06-04 04:04:34 -05:00
;;;;
;;;; This program is free software: you can redistribute it and/or
;;;; modify it under the terms of the GNU General Public License as
;;;; published by the Free Software Foundation, either version 3 of
;;;; the License, or (at your option) any later version.
;;;;
;;;; This program is distributed in the hope that it will be useful,
;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;;;; GNU General Public License for more details.
;;;;
;;;; You should have received a copy of the GNU General Public License
;;;; along with this program. If not, see <https://www.gnu.org/licenses/>.
2023-06-08 21:57:51 -05:00
;;;; FLORA-SEARCH-AURORA
;;;; A simple TUI-game made for the text-flavoured LibreJam of 2023-06!
;;;; See: https://jamgaroo.xyz/jams/2
2023-05-31 23:04:36 -05:00
2023-06-04 04:04:34 -05:00
( in-package :flora-search-aurora )
2023-06-04 02:33:53 -05:00
2023-06-26 11:12:06 -05:00
;;; ———————————————————————————————————
;;; Trans-map entity interactions
;;; ———————————————————————————————————
( defun take-item ( map entity-id )
" Take an entity from the MAP :ENTITIES and place it in the player ’ s pockets.
That is, into MAP ’ s :ITEMS. "
( let ( ( item-plist ( cdr ( getf-entity map entity-id ) ) ) )
( when item-plist
( setf ( aget-item map entity-id ) item-plist )
( removef-entity map entity-id ) ) ) )
( defun take-item-dialogue ( item-plist )
" Return some dialogue expressing surprise/dread or whatever at the collection
of a new item. The attributes set for the entity item should be:
ID
NAME-[EO | EN]
DESC-[EO | EN]
2023-06-27 14:15:12 -05:00
REMARK-[EO | EN]
2023-06-26 11:12:06 -05:00
REACTION-FACE
REACTION-TALKING
All are optional, save ID. "
( start-dialogue
2023-06-26 21:49:34 -05:00
( face 'player ( or ( getf item-plist :reaction-face ) "^_^" )
( or ( getf item-plist :reaction-talking ) "^o^" ) )
2023-06-26 11:12:06 -05:00
( mumble 'player :en ( format nil "(Hey, it's a ~A! ~A!)"
2023-06-26 21:49:34 -05:00
( or ( getf item-plist :name-en ) ( getf item-plist :id ) )
2023-06-27 14:15:12 -05:00
( or ( getf item-plist :remark-en ) "Nice!" ) )
:eo ( format nil "(Ho, jen ~A! ~A)"
2023-06-26 11:12:06 -05:00
( or ( getf item-plist :name-eo ) ( getf item-plist :id ) )
2023-06-27 14:15:12 -05:00
( or ( getf item-plist :remark-eo ) "Interese!" ) ) )
2023-06-26 11:12:06 -05:00
( mumble 'player :en ( if ( getf item-plist :desc-en )
2023-06-27 14:15:12 -05:00
( format nil "~A" ( getf item-plist :desc-en ) )
2023-06-26 11:12:06 -05:00
"(I'm glad I found it.)" )
:eo ( if ( getf item-plist :desc-eo )
2023-06-27 14:15:12 -05:00
( format nil "~A" ( getf item-plist :desc-eo ) )
2023-06-26 11:12:06 -05:00
"(Kia bonŝanco!)" ) ) ) )
( defun take-item-interact ( map interactee-id )
" Try to “ pick up ” the interactee entity, then have the player react to the
pickup. See TAKE-ITEM-DIALOGUE for customizing the reaction dialogue.
Should be the ` interact ` function for takeable items. "
( let ( ( item-plist ( cdr ( getf-entity map interactee-id ) ) ) )
( when ( take-item map interactee-id )
( make-dialogue-state map ( take-item-dialogue item-plist ) ) ) ) )
2023-06-27 14:15:12 -05:00
( defun description-interact ( map interactee-id )
2023-07-01 16:02:22 -05:00
" An interact-function that can be used for displaying simple descriptions of
items on the map. These descriptions can be set by the :DESC-EN and :DESC-EO
properties of an entity ( editable from Tiled ) . "
2023-06-27 14:15:12 -05:00
( let ( ( item-plist ( cdr ( getf-entity map interactee-id ) ) ) )
( make-dialogue-state
map
2023-07-01 16:02:22 -05:00
( apply #' start-dialogue
( loop for en-line in ( str:lines ( getf item-plist :desc-en ) )
for eo-line in ( str:lines ( getf item-plist :desc-eo ) )
collect ( mumble 'player :en en-line :eo eo-line ) ) ) ) ) )
( defun monologue-interact ( map interactee-id )
" An interact-function that can be used for displaying a monologue of a the
interactee — that is, they say the dialogue defined in the interactee ’ s
:SPEECH-EN and :SPEECH-EO properties. "
( let ( ( item-plist ( cdr ( getf-entity map interactee-id ) ) ) )
( make-dialogue-state
map
( apply #' start-dialogue
( loop for en-line in ( str:lines ( getf item-plist :speech-en ) )
for eo-line in ( str:lines ( getf item-plist :speech-eo ) )
collect ( say interactee-id :en en-line :eo eo-line ) ) ) ) ) )
( defun random-monologue-interact ( map interactee-id )
" An interact-function that can be used for displaying a random piece of
dialogue as a monologue of a the interactee — that is, they say the dialogue
defined in the interactee ’ s :SPEECH-EN and :SPEECH-EO properties.
Each line in :SPEECH-EN or :SPEECH-EO is treated as a seperate line of
dialogue, among which one will be selected randomly. "
( let* ( ( item-plist ( cdr ( getf-entity map interactee-id ) ) )
( eo-line ( alexandria:random-elt ( str:lines ( getf item-plist :speech-eo ) ) ) )
( en-line ( alexandria:random-elt ( str:lines ( getf item-plist :speech-en ) ) ) ) )
( make-dialogue-state
map ( start-dialogue ( say interactee-id :en en-line :eo eo-line ) ) ) ) )
2023-06-27 14:15:12 -05:00
2023-07-10 22:42:22 -05:00
( defun move-trigger ( map trigger-plist )
" A trigger function that forces the player to move either relatively or
absolutely, depending on the properties of TRIGGER-PLIST ( :ΔX & ΔY or X & Y ) .
It will also display a monologue of the :DESC-* properties before movement.
Useful for making barriers the player character refuses to traverse. "
( make-dialogue-state
map
( apply
#' start-dialogue
( append
( loop for en-line in ( str:lines ( getf trigger-plist :desc-en ) )
for eo-line in ( str:lines ( getf trigger-plist :desc-eo ) )
collect ( mumble 'player :en en-line :eo eo-line ) )
( list ( move 'player ( list :Δx ( getf trigger-plist :Δx )
:Δy ( getf trigger-plist :Δy )
:x ( getf trigger-plist :x )
:y ( getf trigger-plist :y ) ) ) ) ) ) ) )
2023-07-10 21:59:01 -05:00
( defun entrance-trigger ( map trigger-plist )
" A trigger that can be used to move the user from one MAP to another, via the
:MAP property in a trigger ’ s Tiled entity. "
2023-07-11 22:14:02 -05:00
( list :parameters
( list :map ( 🌍 :merge-maps map ( symbol-value ( read-from-string ( getf trigger-plist :map ) ) ) ) ) ) )
2023-07-10 21:59:01 -05:00
2023-07-12 21:54:00 -05:00
( defun item-refusal-lines ( string item )
" Given an ITEM ’ s symbol ID, return all speech lines according it.
These are encoded in an entity ’ s :ITEM-SPEECH-[EN | EO] properties, with the ID at
the start of each line followed by a tab and the speech line. "
( mapcar #' cadr
( remove-if-not
( lambda ( pair )
( eq ( … :string->symbol ( car pair ) ) item ) )
( mapcar
( lambda ( line )
( str:split #\tab line ) )
( str:lines string ) ) ) ) )
( defun refusal-use ( map item-plist entity )
" A generic use function for entities that can be used to display refusal
messages embedded in the Tiled map data. "
( let* ( ( refusal-en-lines ( item-refusal-lines ( getf-entity-data map entity :item-speech-en )
( … :string->symbol ( getf item-plist :id ) ) ) )
( refusal-eo-lines ( item-refusal-lines ( getf-entity-data map entity :item-speech-eo )
( … :string->symbol ( getf item-plist :id ) ) ) )
( en-lines ( if ( str:emptyp ( str:unlines refusal-en-lines ) )
( str:lines ( or ( getf-entity-data map entity :item-default-en ) "No thanks." ) )
refusal-en-lines ) )
( eo-lines ( if ( str:emptyp ( str:unlines refusal-eo-lines ) )
( str:lines ( or ( getf-entity-data map entity :item-default-eo ) "Ne, dankon." ) )
refusal-eo-lines ) ) )
( make-dialogue-state
map
( apply #' start-dialogue
( loop for eo-line in eo-lines
for en-line in en-lines
collect ( say entity :en en-line :eo eo-line ) ) ) ) ) )
( defun to-person-use ( &optional map item-plist )
" A function ran by INVENTORY 🎒 when an item is “ used ” , which will try to
run the :USE function of the nearest entity, if it has any. "
( let ( ( nearby ( car ( entities-near-entity ( getf-entity map 'player )
( gethash :entities map ) ) ) ) )
( if ( and nearby ( getf ( cdr nearby ) :use ) )
( let ( ( use-result ( funcall ( … :string->symbol ( getf ( cdr nearby ) :use ) )
map item-plist ( car nearby ) ) ) )
( typecase use-result
( list ( nconc ( list :drop 1 ) use-result ) )
( t use-result ) ) )
( nconc
2023-07-13 01:43:53 -05:00
( list :drop 1 )
( make-dialogue-state
map
( start-dialogue ( mumble 'player :en "(They don't seem to want it.)"
:eo "(Ĝi verŝajne ne volas tion.)" ) ) ) ) ) ) )
2023-07-12 21:54:00 -05:00
2023-06-25 14:40:12 -05:00
;;; ———————————————————————————————————
;;; The Outside World™
;;; ———————————————————————————————————
2023-07-06 16:27:55 -05:00
( defun factory-window-interact ( &optional map interactee-id )
( make-dialogue-state
map
( start-dialogue
( face 'player "` `" "`o`" )
( mumble 'player :eo "(Al ĉi tiu fenesto tute mankas vitro!)"
:en "(This window's got no pane at all!)" )
( mumble 'player :eo "(Mi kredeble povus grimpi tien, fakte...)"
:en "(I could probably fit my way in there, actually...)" )
( mumble 'player :eo "(... sed ĉu vere farindas?)"
:en "(... but should I?)"
:face "`o`" ) ) ) )
;;; ———————————————————————————————————
;;; Random outdoors NPCs
;;; ———————————————————————————————————
( defun kvincent-greetings ( map )
( case ( … :incf-0 ( getf-act map :kvincent-greetings ) )
( 0
( start-dialogue
( face 'kvincent "@__@" ">OO<" )
( face 'player ":w:" ":o:" )
( say 'kvincent :eo "AIIII! Ve! Ve! Diablo! Jen diablo!"
:en "AIIII! Woe is me! The Devil has come!" )
( say 'player :eo "Kvincent, Kvincent! Trankviliĝu, trankviliĝu! Estas mi!"
:en "Kvincent, Kvincent! Calm down, it's just me!" )
( say 'kvincent :eo "... bedaŭron?"
:en "... pardon?" )
( face 'kvincent "@w@" "@o@" )
( say 'kvincent :eo "Hooo, vi tute ne estas diablo! Vi estas nura homo!"
:en "Ooooh, you're not the Devil! You're just a person!"
:face "@v@" )
( say 'player :eo "Kompreneble!"
:en "Obviously not!" ) ) )
( otherwise
( start-dialogue
( face 'player "=w='" "=o='" )
( face 'kvincent "@__@" ">OO<" )
( say 'kvincent :eo "AJJJJ! Ve! Ve! Dia..."
:en "AIIII! Woe is me! Dev..." )
( say 'player :eo "Mi ankoraŭ ne estas diablo!!"
:en "I'm still no demon!!" )
( face 'kvincent "@w@" "@o@" )
( say 'kvincent :eo "Ha, jes. Pardonu."
:en "Oh, right. Sorry." ) ) ) ) )
( defun kvincent-dialogue ( map )
( append ( kvincent-greetings map )
( start-dialogue
( face 'player "` `" "`o`" )
( say 'player :eo "Ĉu ĉio enordas, Kvincent?"
:en "Everything alright, Kvincent?" )
( say 'kvincent :eo "Mi apenaŭ trovas fungojn, hodiaŭ... la dioj malbenis min!"
:en "I'm hardly finding any mushrooms... I've been cursed!" )
( say 'player :eo "Nek mi povas trovi florojn! Kia malfacila tago."
:en "I can't find any flowers, either! Today sucks."
:face "vov\'" ) ) ) )
( defun kvincent-interact ( map &optional interactee-id )
( make-dialogue-state map ( kvincent-dialogue map ) ) )
2023-06-24 17:01:12 -05:00
;;; ———————————————————————————————————
;;; Childhood friend (Sasha) arc
;;; ———————————————————————————————————
2023-06-23 22:25:16 -05:00
( defun childhood-friend-greetings ( map )
2023-06-26 15:43:01 -05:00
( … :incf-0 ( getf-act map :sasha-greetings ) )
2023-06-26 21:49:34 -05:00
( let ( ( sasha 'childhood-friend ) )
2023-06-23 22:25:16 -05:00
( case ( getf-act map :sasha-greetings )
( 0
( 💬 :start-dialogue
2023-06-26 21:49:34 -05:00
( 💬 :mumble sasha :en "..." )
( 💬 :say 'player :eo "Kielas apud la mar'?"
:en "How's the view?" )
( 💬 :face 'player "<.<" "<o<" )
( 💬 :say sasha :eo "Kielas apud la ruinoj de via viv'?"
:en "How's your trainwreck of a life?" ) ) )
2023-06-23 22:25:16 -05:00
( 1
( start-dialogue
2023-06-26 21:49:34 -05:00
( mumble 'player :en "..." )
( face 'player "<w<" )
( say sasha :eo "Kial vi restas? Ĉu tiom solecas ke nur ideas ĝeni min?"
:en "Why are you still here? Are you so lonely you've only got me to bother?" )
( face 'player ":w:" ":u:" )
( mumble 'player :eo "(Ŝi parolas pri si mem, ĉu ne?)"
:en "(She's projecting, isn't she?)" ) ) )
2023-06-23 22:25:16 -05:00
( 2
( start-dialogue
2023-06-26 21:49:34 -05:00
( face 'player ":w:" ":o:" )
( say 'player :eo "Nu... Vi staris tie ĉi senmove dum la pastintaj tri tagoj..."
:en "So... You've stood around here for three days already, you've hardly moved..." )
( say sasha :eo "Pŝ! Do?! Mi simple havas multajn pripensindaĵojn! Mi tiom multe okupiĝas!"
:en "Pff! So what?! My mind's just busy! I've got a lot going on right now!"
:face "vov" )
( say sasha :eo "Ne ŝajnigu vin supera al mi, dum vi mem senespere sencelas!!"
:en "Don't act all haughty when you're such an aimless loser yourself!!"
:face ">o<" )
( face 'player "=w=" "=u=" )
( mumble 'player :eo "Eee.. pardonu."
:en "Well... sorry." ) ) )
2023-06-23 22:25:16 -05:00
( 3
( start-dialogue
2023-06-26 21:49:34 -05:00
( say 'player :eo "Nu, vere, mia celo sufiĉe klaras al mi. Jam baldaŭ redungiĝos."
:en "I'm not too aimless, actually. I've got good job prospects, right about now."
:face "<w<" )
( say sasha :eo "Mi tute ne prizorgas."
:en "I really don't care." ) ) ) ) ) )
2023-06-23 22:25:16 -05:00
( defun childhood-friend-partings ( )
( let ( ( partings
' ( ( :eo "Nu? Ĝis! Adiaŭ!"
:en "Well? Bye! Ta-ta!" )
2023-06-26 21:49:34 -05:00
( :eo "Ve! Eĉ via rigardo malkomfortigas!"
:en "God! You're such a creep!" )
2023-06-23 22:25:16 -05:00
( :eo "Lasu! Min! Sooooola!"
:en "Leave me! The hell! Alooooone!" )
( :eo "Subvermo!"
:en "Worm!" ) ) ) )
( start-dialogue
2023-06-26 21:49:34 -05:00
( apply #' say ( append ' ( childhood-friend )
2023-06-23 22:25:16 -05:00
( nth ( random ( length partings ) ) partings ) ) ) ) ) )
( defun childhood-friend-dialogue ( map )
( let ( ( greetings ( getf-act map :sasha-greetings ) ) )
( cond ( ( or ( not greetings )
( < greetings 3 ) )
( childhood-friend-greetings map ) )
( 't
( childhood-friend-partings ) ) ) ) )
2023-06-26 11:12:06 -05:00
( defun childhood-friend-interact ( map &optional interactee-id )
2023-06-23 22:25:16 -05:00
( make-dialogue-state map ( childhood-friend-dialogue map ) ) )
2023-06-09 07:07:28 -05:00
2023-07-13 01:43:53 -05:00
( defun childhood-friend-dialogue-bracelet-intro ( sasha )
( start-dialogue
( face 'player "=w=" "=v=" )
( say 'player :eo "Nu, mi ankoraŭ opinias ke ĉi tio akordas vian stilon tre bone."
:en "I still think this suits you, you know." )
( face sasha ";w:" ":o:" )
( say sasha :en "..." )
( say sasha :eo "... vi ankoraŭ havas tion?"
:en "... you still have that?" )
( say 'player :eo "Kompreneble mi plu havas!"
:en "Of course I do!" )
( say 'player :eo "Sed mi ne plu; ĝi ja estas via."
:en "But I shouldn't; it's yours, after all." )
( say sasha :en "..." ) ) )
( defun childhood-friend-dialogue-bracelet-bad-end ( sasha )
( start-dialogue
( face sasha "v-v" "vov" )
( face 'player ":w;" ":o;" )
( say sasha :eo "Pŝ!"
:en "Psh!" )
( say sasha :eo "Ankoraŭ estas la malĝusta koloro, do..!"
:en "It's still the wrong colour, so..!"
:face "<o<" )
( say sasha :eo "Lasu min!"
:en "Leave me alone!"
:face ">o<" )
( say 'player :eo "Kial vi ankoraŭ forpuŝas min ĉi tiom?"
:en "Why do you still push me away this much?" )
( say sasha :eo "Puŝas? Bona ideo, mi falpuŝu vin al la maro!"
:en "Push? Good idea, I'll push you off this cliff!" )
( say 'player :eo "Vi ankoraŭ kompenas nenion ajn, Saŝa."
:en "You still don't get it at all, Sasha." )
( say 'player :eo "... kaj ne puŝu min, bonvole!!"
:en "... and please don't push me!!"
:face "@o@" ) ) )
( defun childhood-friend-dialogue-bracelet-good-end ( sasha )
( start-dialogue
( face sasha "=v=\"" "=v=\"" )
( say sasha :eo "Nu, dankon..."
:en "Well, thanks..." )
( say sasha :eo "[SAŜA prenas ĈIRKAŬMANON de vi.]"
:en "[SASHA takes BRACELET from you.]" )
( face sasha "<w<" "<w<" )
( say sasha :eo "Verdire, mi mensogetis tiam..."
:en "To be honest, I lied a bit, back then..." )
( say sasha :eo "Ĉi tiu koloro fakte tre plaĉas al mi."
:en "I actually really like this colour."
:face ">w>" )
( say 'player :eo "Mi ĝojas, ke finfine estos via."
:en "I'm glad it's finally yours." )
( say sasha :eo "Mi suportos ĝin ĉiutage!"
:en "I'll wear it every day!"
:face "^o^" )
( say sasha :eo "... ho. Fek! Nu!"
:en "... wait. Shit!"
:face "<w<\"" )
( say sasha :eo "Simple pro la aspekto! Ne gravas al mi, ĉu estas donaco aŭ ne!!"
:en "Just because it's pretty! I don't care that it's a gift or whatever!!"
:face ">o<" )
( face 'player "` `" "`o`" )
( mumble 'player :eo "(Mi supozas, ke la kerno de homo ne tro multe sanĝiĝas...)"
:en "(At the end of the day, I guess people can't change too much...)" )
( mumble 'player :eo "(Sed tamen."
:en "(But y'know what? This is good enough for me.)" )
( face 'player "^w^" "^o^" )
( mumble 'player :eo "(Ĉi tio sufiĉas al mi!)"
:en "(This is good enough for me!)" ) ) )
( defun childhood-friend-dialogue-bracelet ( map sasha )
( append ( childhood-friend-dialogue-bracelet-intro sasha )
( if ( getf-act map :sasha-flourish )
( childhood-friend-dialogue-bracelet-good-end sasha )
( childhood-friend-dialogue-bracelet-bad-end sasha ) ) ) )
( defun childhood-friend-use ( map item-plist &optional entity-id )
( let ( ( item-id ( … :string->symbol ( getf item-plist :id ) ) ) )
( cond ( ( eq item-id 'bracelet )
( make-dialogue-state
map
( childhood-friend-dialogue-bracelet map entity-id ) ) )
( 't
( refusal-use map item-plist entity-id ) ) ) ) )
2023-06-25 14:40:12 -05:00
;;; ———————————————————————————————————
2023-06-27 14:15:12 -05:00
;;; School prologue: Childhood friend
2023-06-25 14:40:12 -05:00
;;; ———————————————————————————————————
2023-07-13 22:55:16 -05:00
( defparameter *flashback-school*
( list
:title ' ( :eo "ANTAŬLUDO I" :en "PROLOGUE I" )
:subtitle ' ( :eo "Antaŭ kvar jaroj..." :en "Four years ago..." )
:side-bar ' ( :eo "Amikoj, ĉu? Mi ne scias, eble..." :en "Friends? I don't know, maybe..." )
:map *flashback-school-map* ) )
2023-07-13 01:43:53 -05:00
( defun flashback-school-trigger ( map &optional trigger-plist )
2023-06-27 19:30:01 -05:00
" This is triggered right as the player enters the map — they literally can 't
avoid triggering this. "
( take-item map 'bracelet )
't )
2023-07-13 01:43:53 -05:00
( defun flashback-childhood-friend-dialogue-intro ( sasha )
( start-dialogue
( face 'player "` `" "`o`" )
( say 'player :eo "Ĉu ĉio enordas, Saŝa? Iom malfruas, ĉu ne?"
:en "Is everything OK, Sasha? It's a bit late, isn't it?" )
( face sasha "=_=" "=o=" )
( say sasha :eo "Ho, jes, mi simple ĵus eliris klubkunvenon."
:en "Yea, I just left a club-meeting, is all." )
( say 'player :eo "Hodiaŭ ne estas klubotago..."
:en "Today isn't club day..." )
( say sasha :eo "Nu, estas escepte speciala klubo!"
:en "Well, whatever, it's a special club!"
:face "<o<" )
( say sasha :eo "Nu, kial VI restas? Ĉu ankoraŭ havas ne amikojn?"
:en "And what're YOU doing here? Still no friends?" )
( say 'player :eo "Ankoraŭ sole vin."
:en "Still just you." )
( face sasha ":v:" ":o:" )
( mumble sasha :en "..." )
( say sasha :eo "Nu..."
:en "Well..." )
( mumble sasha :en "..." )
( face sasha "<-<" "<o<" )
( face 'player ":w:" ":u:" )
( say sasha :eo "Pŝ! Fermu la buŝon, vermo!"
:en "Ugh! Just shut up, you loser!" )
( face sasha ">->" ">o>" )
( say sasha :eo "Kvazaŭ ni povus esti tiel!"
:en "As if!" )
( face sasha "<-<" "<o<" )
( say sasha :eo "Simple lasu min al paco, fek'!!"
:en "Just get the hell out of my face!!"
:face ">o<" )
( say 'player :eo "Bedaŭron..."
:en "Sorry..." )
( move 'player ' ( :x 46 :y 11 ) :delay .05 )
( face 'player "^_^" "^o^" )
( say 'player :eo "Ho, jes!"
:en "Oh, yea!" )
( move sasha ' ( :x 36 :y 3 ) :delay .03 )
( move 'player ' ( :x 43 :y 4 ) :delay .05 )
( move sasha ' ( :x 37 :y 3 ) ) ) )
( defun flashback-childhood-friend-dialogue-bracelet ( map sasha )
( start-dialogue
( say 'player :eo "Mi freŝe trovis ĉi tion, ĝi ŝajnis akorda al via stilo."
:en "I found this a while back, I thought you'd like it." )
( face sasha ";w:" ";u:" )
( mumble 'player :eo "[Vi donas al SAŜA ĉirkaŭmanon belbrilan.]"
:en "[You give SASHA a shiny bracelet.]" )
( say sasha :eo "Ho, tio surprize afablis...."
:en "Oh, that's surprisingly nice..." )
( face sasha "<w<" "<o<" )
( face 'player ":w;" ":o;" )
( say sasha :eo "... jen la sola koloro, kiun mi malamas."
:en "... this is literally the one color I hate." )
( say sasha :eo "Kial vi ne elektis bluan? Dio mia."
:en "You couldn't have gotten blue?" )
( say sasha :eo "Mi jam bone sciu, ne atendi bonon de vi."
:en "I should know not to expect so much from you..." )
( say sasha :eo "Bonaj \"amikoj\" ni estas, ja..."
:en "\"Friends,\" whatever..." )
( face 'player "T_T" "ToT" )
( move 'player ' ( :x 41 :y 3 ) :delay .05 )
( face sasha ":_;" )
( mumble 'player :eo "[Vi prenas de SAŜA ĉirkaŭmanon belbrilan.]"
:en "[You take a shiny bracelet from SASHA.]" )
( move 'player ' ( :x 46 :y 5 ) :delay .05 )
2023-07-13 22:55:16 -05:00
( move 'player ' ( :x 51 :y 19 ) :delay .05 )
( list ( make-leave-flashback-state map ) ) ) )
2023-07-13 01:43:53 -05:00
( defun flashback-childhood-friend-dialogue ( map sasha )
( append ( flashback-childhood-friend-dialogue-intro sasha )
( flashback-childhood-friend-dialogue-bracelet map sasha ) ) )
( defun flashback-childhood-friend-interact ( map entity-id )
2023-06-27 14:15:12 -05:00
( make-dialogue-state
map
2023-07-13 01:43:53 -05:00
( flashback-childhood-friend-dialogue map entity-id ) ) )
( defun flashback-childhood-friend-use ( map item-plist &optional entity-id )
( if ( eq ( … :string->symbol ( getf item-plist :id ) ) 'bracelet )
;; If player gives her the special bracelet, skip the dialogue intro
( make-dialogue-state
map
( flashback-childhood-friend-dialogue-bracelet map entity-id ) )
;; Otherwise, have her politely refuse. =w=
( refusal-use map item-plist entity-id ) ) )
2023-06-27 19:30:01 -05:00
2023-06-25 14:40:12 -05:00
2023-06-27 14:15:12 -05:00
;;; ———————————————————————————————————
;;; Casino!
;;; ———————————————————————————————————
2023-06-24 17:01:12 -05:00
2023-07-13 01:43:53 -05:00
( defun boozy-lady-dialogue-ring ( )
( start-dialogue
( say 'boozy-lady :eo "Vi volas edzinigi min, belulo?"
:en "Ya wanna marry me, hot stuff?" )
( say 'boozy-lady :eo "Booone, niiru tuj!"
:en "Shuree, lez go now!" )
( say 'boozy-friend :eo "Vi NE iros tuj."
:en "You \"shure\" as hell WON'T." ) ) )
( defun boozy-lady-dialogue-wallet ( )
( start-dialogue
( say 'boozy-lady :eo "...? Monujo?"
:en "...? Wallet?" )
( say 'boozy-lady :eo "Monujo, mono! Mono, biero!!"
:en "Wallet, money! Money, booze!" )
( say 'boozy-friend :eo "... kaj vi certe ne bezonas pli da TIO."
:en "You don't need any more of THAT, hon." ) ) )
( defun boozy-lady-dialogue-bracelet ( )
( start-dialogue
( face 'player "^_^" "^o^" )
( say 'boozy-lady :eo "Ooo, brile! Ĝi belasss"
:en "Ooo, shiny! It's prettyy" )
( say 'boozy-lady :eo "Ĉu por mji?"
:en "Is it for mi?" )
( say 'player :eo "Estos via, se vi ĵuras ne plu drinki ĉi-nokte."
:en "It's yours, if you promise to slow your roll a bit." )
( say 'boozy-lady :eo "Jeŝ! Ne pluuu!"
:en "Okieeee!"
:face "= w =" )
( say 'boozy-lady :eo "[MAJA prenas ĈIRKAŬMANON de vi.]"
:en "[MAJA takes BRACELET from you.]"
:face "= w =" )
( face 'player ">_<" ">o<" )
( say 'boozy-lady :eo "Plia biero por mi, sinjoro!!"
:en "Jkjk. One more for me, barkeep!!" )
( say 'casino-bartender
:eo "Bone, momenton!!"
:en "Sure thing, lady!"
:face "xD " ) ) )
( defun boozy-lady-use ( map item-plist &optional entity-id )
( let ( ( item-id ( … :string->symbol ( getf item-plist :id ) ) ) )
( cond ( ( eq item-id 'ring )
( make-dialogue-state
map
( boozy-lady-dialogue-ring ) ) )
( ( eq item-id 'bracelet )
( make-dialogue-state
map
( boozy-lady-dialogue-bracelet ) ) )
( ( eq item-id 'wallet )
( make-dialogue-state
map
( boozy-lady-dialogue-wallet ) ) )
( 't
( refusal-use map item-plist entity-id ) ) ) ) )
2023-06-24 17:01:12 -05:00
;;; ———————————————————————————————————
;;; Destitute Gambler arc
;;; ———————————————————————————————————
( defun bad-gambler-greetings ( map )
2023-06-26 15:43:01 -05:00
( … :incf-0 ( getf-act map :gambler-greetings ) )
2023-06-26 21:49:34 -05:00
( let ( ( gambler 'bad-gambler ) )
2023-06-24 17:01:12 -05:00
( case ( getf-act map :gambler-greetings )
( 0
( start-dialogue
( face gambler "xD " )
( mumble gambler :en "Hahaha... haha." )
( say gambler :eo "Kia spektalo! Hahaha!"
2023-06-24 19:49:18 -05:00
:en "Good one! Hahaha!"
:face "xDD" )
2023-06-24 17:01:12 -05:00
( say gambler :en "Hahahaha! Hahahahaha!"
2023-06-24 19:49:18 -05:00
:face "x'D" )
( face 'player "^^'" )
( face gambler ">V<" ">O<" )
2023-06-24 17:01:12 -05:00
( say gambler :eo "Tiom amuze! Bona ŝerco!"
:en "Shit that's funny!"
:face ">V<" )
2023-06-24 19:49:18 -05:00
( face 'player "^^\"" )
2023-06-24 17:01:12 -05:00
( say gambler :eo "Mi tute ruinigis mian vivon! MDR!"
:en "I totally fucked my life! LMAO!"
:face ">V<" )
2023-06-24 19:49:18 -05:00
( face 'player "o-o" "ouo" )
( say gambler :en "HAHAHAHAHAHAHAHAHAHAAHAHAHAHAHAHAHAHAHAHAHAHAHAHAHAHAHAHAHAHAAHAHAHAHAHAHAHAHAHAHAHAHA" )
( face gambler "=w=" "=w=" )
2023-06-24 17:01:12 -05:00
( mumble gambler :en "..." )
( mumble gambler :eo "Fek'."
:en "Fuck." ) ) )
( 1
( start-dialogue
( say 'player :eo "Ĉu... ĉio enordas, samideano?"
2023-07-03 22:12:42 -05:00
:en "Everything alright, man..?" )
2023-06-24 17:01:12 -05:00
( say gambler :eo "Jes! Tutorde! Bonas!"
:en "Yup! For sure! All good!" )
( say gambler :eo "Mi simple trafis iujn monproblemojn, ne problemas."
:en "I've just hit a slight snag financially, it's no problem." )
( say gambler :eo "Nur povas ne repagi monprunton."
:en "I just can't repay my mortgage." )
( say gambler :eo "... kiun mi pruntis por repagi alian monprunton."
:en "... that I took out to pay back a loan." )
( say gambler :eo "... kiun mi pruntis por subteni la kompanion de mia frato."
:en "... that I borrowed to support my bro's company." )
( say gambler :eo "... kiu estas senpaga lernejo por handikapuloj."
:en "... that's a non-profit school for disabled kids." )
( say gambler :eo "... danke al kiu mia filino povas edukiĝi bone."
:en "... thanks to which my daughter can receive a good education." )
2023-06-24 20:05:10 -05:00
( say gambler :eo "Kial mi forĵetis la monon tie ĉi?! FEK'!!"
:en "Why'd I waste it all here?! FUCK!!"
2023-06-24 19:49:18 -05:00
:face ">O<" ) ) )
2023-06-24 17:01:12 -05:00
( 2
( start-dialogue
( say gambler :eo "Nu, ĉio enordas! Ni simple perdos la domon, jen ĉio."
:en "Anyways, all good! We'll just lose the house, that's all." )
( say gambler :eo "Kompreneble, perdinte la domon, mia edzino lasos min, kunprenante la gefilojn..."
:en "Obviously, after losing the house, my wife'll take leave with the kids..." )
( say gambler :eo "Dommastrino sen domo ja ne sencas, do!"
:en "A house-mistress without a house is no mistress at all, of course!" )
( say gambler :eo "Kaj kromvirino sen edzino ja ne sencas, do lasos min ankaŭ ŝi..."
:en "And a side-piece without a 'main-piece' is no side-piece at all, so she'll leave me too..." )
( say gambler :eo "Neniu mono, neniu domo, neniu filo, neniu edzino, neniu kromvirino!"
:en "No money, no house, no children, no wife, no mistress!" )
( say gambler :eo "Neniu vivo!"
:en "No life!" ) ) )
( 3
( start-dialogue
2023-07-03 22:12:42 -05:00
( say gambler :eo "Mi cedu al la vakuo. Tre komfortas ĉe la fundo, kara."
:en "I'm giving into the void. It's quite peaceful down here, buddy." )
2023-06-24 19:49:18 -05:00
( say gambler :eo "Mi lasu min falu entute, ĉu ne, kara amiko? Ĉu neee?"
:en "I should let go, right, pal? Righttt?" )
( face 'player ";w;" ";o;" )
( say 'player :eo "Ne tro senesperu -- sinjoro, restas al vi fuĝvojo, sendube!"
:en "Guy, there's still hope for you, somewhere!" )
( face gambler "=v=" "=v=" )
( say gambler :eo "Mi ideas IAN fuĝvojon..."
:en "Five feet under, maybe..." ) ) ) ) ) )
2023-06-24 17:01:12 -05:00
2023-06-26 21:49:34 -05:00
( defun bad-gambler-partings ( &optional map )
2023-06-24 17:01:12 -05:00
( let ( ( messages
' ( ( :eo "... kaj ŝiaj mamoj tiom belis..."
:en "... her titties were so nice, too..." )
( :eo "... vakuo, nenio, detruo..."
:en "... emptiness, nothing, destruction..." )
( :eo "... vivo, bruo. morto, paco..."
:en "... life, noise. death, peace..." )
( :eo "... laco, kaco, maĉo-veneno..."
:en "... tired, weary, dreary..." )
( :eo "... tempo, tik-tok, malfruo..."
:en "... time, tick-tock, too late..." )
( :eo "... finiĝo, ĉesiĝo, paciĝo..."
:en "... endings, partings, peace-ings..." ) ) ) )
( start-dialogue
2023-06-26 21:49:34 -05:00
( apply #' say ( append ' ( bad-gambler )
2023-06-24 17:01:12 -05:00
( nth ( random ( length messages ) ) messages ) ) ) ) ) )
( defun bad-gambler-dialogue ( map )
( let ( ( greetings ( getf-act map :gambler-greetings ) ) )
( cond ( ( or ( not greetings )
( < greetings 3 ) )
( bad-gambler-greetings map ) )
( 't
( bad-gambler-partings ) ) ) ) )
2023-06-26 11:12:06 -05:00
( defun bad-gambler-interact ( map &optional interactee-id )
2023-06-24 17:01:12 -05:00
( make-dialogue-state map ( bad-gambler-dialogue map ) ) )
2023-07-03 22:12:42 -05:00
;;; ———————————————————————————————————
;;; Casino prologue: Bad gambler!
;;; ———————————————————————————————————
2023-07-13 22:55:16 -05:00
( defparameter *flashback-casino*
( list
:title ' ( :eo "ANTAŬLUDO II" :en "PROLOGUE II" )
:subtitle ' ( :eo "Antaŭ du jaroj..." :en "Two years ago..." )
:side-bar ' ( :eo "anyway whatever bruh" :en "sldkajsldkja" )
:map *flashback-casino-map* ) )
2023-07-03 22:12:42 -05:00
( defun flashback-bad-gambler-dialogue ( )
( let ( ( gambler 'flashback-bad-gambler ) )
( start-dialogue
( say gambler :eo "Kia spektalo! Hahaha!"
:en "How nice! Hahaha!" )
( face gambler ">v<" ">u<" )
( face 'player "o-o" )
( say gambler :en "HAHAHAHAHAHAHAHAHAHAHAHA!" )
( face 'player "=w='" )
( say gambler :eo "Hehe-Heŭreka! Ĉi-nokte, Sorto estas mia kromvirino!"
:en "Haha-Hot damn, lady luck's my mistress tonight!" )
( face gambler "^#^" "^O^" )
( say gambler :eo "ĉu vi sciis, kara?"
:en "Guess what, buddy?" )
( say gambler :eo "Por tiu ĉi ludo mi havas feke legendan vetaĵon!"
:en "I've got one hell of a chip for this game!" )
( face 'player "^_^" "^o^" ) ) ) )
( defun flashback-bad-gambler-interact ( map &optional interactee-id )
( make-dialogue-state map ( flashback-bad-gambler-dialogue ) ) )
( defun flashback-casino-dialogue-intro ( )
( let ( ( father 'flashback-casino-father )
( mother 'flashback-casino-mother )
( dealer 'flashback-casino-dealer )
( geezer 'flashback-casino-geezer )
( gambler 'flashback-bad-gambler ) )
( start-dialogue
( say dealer :eo "Ĉu ĉiu pretas?"
:en "Everyone ready?" )
( say geezer :eo "Pretis jam antaŭ vi eĉ naskiĝis!"
:en "I've been ready since before you were born!" )
( say gambler :eo "Ĉu necesas demandi?"
:en "Need you ask?" )
( say father :eo "Ni pretas, ankaŭ!"
:en "We're ready, too!" )
( say 'player :en "Eee...!"
:face ";w:" )
( say father :eo "Ho, jes, vi ne scipovas ludi."
:en "Oh, right, you still don't know how to play." )
( say father :eo "Atendu momente, ĉiu, mi klarigos rapide!"
:en "Hold on, everyone, we'll be quick!" )
( say father :eo "Bone: Vikĉjo, atentu! Mi ne min ripetos!"
:eo "Alright, Vic! Listen up, 'cuz I'm not gonna repeat myself!" )
( say father :eo "La ludestro disdonas kartojn hazardkaŝe. La kartvaloro sekvas la numerojn, krom se estas kolora karto. Se estas kolora karto, la valoro sekvas la Epokon de la ludo. La unua Epoko estas Antaŭhistorio, kaj kompreneble plej valoras tiam la..."
:en "The dealer hands out cards randomly and discretely. The cards' value is based on their number, unless they're a coloured card, in which case the value is based on the Era of the game. The first Era is Pre-history, and of course the most imporant card in Pre-history is..." )
( face 'player "=w='" "=o='" )
( say 'player :eo "(Mi ĝojas festi kun ĉiu, sed ĉi tiu ludo sonas jam tro malsimpla por mi...)"
:en "(I'm glad to party like this with my folks, but this game sounds a bit complicated...)" )
( say 'player :eo "(Ĉu ne ni povus simple ludi Hanafudan?)"
:en "(Why can't we just play Hanafuda?)"
:face "=w='" )
( say father :eo "... kaj tiel oni venkas la markizon." )
( say father :eo "Bone, jen ĉio!" )
( say gambler :eo "Finfine!!" ) ) ) )
( defun flashback-casino-dialogue-bet ( map )
( let ( ( father 'flashback-casino-father )
( mother 'flashback-casino-mother )
( dealer 'flashback-casino-dealer )
( geezer 'flashback-casino-geezer )
( gambler 'flashback-bad-gambler ) )
( append
( start-dialogue
( say gambler :eo "Jen, mi vetas ĉi tion!"
:en "Now, I'm putting this down!" )
( say gambler :eo "Momenton..."
:en "Wait..."
:face "`o`" )
( say gambler :eo "He?! Mi perdis la geeziĝan ringon?!"
:en "Wait... what?! I lost my wedding ring?!"
:face "O_O" ) )
( if ( aget-item map 'ring )
( start-dialogue
( face 'player "^_^" "^o^" )
( say 'player :eo "Ho! jen!"
:en "Ĉu estas tia ĉi?" )
( mumble 'player :eo "[Vi donas RINGON al HAZARDLUDEMULO]"
:en "[You give RING to GAMBLER]" )
( face gambler "=w=" "=w=" )
( say gambler :eo "..."
:en "..." )
( face gambler ":W:" ":O:" )
( say gambler :eo "Je dio!! Dankegon, kara!!"
:en "God damn!! Thank you, buddy!!" )
( say gambler :eo "Mi preskaŭ tiom senesperis!"
:en "I almost gave into despair!" )
( face gambler "^#^" "^O^" )
( say gambler :eo "NUN mi pretas ludi."
:en "NOW I'm ready to play." )
( mumble gambler :eo "[HAZARDLUDEMULO donas GEEDZIĜAN RINGON al TABLO]"
:en "[GAMBLER gives WEDDING RING to TABLE" )
( say 'player :eo "Ĉu vere?! Kio misas je vi!?!"
:en "Seriously?! What's wrong with you!?!"
:face ">O<" )
( face 'player ":w:'" )
( say gambler :en "Live fast die hard, brother!"
:eo "Vivu akre mortu frue!"
:face "B-)" )
( say mother :eo "Vi provis, Vikĉjo..."
:en "You tried, Vic..." ) )
( start-dialogue
( say gambler :eo "... bone do, tio ĉi sufiĉos."
:en "... fine then, this'll have to do."
:face "<v<" )
( mumble gambler :eo "[HAZARDLUDEMULO donas BIENPOSEDON al TABLO]"
:en "[GAMBLER gives LAND OWNERSHIP to TABLE]" )
( face 'player ":w:" ":0:" )
( say 'player :eo "Ĉu vi frenezas?! Ne tion faru!!"
:en "Are you crazy?! Don't do that!!" )
( say mother :eo "Lasu lin, Vikĉjo."
:en "Leave him, Vic." )
( say mother :eo "La fonto de amuzo estas senzorga risko!"
:en "The root of all of joy is careless, unthinking risk!" )
( say father :eo "Tielas la animokerno de la kartoj."
:en "Such is the nature of the cards." ) ) ) ) ) )
( defun flashback-casino-outro ( map )
( start-dialogue
( say 'flashback-casino-dealer
:eo "Nu, ĉiu krom li, metu viajn vetaĵojn. Ni komencos je la Nula Epoko!"
:en "As for the rest of you, place your bets. It's time for the Zeroth Era!" )
2023-07-13 22:55:16 -05:00
( list ( make-leave-flashback-state map ) ) ) )
2023-07-03 22:12:42 -05:00
( defun flashback-casino-dialogue ( map )
( append ( flashback-casino-dialogue-intro )
( flashback-casino-dialogue-bet map )
( flashback-casino-outro map ) ) )
2023-07-10 22:42:22 -05:00
( defun flashback-casino-seat-trigger ( map &optional trigger-plist )
2023-07-03 22:12:42 -05:00
( make-dialogue-state
map
( flashback-casino-dialogue map ) ) )
2023-07-13 22:55:16 -05:00
;;; ———————————————————————————————————
;;; Flashbacks, generally
;;; ———————————————————————————————————
( defparameter *flashbacks*
( list *flashback-casino* *flashback-school* ) )
( defparameter *numerology-excerpts*
' ( ( :en "No. 1: \"VEGETATION: Azalea, Iris, Lilac. [...] Activate today's energetic, independent mood by choosing to wear a red tie or dress. Above all, do not be lazy.\""
:eo "N-ro 1: \"KRESKAĴOJ: Azaleo, irido, siringo. [...] Konduku vian tagon al energia kaj sendependa etoso per ruĝa kravato aŭ robo. Plej grave, ne maldiligentu.\"" ) ) )
( defun make-flashback-function ( flashback )
( 🎭 :make-intermission-function
( getf flashback :title )
( getf flashback :subtitle )
( getf flashback :side-text )
( list :drop 1
:function
( 🌍 :make-overworld-function ( getf flashback :map ) ) ) ) )
( defun make-leave-flashback-state ( map )
( list
:parameters ' ( )
:function
( 🎭 :make-intermission-function
' ( :eo "NUNA TEMPO, NUNA DATO" :en "PRESENT DAY, PRESENT TIME" )
' ( :eo "La 3a de junio, 2006" :en "June 3rd, 2006" )
( car *numerology-excerpts* )
( list :drop 1 :parameters ( list :map ( merge-maps map *outdoors-map* ) ) ) ) ) )
2023-06-24 17:01:12 -05:00
;;; ———————————————————————————————————
;;; Main-menu data
;;; ———————————————————————————————————
2023-06-26 21:49:34 -05:00
( defun submenu ( )
2023-07-02 12:40:47 -05:00
` ( ( :en "IDK"
:selection 100 :selected t )
( :en "GO BACK"
2023-07-11 22:14:02 -05:00
:drop 1 ) ) )
2023-06-22 11:31:49 -05:00
2023-06-26 21:49:34 -05:00
( defun main-menu ( )
2023-07-02 12:40:47 -05:00
` ( ( :en "PLAY" :eo "EKLUDI"
:selection 100 :selected t
2023-07-13 22:55:16 -05:00
:function , ( make-flashback-function ( alexandria:random-elt *flashbacks* ) ) )
2023-07-02 18:31:35 -05:00
( :en "SUBMENU" :eo "SUBMENUO" :row 1
2023-07-13 22:55:16 -05:00
:function , ( 📋 :make-menu-function ( submenu ) ) )
2023-07-02 18:31:35 -05:00
( :en "TERURE" :eo "BADLY" :row 1 )
( :en "QUIT" :eo "REZIGNI" :row 2
2023-07-11 22:14:02 -05:00
:drop 1 ) ) )
2023-06-23 11:28:34 -05:00
2023-06-24 17:01:12 -05:00
;;; ———————————————————————————————————
2023-06-28 10:15:23 -05:00
;;; Invocation station! Choo-choo! 🚆
2023-06-24 17:01:12 -05:00
;;; ———————————————————————————————————
2023-06-04 02:33:53 -05:00
( defun main ( )
2023-06-23 13:29:09 -05:00
" A pathetic fascimile of a main loop. What does it do? WHAST DOES TI DODOO?
What a mysteryyy! You ’ ll have to check out the engine to uncover it.
engine.lisp, that is. Cheers! :D "
2023-07-13 22:55:16 -05:00
( ⚙ :main ( list ( 📋 :make-menu-function ( main-menu ) ) ) ) )
2023-06-08 21:57:51 -05:00
2023-06-04 02:33:53 -05:00
2023-06-09 16:14:46 -05:00
;; — Who’ s there?
;; — Yo momma!
;; — “Yo momma” who?
2023-06-17 22:47:30 -05:00
;; — Yo momma’ s a sweet lady, and I’ d like to take her out for some tea!