Mercurial > parpg-core
diff src/parpg/dialogue.py @ 0:1fd2201f5c36
Initial commit of parpg-core.
author | M. George Hansen <technopolitica@gmail.com> |
---|---|
date | Sat, 14 May 2011 01:12:35 -0700 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/parpg/dialogue.py Sat May 14 01:12:35 2011 -0700 @@ -0,0 +1,192 @@ +# This file is part of PARPG. +# +# PARPG 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. +# +# PARPG 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 PARPG. If not, see <http://www.gnu.org/licenses/>. +""" +Provides classes used to contain and organize dialogue data for use within +in-game dialogues between the player character and NPCs. +""" +try: + from collections import OrderedDict +except ImportError: + # Python version 2.4-2.6 doesn't have the OrderedDict + from parpg.common.ordereddict import OrderedDict + +class Dialogue(object): + """ + Represents a complete dialogue and acts as a container for the dialogue + data belonging to a particular NPC. + """ + __slots__ = ['npc_name', 'avatar_path', 'default_greeting', + 'greetings', 'sections'] + + def __init__(self, npc_name, avatar_path, default_greeting, greetings=None, + sections=None): + """ + Initialize a new L{Dialogue} instance. + + @param npc_name: name displayed for the NPC in the dialogue. + @type npc_name: basestring + @param avatar_path: path to the image that should be displayed as the + NPC's avatar. + @type avatar_path: basestring + @param default_greeting: section of dialogue that should be + displayed when the dialogue is first initiated and no other start + sections are available. + @type default_greeting: L{DialogueSection} + @param greetings: sections of dialogue defining the conditions + under which each should be displayed when the dialogue is first + initiated. + @type greetings: list of + L{RootDialogueSections<DialogueGreeting>} + @param sections: sections of dialogue that make up this + L{Dialogue} instance. + @type sections: list of L{DialogueSections<DialogueSection>} + """ + self.npc_name = npc_name + self.avatar_path = avatar_path + self.default_greeting = default_greeting + self.greetings = greetings if greetings is not None else [] + self.sections = OrderedDict() + all_sections = [default_greeting] + if (greetings is not None): + all_sections += greetings + if (sections is not None): + all_sections += sections + if (__debug__): + section_ids = [section.id for section in all_sections] + for section in all_sections: + # Sanity check: All DialogueResponses should have next_section_id + # attributes that refer to valid DialogueSections in the Dialogue. + if (__debug__): + for response in section.responses: + assert response.next_section_id in section_ids + \ + ['end', 'back'], ('"{0}" does not refer to a ' + 'DialogueSection in this Dialogue')\ + .format(response.next_section_id) + self.sections[section.id] = section + + def __str__(self): + """Return the string representation of a L{Dialogue} instance.""" + string_representation = 'Dialogue(npc_id={0.npc_name})'.format(self) + return string_representation + + +class DialogueNode(object): + """ + Abstract base class that represents a node or related group of attributes + within a Dialogue. + """ + def __init__(self, text, actions=None): + """ + Initialize a new L{DialogueNode} instance. + + @param text: textual content of the L{DialogueNode}. + @type text: basestring + @param actions: dialogue actions associated with the L{DialogueNode}. + @type actions: list of L{DialogueActions<DialogueAction>} + """ + self.text = text + self.actions = actions or [] + + +class DialogueSection(DialogueNode): + """DialogueNode that represents a distinct section of the dialogue.""" + __slots__ = ['id', 'text', 'responses', 'actions'] + + def __init__(self, id_, text, responses=None, actions=None): + """ + Initialize a new L{DialogueSection} instance. + + @param id_: named used to uniquely identify the L{DialogueSection} + within a L{Dialogue}. + @type id_: basestring + @param text: text displayed as the NPC's part of the L{Dialogue}. + @type text: basestring + @param responses: possible responses that the player can choose from. + @type responses: list of L{DialogueResponses<DialogueResponse>} + @param actions: dialogue actions that should be executed when the + L{DialogueSection} is reached. + @type actions: list of L{DialogueActions<DialogueAction>} + """ + DialogueNode.__init__(self, text=text, actions=actions) + self.id = id_ + if (responses is not None): + self.responses = list(responses) + + +class DialogueGreeting(DialogueSection): + """ + Represents a root section of dialogue in a L{Dialogue} along with the + conditional statement used to determine the whether this section should be + displayed first upon dialogue initiation. + + @ivar id: Name used to uniquely identify the L{DialogueSection} to which + the L{DialogueRootSectionReference} points. + @type id: basestring + @ivar condition: Boolean Python expression used to determine if the + L{DialogueSection} referenced is a valid starting section. + @type condition: basestring + """ + __slots__ = ['id', 'condition', 'text', 'actions', 'responses'] + + def __init__(self, id_, condition, text, responses=None, actions=None): + """ + Initialize a new L{DialogueGreeting} instance. + + @param id_: named used to uniquely identify the L{DialogueSection} + within a L{Dialogue}. + @type id_: basestring + @param condition: Boolean Python expression used to determine if this + root dialogue section should be displayed. + @type condition: basestring + @param text: text displayed as the NPC's part of the L{Dialogue}. + @type text: basestring + @param responses: possible responses that the player can choose from. + @type responses: list of L{DialogueResponses<DialogueResponse>} + @param actions: dialogue actions that should be executed when the + L{DialogueSection} is reached. + @type actions: list of L{DialogueActions<DialogueAction>} + """ + DialogueSection.__init__(self, id_=id_, text=text, responses=responses, + actions=actions) + self.condition = condition + + +class DialogueResponse(DialogueNode): + """ + L{DialogueNode} that represents one possible player response to a + particular L{DialogueSection}. + """ + __slots__ = ['text', 'actions', 'condition', 'next_section_id'] + + def __init__(self, text, next_section_id, actions=None, condition=None): + """ + Initialize a new L{DialogueResponse} instance. + + @param text: text displayed as the content of the player's response. + @type text: basestring + @param next_section_id: ID of the L{DialogueSection} that should be + jumped to if this response is chosen by the player. + @type next_section_id: basestring + @param actions: dialogue actions that should be executed if this + response is chosen by the player. + @type actions: list of L{DialogueActions<DialogueAction>} + @param condition: Python expression that when evaluated determines + whether the L{DialogueResponse} should be displayed to the player + as a valid response. + @type condition: basestring + """ + DialogueNode.__init__(self, text=text, actions=actions) + self.condition = condition + self.next_section_id = next_section_id