The open source repository for the action RPG game in development by Sig Productions titled 'Adventures in Lestoria'!
https://forums.lestoria.net
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
354 lines
15 KiB
354 lines
15 KiB
1 year ago
|
#pragma region License
|
||
|
/*
|
||
|
License (OLC-3)
|
||
|
~~~~~~~~~~~~~~~
|
||
|
|
||
1 year ago
|
Copyright 2024 Joshua Sigona <sigonasr2@gmail.com>
|
||
1 year ago
|
|
||
|
Redistribution and use in source and binary forms, with or without modification,
|
||
|
are permitted provided that the following conditions are met:
|
||
|
|
||
|
1. Redistributions or derivations of source code must retain the above copyright
|
||
|
notice, this list of conditions and the following disclaimer.
|
||
|
|
||
|
2. Redistributions or derivative works in binary form must reproduce the above
|
||
|
copyright notice. This list of conditions and the following disclaimer must be
|
||
|
reproduced in the documentation and/or other materials provided with the distribution.
|
||
|
|
||
|
3. Neither the name of the copyright holder nor the names of its contributors may
|
||
|
be used to endorse or promote products derived from this software without specific
|
||
|
prior written permission.
|
||
|
|
||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
|
||
|
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
||
|
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
|
||
|
SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||
|
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
|
||
|
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
||
|
BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||
|
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
||
|
ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||
|
SUCH DAMAGE.
|
||
1 year ago
|
|
||
|
Portions of this software are copyright © 2023 The FreeType
|
||
|
Project (www.freetype.org). Please see LICENSE_FT.txt for more information.
|
||
|
All rights reserved.
|
||
1 year ago
|
*/
|
||
|
#pragma endregion
|
||
|
#include "VisualNovel.h"
|
||
|
#include "GameState.h"
|
||
1 year ago
|
#include "AdventuresInLestoria.h"
|
||
1 year ago
|
#include <fstream>
|
||
1 year ago
|
#include "DEFINES.h"
|
||
|
#include "Unlock.h"
|
||
|
#include "Menu.h"
|
||
1 year ago
|
#include "util.h"
|
||
1 year ago
|
|
||
|
INCLUDE_game
|
||
|
INCLUDE_GFX
|
||
1 year ago
|
|
||
|
VisualNovel VisualNovel::novel;
|
||
|
safemap<std::string,std::vector<std::unique_ptr<Command>>>VisualNovel::storyLevelData;
|
||
1 year ago
|
std::set<std::string>VisualNovel::graphicsToLoad;
|
||
1 year ago
|
Font VisualNovel::font,VisualNovel::narratorFont,VisualNovel::locationFont;
|
||
1 year ago
|
|
||
|
void VisualNovel::Initialize(){
|
||
1 year ago
|
font=Font("GFX_Prefix"_S+"dialog_font_size"_s[0]+".ttf","dialog_font_size"_i[1]);
|
||
1 year ago
|
narratorFont=Font("GFX_Prefix"_S+"narrator_font_size"_s[0]+".ttf","narrator_font_size"_i[1]);
|
||
|
locationFont=Font("GFX_Prefix"_S+"location_font_size"_s[0]+".ttf","location_font_size"_i[1]);
|
||
1 year ago
|
for(int chapter=1;chapter<=6;chapter++){
|
||
1 year ago
|
std::string chapterFilename="assets/"+"story_directory"_S+"Chapter "+std::to_string(chapter)+".txt";
|
||
1 year ago
|
std::ifstream file(chapterFilename);
|
||
|
if(!file)ERR("Failed to open file "<<chapterFilename)
|
||
|
std::string line;
|
||
|
std::string currentStory;
|
||
|
while(file.good()){
|
||
|
|
||
|
auto trim=[](std::string line){
|
||
|
for(int counter=0;counter<line.length();counter++){
|
||
|
if(line[counter]!=' '&&line[counter]!='\t')return line.substr(counter);
|
||
|
}
|
||
|
return line;
|
||
|
};
|
||
|
auto ReadCSVArgs=[](std::string text){
|
||
|
std::vector<std::string>args;
|
||
|
size_t counter=0;
|
||
|
while(text.find(',',counter)!=std::string::npos){
|
||
|
std::string arg=text.substr(counter,text.find(',',counter)-counter);
|
||
|
counter=text.find(',',counter)+1;
|
||
|
if(arg.find(',')!=std::string::npos)ERR("Found an erraneous comma inside parsed arg "<<arg<<". Inside of main text: "<<text);
|
||
|
args.push_back(arg);
|
||
|
}
|
||
|
std::string arg=text.substr(counter);
|
||
|
args.push_back(arg);
|
||
|
return args;
|
||
|
};
|
||
|
std::getline(file,line);
|
||
|
line=trim(line);
|
||
|
if(line.length()==0)continue; //It's a blank line, so we skip it.
|
||
|
|
||
|
switch(line[0]){
|
||
|
case '=':{ //Initializes a new story section.
|
||
|
currentStory=line.substr(3,line.find('=',3)-3);
|
||
|
if(currentStory.find('=')!=std::string::npos)ERR("Story name "<<currentStory<<" is an invalid name! Failed to load story.");
|
||
|
storyLevelData[currentStory];
|
||
|
}break;
|
||
|
case '{':{ //Start of a command.
|
||
1 year ago
|
auto&data=storyLevelData.at(currentStory);
|
||
|
|
||
1 year ago
|
auto AddImagesForLoading=[](std::vector<std::string>&arguments){
|
||
|
for(std::string&arg:arguments){
|
||
|
if(arg=="story_player_name"_S){
|
||
|
graphicsToLoad.insert("character_image_location"_S+"Player_F.png");
|
||
|
graphicsToLoad.insert("character_image_location"_S+"Player_M.png");
|
||
|
}else{
|
||
|
graphicsToLoad.insert("character_image_location"_S+arg+".png");
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
1 year ago
|
size_t spacePos=line.find(' ');
|
||
1 year ago
|
std::vector<std::string>arguments;
|
||
|
|
||
|
if(spacePos!=std::string::npos){//There are arguments to parse...
|
||
|
size_t endingBracePos=line.find('}',spacePos+1);
|
||
|
if(endingBracePos==std::string::npos)ERR("Cannot parse arguments from "<<line<<": No closing ending brace found")
|
||
|
std::string args=line.substr(spacePos+1,endingBracePos-(spacePos+1));
|
||
|
if(line.size()==0||line[0]==' '||args.find('}')!=std::string::npos)ERR("Cannot parse args, found invalid tokens in "<<args)
|
||
|
arguments=ReadCSVArgs(args);
|
||
|
}
|
||
1 year ago
|
|
||
|
if(line.find("{LOCATION")!=std::string::npos){//Location command
|
||
|
if(arguments.size()!=1)ERR("Arguments size is "<<arguments.size()<<". Expecting only 1 argument.")
|
||
|
data.push_back(std::make_unique<LocationCommand>(arguments[0]));
|
||
|
}else
|
||
|
if(line.find("{BACKGROUND")!=std::string::npos){//Background command
|
||
|
if(arguments.size()!=1)ERR("Arguments size is "<<arguments.size()<<". Expecting only 1 argument.")
|
||
1 year ago
|
graphicsToLoad.insert("story_background_image_location"_S+arguments[0]);
|
||
1 year ago
|
data.push_back(std::make_unique<BackgroundCommand>(arguments[0]));
|
||
|
}else
|
||
|
if(line.find("{LEFT")!=std::string::npos){//Left command
|
||
1 year ago
|
AddImagesForLoading(arguments);
|
||
1 year ago
|
data.push_back(std::make_unique<LeftCommand>(arguments));
|
||
|
}else
|
||
|
if(line.find("{RIGHT")!=std::string::npos){//Right command
|
||
1 year ago
|
AddImagesForLoading(arguments);
|
||
1 year ago
|
data.push_back(std::make_unique<RightCommand>(arguments));
|
||
|
}else
|
||
|
if(line.find("{PAUSE")!=std::string::npos){//Pause command
|
||
|
if(arguments.size()!=0)ERR("Arguments size is "<<arguments.size()<<". Expecting no arguments.")
|
||
|
data.push_back(std::make_unique<PauseCommand>());
|
||
|
}else{
|
||
|
ERR("Unknown command "<<line<<". Could not parse!");
|
||
|
}
|
||
|
}break;
|
||
|
case '[':{
|
||
1 year ago
|
auto&data=storyLevelData.at(currentStory);
|
||
|
|
||
1 year ago
|
size_t endingBracePos=line.find(']');
|
||
|
if(endingBracePos==std::string::npos)ERR("Cannot parse arguments from "<<line<<": No closing ending bracket found")
|
||
|
std::string args=line.substr(1,endingBracePos-1);
|
||
|
if(args.find(']')!=std::string::npos)ERR("Cannot parse args, found invalid tokens in "<<args)
|
||
|
std::vector<std::string>arguments=ReadCSVArgs(args);
|
||
|
if(arguments.size()>2)ERR("Expecting a maximum of two arguments for parsed args in "<<args<<", got "<<arguments.size()<<" arguments instead.");
|
||
|
if(arguments.size()!=2){
|
||
|
data.push_back(std::make_unique<SpeakerCommand>(arguments[0]));
|
||
|
}else{
|
||
|
data.push_back(std::make_unique<SpeakerCommand>(arguments[0],arguments[1]));
|
||
|
}
|
||
|
}break;
|
||
|
default:{
|
||
1 year ago
|
auto&data=storyLevelData.at(currentStory);
|
||
|
|
||
1 year ago
|
data.push_back(std::make_unique<DialogCommand>(line));
|
||
|
}break;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
storyLevelData.SetInitialized();
|
||
|
};
|
||
|
void VisualNovel::LoadVisualNovel(std::string storyLevelName){
|
||
|
novel.storyLevel=storyLevelName;
|
||
1 year ago
|
novel.activeText=U"";
|
||
1 year ago
|
novel.leftCharacters.clear();
|
||
|
novel.rightCharacters.clear();
|
||
1 year ago
|
novel.backgroundFilename=novel.prevBackgroundFilename="";
|
||
1 year ago
|
novel.commands.clear();
|
||
1 year ago
|
novel.commandIndex=0;
|
||
1 year ago
|
for(std::unique_ptr<Command>&command:storyLevelData.at(storyLevelName)){
|
||
|
novel.commands.push_back(command.get());
|
||
|
}
|
||
|
GameState::ChangeState(States::STORY,0.5f);
|
||
|
novel.ExecuteNextCommand();
|
||
|
novel.prevTheme=Menu::GetCurrentTheme().GetThemeName();
|
||
|
Menu::themeSelection="Purple";
|
||
1 year ago
|
}
|
||
|
void VisualNovel::Update(){
|
||
1 year ago
|
if(transitionTime==0&&game->KEY_CONFIRM.Pressed()){
|
||
1 year ago
|
activeText=U"";
|
||
1 year ago
|
novel.ExecuteNextCommand();
|
||
|
}
|
||
|
locationDisplayTime=std::max(0.f,locationDisplayTime-game->GetElapsedTime());
|
||
1 year ago
|
transitionTime=std::max(0.f,transitionTime-game->GetElapsedTime());
|
||
1 year ago
|
textScrollTime=std::max(0.f,textScrollTime-game->GetElapsedTime());
|
||
1 year ago
|
}
|
||
|
void VisualNovel::ExecuteNextCommand(){
|
||
|
if(commandIndex<commands.size()){
|
||
|
commandIndex++;
|
||
1 year ago
|
commands[size_t(commandIndex-1)]->Execute(novel);
|
||
1 year ago
|
}else{
|
||
|
Unlock::UnlockCurrentMap();
|
||
|
Menu::themeSelection=novel.prevTheme;
|
||
|
GameState::ChangeState(States::OVERWORLD_MAP,0.5f);
|
||
|
}
|
||
1 year ago
|
}
|
||
|
void VisualNovel::Draw(){
|
||
1 year ago
|
if(backgroundFilename!=""){
|
||
1 year ago
|
float alpha=1;
|
||
|
if(transitionTime>0){
|
||
|
alpha=util::lerp(0,1,1-(transitionTime/maxTransitionTime));
|
||
|
}
|
||
|
if(prevBackgroundFilename!=""){
|
||
1 year ago
|
game->DrawDecal({0,0},GFX["story_background_image_location"_S+prevBackgroundFilename].Decal());
|
||
1 year ago
|
}
|
||
1 year ago
|
game->DrawDecal({0,0},GFX["story_background_image_location"_S+backgroundFilename].Decal(),{1,1},{255,255,255,uint8_t(255*alpha)});
|
||
1 year ago
|
}else{
|
||
|
game->FillRectDecal({0,0},game->GetScreenSize());
|
||
|
}
|
||
1 year ago
|
for(int i=leftCharacters.size()-1;i>=0;i--){
|
||
|
//Start 72 from the bottom.
|
||
1 year ago
|
std::u32string character(leftCharacters[i].begin(),leftCharacters[i].end());
|
||
1 year ago
|
Pixel fadeColor=WHITE;
|
||
|
if(character!=actualSpeakerName)fadeColor={128,128,128,255};
|
||
|
game->DrawDecal(vi2d{0,game->GetScreenSize().y}-vi2d{-i*64,72+168},GFX[GetCharacterImage(character)].Decal(),{1,1},fadeColor);
|
||
|
}
|
||
|
for(int i=rightCharacters.size()-1;i>=0;i--){
|
||
|
//Start 72 from the bottom.
|
||
1 year ago
|
std::u32string character(rightCharacters[i].begin(),rightCharacters[i].end());
|
||
1 year ago
|
Pixel fadeColor=WHITE;
|
||
|
if(character!=actualSpeakerName)fadeColor={128,128,128,255};
|
||
|
float spriteWidth=GFX[GetCharacterImage(character)].Sprite()->width;
|
||
|
game->DrawRotatedDecal(game->GetScreenSize()-vi2d{i*64+int(spriteWidth)/2,72+168},GFX[GetCharacterImage(character)].Decal(),0,vf2d{spriteWidth/2,0.f},{-1,1},fadeColor);
|
||
|
}
|
||
1 year ago
|
if(locationDisplayTime>0){
|
||
1 year ago
|
std::u32string locationStr=std::u32string(locationDisplayText.begin(),locationDisplayText.end());
|
||
|
FontRect textSize=locationFont.GetStringBounds(locationStr);
|
||
|
textSize.offset*=2;
|
||
|
textSize.size*=2;
|
||
|
game->FillRectDecal(game->GetScreenSize()/2-textSize.size/2-vi2d{4,4}+textSize.offset/2,textSize.size+vi2d{8,8},BLACK);
|
||
|
game->DrawRectDecal(game->GetScreenSize()/2-textSize.size/2-vi2d{4,4}+textSize.offset/2,textSize.size+vi2d{8,8},WHITE);
|
||
|
game->DrawShadowStringDecal(locationFont,game->GetScreenSize()/2-textSize.size/2+textSize.offset/2,locationStr,WHITE,VERY_DARK_BLUE,{2.f,2.f});
|
||
1 year ago
|
}
|
||
|
if(activeText.length()>0){
|
||
1 year ago
|
vf2d nameDisplayPos={24.f,game->GetScreenSize().y-60.f};
|
||
|
vf2d nameDisplayWindowSize={48.f,-12.f};
|
||
1 year ago
|
if(speakerDisplayName.length()>0){
|
||
|
Menu::DrawThemedWindow(nameDisplayPos,nameDisplayWindowSize);
|
||
|
}
|
||
1 year ago
|
vf2d dialogDisplayPos={24.f,game->GetScreenSize().y-48.f};
|
||
1 year ago
|
vf2d dialogDisplaySize={game->GetScreenSize().x-48.f,20.f};
|
||
|
Menu::DrawThemedWindow(dialogDisplayPos,dialogDisplaySize);
|
||
1 year ago
|
FontRect dialogTextSize=font.GetStringBounds(activeText);
|
||
1 year ago
|
FontRect speakerTextSize=font.GetStringBounds(speakerDisplayName);
|
||
1 year ago
|
if(speakerDisplayName.length()>0){
|
||
|
game->DrawShadowStringDecal(font,nameDisplayPos-vf2d{10,7}+(nameDisplayWindowSize+vf2d{24,0})/2-speakerTextSize.size/2+speakerTextSize.offset/2,speakerDisplayName);
|
||
|
game->DrawShadowStringDecal(font,dialogDisplayPos-vf2d{10,6}+dialogTextSize.offset,activeText);
|
||
|
}else{
|
||
|
game->DrawDropShadowStringDecal(narratorFont,dialogDisplayPos-vf2d{10,6}+dialogTextSize.offset,activeText);
|
||
|
}
|
||
1 year ago
|
float yOffset=util::lerp(dialogDisplaySize.y+12,-8,textScrollTime/maxTextScrollTime);
|
||
|
game->DrawPolygonDecal(
|
||
|
Menu::GetPatchPart(1,1).Decal(),
|
||
|
{dialogDisplayPos-vf2d{12,-yOffset},dialogDisplayPos+vf2d{-12,dialogDisplaySize.y+12},dialogDisplayPos+dialogDisplaySize+vf2d{12,12},dialogDisplayPos+vf2d{dialogDisplaySize.x+12,yOffset}},
|
||
|
{{0,0},{0,1},{1,1},{1,0}},
|
||
|
{{255,255,255,240},{255,255,255,255},{255,255,255,255},{255,255,255,240}});
|
||
1 year ago
|
}
|
||
1 year ago
|
}
|
||
1 year ago
|
std::string VisualNovel::GetCharacterImage(std::u32string name){
|
||
|
if(name==U"You"){ //Assume we are using female player avatar for now!
|
||
1 year ago
|
return "character_image_location"_S+"Player_F.png";
|
||
|
}
|
||
1 year ago
|
return "character_image_location"_S+std::string(name.begin(),name.end())+".png";
|
||
1 year ago
|
}
|
||
1 year ago
|
|
||
1 year ago
|
VisualNovel::VisualNovel(){}
|
||
|
|
||
1 year ago
|
Command::Command(){}
|
||
|
|
||
|
void LocationCommand::Execute(VisualNovel&vn){
|
||
1 year ago
|
vn.locationDisplayTime=5.f;
|
||
|
vn.locationDisplayText=location;
|
||
|
vn.ExecuteNextCommand();
|
||
1 year ago
|
}
|
||
1 year ago
|
LocationCommand::LocationCommand(std::string location)
|
||
1 year ago
|
:location(location){}
|
||
1 year ago
|
CommandType::CommandType LocationCommand::GetType(){return CommandType::LOCATION;}
|
||
1 year ago
|
|
||
|
void BackgroundCommand::Execute(VisualNovel&vn){
|
||
1 year ago
|
vn.prevBackgroundFilename=vn.backgroundFilename;
|
||
1 year ago
|
vn.backgroundFilename=backgroundFilename;
|
||
1 year ago
|
vn.transitionTime=2.0f;
|
||
1 year ago
|
vn.ExecuteNextCommand();
|
||
1 year ago
|
}
|
||
1 year ago
|
BackgroundCommand::BackgroundCommand(std::string backgroundFilename)
|
||
1 year ago
|
:backgroundFilename(backgroundFilename){}
|
||
1 year ago
|
CommandType::CommandType BackgroundCommand::GetType(){return CommandType::BACKGROUND;}
|
||
1 year ago
|
|
||
|
void LeftCommand::Execute(VisualNovel&vn){
|
||
1 year ago
|
vn.leftCharacters=characters;
|
||
|
vn.ExecuteNextCommand();
|
||
1 year ago
|
}
|
||
1 year ago
|
LeftCommand::LeftCommand(std::vector<std::string>characters)
|
||
1 year ago
|
:characters(characters){}
|
||
1 year ago
|
CommandType::CommandType LeftCommand::GetType(){return CommandType::LEFT;}
|
||
1 year ago
|
|
||
|
void RightCommand::Execute(VisualNovel&vn){
|
||
1 year ago
|
vn.rightCharacters=characters;
|
||
|
vn.ExecuteNextCommand();
|
||
1 year ago
|
}
|
||
1 year ago
|
RightCommand::RightCommand(std::vector<std::string>characters)
|
||
1 year ago
|
:characters(characters){}
|
||
1 year ago
|
CommandType::CommandType RightCommand::GetType(){return CommandType::RIGHT;}
|
||
1 year ago
|
|
||
|
void SpeakerCommand::Execute(VisualNovel&vn){
|
||
1 year ago
|
vn.speakerDisplayName.assign(displayedName.begin(),displayedName.end());
|
||
|
vn.actualSpeakerName.assign(actualSpeakerName.begin(),actualSpeakerName.end());
|
||
1 year ago
|
vn.ExecuteNextCommand();
|
||
1 year ago
|
}
|
||
1 year ago
|
SpeakerCommand::SpeakerCommand(std::string speaker)
|
||
1 year ago
|
:displayedName(speaker),actualSpeakerName(speaker){}
|
||
1 year ago
|
SpeakerCommand::SpeakerCommand(std::string displayedName,std::string speaker)
|
||
1 year ago
|
:displayedName(displayedName),actualSpeakerName(speaker){}
|
||
1 year ago
|
CommandType::CommandType SpeakerCommand::GetType(){return CommandType::SPEAKER;}
|
||
1 year ago
|
|
||
|
void DialogCommand::Execute(VisualNovel&vn){
|
||
1 year ago
|
vn.textScrollTime=VisualNovel::maxTextScrollTime;
|
||
1 year ago
|
bool mustDisplay=vn.activeText.length()==0;
|
||
1 year ago
|
Font*displayFont=&VisualNovel::font;
|
||
|
if(vn.actualSpeakerName.length()==0)displayFont=&VisualNovel::narratorFont;
|
||
|
std::u32string newText=util::WrapText(game,vn.activeText+(vn.activeText.length()>0?U" ":U"")+std::u32string(dialog.begin(),dialog.end()),game->GetScreenSize().x-24,*displayFont,{1,1});
|
||
|
if(VisualNovel::font.GetStringBounds(newText).size.y>48){//Hit the maximum of 3 lines.
|
||
1 year ago
|
if(!mustDisplay){
|
||
|
vn.commandIndex--;
|
||
1 year ago
|
}else{
|
||
|
vn.activeText=newText;
|
||
1 year ago
|
}
|
||
|
}else{
|
||
|
vn.activeText=newText;
|
||
|
if(vn.commandIndex<vn.commands.size()){
|
||
|
CommandType::CommandType nextCommandType=vn.commands[vn.commandIndex]->GetType();
|
||
|
if(nextCommandType==CommandType::DIALOG){ //Only add to dialog when the next command type is a dialog as well.
|
||
|
vn.ExecuteNextCommand();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
1 year ago
|
}
|
||
1 year ago
|
DialogCommand::DialogCommand(std::string dialog)
|
||
1 year ago
|
:dialog(dialog){}
|
||
1 year ago
|
CommandType::CommandType DialogCommand::GetType(){return CommandType::DIALOG;}
|
||
1 year ago
|
|
||
1 year ago
|
void PauseCommand::Execute(VisualNovel&vn){}
|
||
1 year ago
|
PauseCommand::PauseCommand(){}
|
||
|
CommandType::CommandType PauseCommand::GetType(){return CommandType::PAUSE;}
|