Skip to content

Latest commit

 

History

History
42 lines (25 loc) · 9.4 KB

README.md

File metadata and controls

42 lines (25 loc) · 9.4 KB

ProjectVisualNovel (Fall 2016 - 2018)

Overview

The goal of this project is to create a branching dialogue system that can be used by game developers to add dialogue to LibGDX games. The original intent of the project was to be used to create visual novels, so the feature set is created with that in mind. However, the system should be flexible enough to be used for any type of dialogue, making it useful for other genres like RPGs or even just for cutscenes in games.

Visual novels are a type of video game that feature little gameplay and instead are made mostly of dialogue, with most interaction being in the form of making choices that influence the story in some way. Visual novels are less common for American gamers, but are extremely popular in Japan. This project was inspired by visual novels such as Ace Attorney and Dangan Ronpa and some of the ideas used in structuring the conversation files were based on Ren'PY, a Python engine for making visual novels.

I chose to make this project instead of using something like Ren'Py for several reasons. Writing the code myself allows me to have much more control and allows me to implement commands however I feel is most convenient. Commands that would be inconvenient or impossible in Ren'Py can be implemented more easily. Additionally, by making the system as flexible as possible, it can eaily be used for much more than visual novels. Ren'Py is great for visual novels, but doesn't necessarily have functionality for other types of games. And perhaps most importantly, I thought it would be good experience to try doing something like this myself. There's little point in reinventing the wheel, but learning to make a better wheel is still useful.

Demonstrations

This project was used by team members and I to make a game during a Hackathon! Some new features were added just for the game, but the flexibility of this system made it easy and fast to do!

Note: Graphics are placeholder and not mine. They are just for demonstration purposes. gif

The file containing the Conversation used in the example can be seen here

How it works

Almost all of the code used for the visual novel system can be found in the GUI package. Many of the terms below are simply terms I created to refer to things in the dialogue system. Other visual novel systems may refer to them by other names.

The smallest component of dialogue is a command. Each command makes some change to current status of the game. These changes can be things such as playing an animation, displaying a sequence of text, or offering the player a choice and reacting to it. Most commands can even be embedded in messages to be automatically executed once the text scrolls to that point! There are a wide variety of commands already implemented and more are being added still.

A named sequence of commands is a branch. The game will go through all commands in the current branch and execute them in order, sometimes stopping between commands to let the previous one finish, depending on what command it was. For example, playing a sound effect does not pause the execution of commands, but displaying a message does pause the execution of commands until the players clicks to confirm that they want to continue. Writing the code in a way that allowed commands to determine if they should pause execution has already proven to be extremely helpful. I can even implement commands that sometimes pause and sometimes don't, such as the set animation command which, depending on a variable, will either go straight to the next command or wait for the animation to finish.

A group of branches is called a Conversation. Branches within a Conversation are freely traveled between by using the change branch command. An example where this would be useful is in the show choice command, which presents the player with several choices. Depending on which choice they make, a different command can be executed. Executing a change branch command lets the player go through a new series of dialogue depending on their choice. The player can even be returned to a branch they've already been through for looping dialogue.

Writing Conversations using XML

Because of the sheer number of commands that would be required to make a full visual novel, it has always been a priority to make it convenient and fast to write Conversations. After some contemplation, I settled on using XML files to write Conversations. This allows me to add tags that can have attributes and other elements within them. These tags are then converted to the corresponding commands in game by the XML Loader.

I have taken several steps to make writing XML conversations as fast, clean, and error free as possible. I have created an XSD schema for the conversation files. This adds simple validation, making sure the creator is immediately aware of missing attributes or misplaced elements. Additionally, in many IDEs, including IntelliJ, the conversation schema adds autocompletion to many of the tags. For example, mandatory attributes can be automatically added for the creator to fill in.

I have also simplified Conversation files by allowing some elements to use text instead of XML tags. For example, dialogue can be added by writing "Speaker: Text to say." instead of . This slightly reduces the autocompletion capabilities and validation, but it makes it much easier to read and write. I intend to write a more extensive validator in Java that would catch additional problems.

Scripting support

I have also added support for scripting languages to allow creators to make more dynamic content. Currently, Lua and Groovy are both supported, although only Lua works on Android. Creators can add script commands to Conversation files that execute a line of code or load and execute a file containing code. The primary use of scripting languages is to let the user modify variables at different points. Then, a command can be used to check if a condition is met and execute commands only if it is. Here's a hypothetical example. The creator makes a Lua variable called "money". Throughout the game, different characters may ask them questions, and depending on how they answer, the player receives money, with something like "money = money + 10". The player could even gain random amounts of money by using a random function! Later in the game, there could be an item that the player can only buy with a certain amount of money. By using commands, the Conversation could check if the player has enough money ("money >= 50"), subtract it if they do ("money = money - 50") and give the player an item ("hasitem = true"), in addition to showing different dialogue to the player!

Saving and Loading

The system has been set up to use the Kryo framework for saving and loading progress. Serializers have been written for the critical game components. When the player saves, the text, characters, and commands in progress are written to a file to be loaded later. All variables defined in the scripting languages are also saved!

Resource Management

Changes are in progress to make resource management as easy as possible! The creator simply has to edit the Resources.xml file, adding their own files to each category. They can give each resource an identifier, the name through which it is referred when writing Conversations. The identifier allows the creator to change swap resources with new ones without changing Conversation files, as long as they keep the identifier the same! Additionally, the new Resource Schema Generator can be used to automatically generate an XSD schema containing all resources defined in Resources.xml. By creating a schema, whenever the creator references a resource in their Conversation files they can use autocompletion and validation! IntelliJ's "file watcher" plugin even allows the schema generator jar file to be run automatically every time Resources.xml changes, so the user doesn't have to do anything!