140 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			140 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """This module defines the basic interpreter system."""
 | |
| 
 | |
| import fnmatch
 | |
| 
 | |
| from typing import Optional, List, Set, Tuple, cast
 | |
| 
 | |
| from .things import Thing, Container, Place, Player
 | |
| from .base import VerbCallFrame, PyooVerbNotFound, PyooObjectNotFound
 | |
| 
 | |
| 
 | |
| class Interpreter:
 | |
|     """Base interpreter class.
 | |
| 
 | |
|     Manages a collection of objects, and invoking verbs on them.
 | |
|     """
 | |
|     def __init__(self, contents: Optional[List[Thing]] = None):
 | |
|         """Initialize an interpreter.
 | |
| 
 | |
|         Arguments:
 | |
|             contents (optional list of things): The initial contents to populate the interpreter with.
 | |
| 
 | |
|         """
 | |
|         if contents is None:
 | |
|             contents = []
 | |
|         self.contents: Set[Thing] = set(contents)
 | |
|         for cont in self.contents:
 | |
|             cont.interpreter = self
 | |
|         self.content_cache: List[Tuple[str, Thing]] = []
 | |
|         self.update()
 | |
| 
 | |
|     def add_player(self, player_object: Player) -> None:
 | |
|         """Add a player to the interpreter."""
 | |
|         self.contents.add(player_object)
 | |
|         player_object.interpreter = self
 | |
|         self.update()
 | |
| 
 | |
|     def remove_player(self, player_object: Player) -> None:
 | |
|         """Remove a player from the interpreter."""
 | |
|         self.contents.remove(player_object)
 | |
|         self.update()
 | |
| 
 | |
|     def update(self) -> None:
 | |
|         """Do any updates required when objects are added or removed."""
 | |
|         self.update_caches()
 | |
| 
 | |
|     def update_caches(self) -> None:
 | |
|         """Update the caches for objects contained in the interpreter, as well as updating all of the caches
 | |
|         on contained containers.
 | |
|         """
 | |
|         self.content_cache = []
 | |
|         for obj in self.contents:
 | |
|             for name in obj.names:
 | |
|                 self.content_cache.append((name, obj))
 | |
|             try:
 | |
|                 cast(Container, obj).update_caches()
 | |
|             except AttributeError:
 | |
|                 pass
 | |
| 
 | |
|     def handle_move(self, newroom: Place, player: Player) -> None:
 | |
|         """Move a player to a new contained room."""
 | |
|         if player.location:
 | |
|             player.location.handle_exit(player)
 | |
|         newroom.handle_enter(player)
 | |
| 
 | |
|     def lookup_global_object(self, objstr: str) -> List[Tuple[str, Thing]]:
 | |
|         """Find an object within the soup by objstr."""
 | |
|         return [x for x in self.content_cache if fnmatch.fnmatch(objstr, x[0])]
 | |
| 
 | |
|     def lookup_object(self, player: Player, objstr: str) -> Optional[Thing]:
 | |
|         """Find an object relative to a player
 | |
| 
 | |
|         This first checks the player's inventory, and then the container which contains the
 | |
|         player.
 | |
|         """
 | |
|         m: Optional[List[Tuple[str, Thing]]] = None
 | |
|         try:
 | |
|             m = player.get_name_matches(objstr)
 | |
|         except PyooObjectNotFound:
 | |
|             m = None
 | |
|         if not m and player.location:
 | |
|             m = player.location.get_name_matches(objstr)
 | |
|         if m:
 | |
|             return m[0][1]
 | |
|         else:
 | |
|             return None
 | |
| 
 | |
|     def interpret(self, command: str, player: Player) -> None:
 | |
|         """Interpret a player's command by splitting it into components and finding matching verbs."""
 | |
|         # FIXME make this better to support mulitple word objstrs and prepstr
 | |
|         if not command:
 | |
|             return
 | |
| 
 | |
|         cmd_comps = command.split()
 | |
|         verbstr = cmd_comps[0]
 | |
|         dobjstr = ""
 | |
|         prepstr = ""
 | |
|         iobjstr = ""
 | |
|         argstr = ""
 | |
| 
 | |
|         try:
 | |
|             argstr = " ".join(cmd_comps[1:])
 | |
|         except IndexError:
 | |
|             pass
 | |
| 
 | |
|         try:
 | |
|             dobjstr = cmd_comps[1]
 | |
|             prepstr = cmd_comps[2]
 | |
|             iobjstr = cmd_comps[3]
 | |
|         except IndexError:
 | |
|             pass
 | |
| 
 | |
|         try:
 | |
|             cmdmatches = player.get_command_matches(command)
 | |
|         except PyooVerbNotFound:
 | |
|             if player.location:
 | |
|                 cmdmatches = player.location.get_command_matches(command)
 | |
|             else:
 | |
|                 raise PyooVerbNotFound
 | |
|         cmd = cmdmatches[0]
 | |
|         # glob = cmd[0]
 | |
|         # comps = cmd[1]
 | |
|         verb = cmd[2]
 | |
|         this = cmd[3]
 | |
| 
 | |
|         dobj = None
 | |
|         if verb.callspec[0] == "this":
 | |
|             dobj = this
 | |
|         elif verb.callspec[0] == "that":
 | |
|             # lookup object
 | |
|             dobj = self.lookup_object(player, dobjstr)
 | |
| 
 | |
|         iobj = None
 | |
|         if verb.callspec[2] == "this":
 | |
|             iobj = this
 | |
|         elif verb.callspec[2] == "that":
 | |
|             # lookp object
 | |
|             iobj = self.lookup_object(player, iobjstr)
 | |
| 
 | |
|         return verb(VerbCallFrame(self, player, verbstr, dobj, dobjstr, prepstr, iobj, iobjstr, argstr))
 |