stk-code_catmod/src/states_screens/credits.cpp
2015-04-16 15:28:49 +10:00

434 lines
13 KiB
C++

// SuperTuxKart - a fun racing game with go-kart
// Copyright (C) 2009-2015 Marianne Gagnon
//
// 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, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#include "states_screens/credits.hpp"
#include <fstream>
#include <algorithm>
#include "irrString.h"
using irr::core::stringw;
using irr::core::stringc;
#include "guiengine/engine.hpp"
#include "guiengine/scalable_font.hpp"
#include "guiengine/screen.hpp"
#include "guiengine/widget.hpp"
#include "io/file_manager.hpp"
#include "states_screens/state_manager.hpp"
#include "utils/string_utils.hpp"
#include "utils/translation.hpp"
DEFINE_SCREEN_SINGLETON( CreditsScreen );
using namespace GUIEngine;
const float TIME_SECTION_FADE = 0.8f;
const float ENTRIES_FADE_TIME = 0.3f;
// ----------------------------------------------------------------------------
class CreditsEntry
{
public:
stringw m_name;
std::vector<stringw> m_subentries;
CreditsEntry(stringw& name)
{
m_name = name;
}
}; // CreditsEntry
// ----------------------------------------------------------------------------
class CreditsSection
{
public:
// read-only
std::vector<CreditsEntry> m_entries;
stringw m_name;
CreditsSection(stringw name) { m_name = name; }
// ------------------------------------------------------------------------
void addEntry(CreditsEntry& entry) {m_entries.push_back(entry); }
// ------------------------------------------------------------------------
void addSubEntry(stringw& subEntryString)
{
m_entries[m_entries.size()-1].m_subentries.push_back(subEntryString);
}
}; // CreditdsSection
// ----------------------------------------------------------------------------
CreditsSection* CreditsScreen::getCurrentSection()
{
return m_sections.get(m_sections.size()-1);
} // getCurrentSection
// ----------------------------------------------------------------------------
bool CreditsScreen::getWideLine(std::ifstream& file, core::stringw* out)
{
if (!file.good())
{
Log::error("CreditsScreen", "getWideLine: File is not good!");
return false;
}
wchar_t wide_char;
bool found_eol = false;
stringw line;
char buff[2];
while (true)
{
file.read( buff, 2 );
if (file.good())
{
// We got no complaints so I assume the endianness code here is OK
wide_char = unsigned(buff[0] & 0xFF)
| (unsigned(buff[1] & 0xFF) << 8);
line += wide_char;
if (wide_char == L'\n')
{
found_eol = true;
break;
}
}
else
{
break;
}
}
if (!found_eol) return false;
*out = line;
return true;
} // getWideLine
// ----------------------------------------------------------------------------
#if 0
#pragma mark -
#pragma mark CreditsScreen
#endif
CreditsScreen::CreditsScreen() : Screen("credits.stkgui")
{
m_is_victory_music = false;
} // CreditsScreen
// ----------------------------------------------------------------------------
void CreditsScreen::loadedFromFile()
{
reset();
m_throttle_FPS = false;
std::string creditsfile = file_manager->getAsset("CREDITS");
std::ifstream file( creditsfile.c_str() ) ;
if (file.fail() || !file.is_open() || file.eof())
{
Log::error("CreditsScreen", "Failed to open file at '%s'.",
creditsfile.c_str());
return;
}
stringw line;
// skip Unicode header
file.get();
file.get();
if (file.fail() || !file.is_open() || file.eof())
{
Log::error("CreditsScreen", "Failed to read file at '%s', unexpected EOF.",
creditsfile.c_str());
return;
}
int lineCount = 0;
#undef DEBUG_TRANSLATIONS // Enable to only see the translator credits
#ifdef DEBUG_TRANSLATIONS
int my_counter = 0;
#endif
// let's assume the file is encoded as UTF-16
while (getWideLine( file, &line ))
{
#ifdef DEBUG_TRANSLATIONS
if (my_counter > 0)
break;
my_counter++;
#endif
stringc cversion = line.c_str();
//printf("CREDITS line : %s\n", cversion.c_str());
line = line.trim();
if (line.size() < 1) continue; // empty line
lineCount++;
if ((line[0] & 0xFF) == '=' && (line[line.size()-1] & 0xFF) == '=')
{
line = stringw( line.subString(1, line.size()-2).c_str() );
line = line.trim();
m_sections.push_back( new CreditsSection(line) );
}
else if ((line[0] & 0xFF) == '-')
{
line = stringw( line.subString(1, line.size()-1).c_str() );
getCurrentSection()->addSubEntry( line );
}
else
{
CreditsEntry entry(line);
getCurrentSection()->addEntry( entry );
}
} // end while
if (lineCount == 0)
{
Log::error("CreditsScreen", "Could not read anything from CREDITS file!");
return;
}
irr::core::stringw translators_credits = _("translator-credits");
const int MAX_PER_SCREEN = 6;
if (translators_credits != L"translator-credits")
{
std::vector<irr::core::stringw> translator =
StringUtils::split(translators_credits, '\n');
m_sections.push_back( new CreditsSection("Translations"));
for(unsigned int i = 1; i < translator.size(); i = i + MAX_PER_SCREEN)
{
line = stringw(translations->getCurrentLanguageName().c_str());
CreditsEntry entry(line);
getCurrentSection()->addEntry( entry );
for(unsigned int j = 0; i + j < translator.size() && j < MAX_PER_SCREEN; j ++)
{
getCurrentSection()->addSubEntry(translator[i + j]);
}
}
assert(m_sections.size() > 0);
// translations should be just before the last screen
m_sections.swap( m_sections.size() - 1, m_sections.size() - 2 );
}
} // loadedFromFile
// ----------------------------------------------------------------------------
void CreditsScreen::init()
{
Screen::init();
Widget* w = getWidget<Widget>("animated_area");
assert(w != NULL);
reset();
setArea(w->m_x + 15, w->m_y + 8, w->m_w - 30, w->m_h - 16);
} // init
// ----------------------------------------------------------------------------
void CreditsScreen::setArea(const int x, const int y, const int w, const int h)
{
m_x = x;
m_y = y;
m_w = w;
m_h = h;
m_section_rect = core::rect< s32 >( x, y, x + w, y + h/6 );
} // setArea
// ----------------------------------------------------------------------------
void CreditsScreen::reset()
{
m_curr_section = 0;
m_curr_element = -1;
time_before_next_step = TIME_SECTION_FADE;
m_time_element = 2.5f;
} // reset
// ----------------------------------------------------------------------------
void CreditsScreen::onUpdate(float elapsed_time)
{
// multiply by 0.8 to slow it down a bit as a whole
time_before_next_step -= elapsed_time*0.8f;
const bool before_first_elem = (m_curr_element == -1);
const bool after_last_elem =
(m_curr_element >= (int)m_sections[m_curr_section].m_entries.size());
// ---- section name
video::SColor color( 255 /* a */, 0 /* r */, 0 /* g */ , 75 /* b */ );
video::SColor white_color( 255, 255, 255, 255 );
// manage fade-in
if (before_first_elem)
{
// I use 425 instead of 255 so that there is a little pause after
int alpha = 425 - (int)(time_before_next_step/TIME_SECTION_FADE * 425);
if (alpha < 0) alpha = 0;
else if (alpha > 255) alpha = 255;
white_color.setAlpha( alpha );
}
// manage fade-out
else if (after_last_elem)
{
// I use 425 instead of 255 so that there is a little pause after
int alpha =
(int)(time_before_next_step/TIME_SECTION_FADE * 425) - (425-255);
if (alpha < 0) alpha = 0;
else if (alpha > 255) alpha = 255;
white_color.setAlpha( alpha );
}
GUIEngine::getTitleFont()->draw(m_sections[m_curr_section].m_name.c_str(),
m_section_rect, white_color,
true /* center h */, true /* center v */ );
// draw entries
if (!before_first_elem && !after_last_elem)
{
int text_offset = 0;
// fade in
if (time_before_next_step < ENTRIES_FADE_TIME)
{
const float fade_in = time_before_next_step / ENTRIES_FADE_TIME;
int alpha = (int)(fade_in * 255);
if (alpha < 0) alpha = 0;
else if (alpha > 255) alpha = 255;
color.setAlpha( alpha );
text_offset = (int)((1.0f - fade_in) * 100);
}
// fade out
else if (time_before_next_step >= m_time_element - ENTRIES_FADE_TIME)
{
const float fade_out =
(time_before_next_step - (m_time_element - ENTRIES_FADE_TIME))
/ ENTRIES_FADE_TIME;
int alpha = 255 - (int)(fade_out * 255);
if(alpha < 0) alpha = 0;
else if(alpha > 255) alpha = 255;
color.setAlpha( alpha );
text_offset = -(int)(fade_out * 100);
}
GUIEngine::getFont()->draw(
m_sections[m_curr_section].m_entries[m_curr_element]
.m_name.c_str(),
core::recti( m_x + text_offset, m_y + m_h/6,
m_x + m_w + text_offset, m_y + m_h/3 ),
color, false /* center h */, true /* center v */, NULL,
true /* ignore RTL */ );
const int subamount = (int)m_sections[m_curr_section]
.m_entries[m_curr_element].m_subentries.size();
int suby = m_y + m_h/3;
const int inc = subamount == 0 ? m_h/8
: std::min(m_h/8,
(m_h - m_h/3)/(subamount+1));
for(int i=0; i<subamount; i++)
{
GUIEngine::getFont()->draw(m_sections[m_curr_section]
.m_entries[m_curr_element]
.m_subentries[i].c_str(),
core::recti( m_x + 32,
suby + text_offset/(1+1),
m_x + m_w + 32,
suby + m_h/8
+ text_offset/(1+1) ),
color, false/* center h */,
true /* center v */, NULL,
true /* ignore RTL */ );
suby += inc;
}
}
// is it time to move on?
if (time_before_next_step < 0)
{
if (after_last_elem)
{
// switch to next element
m_curr_section++;
m_curr_element = -1;
time_before_next_step = TIME_SECTION_FADE;
if (m_curr_section >= (int)m_sections.size()) reset();
}
else
{
// move on
m_curr_element++;
if (m_curr_element >=
(int)m_sections[m_curr_section].m_entries.size())
{
time_before_next_step = TIME_SECTION_FADE;
}
else
{
const int count =
(int)m_sections[m_curr_section].m_entries[m_curr_element]
.m_subentries.size();
m_time_element = 2.0f + count*0.6f;
time_before_next_step = m_time_element;
}
}
}
} // onUpdate
// ----------------------------------------------------------------------------
void CreditsScreen::eventCallback(GUIEngine::Widget* widget,
const std::string& name, const int playerID)
{
if (name == "back")
{
StateManager::get()->escapePressed();
}
}
// ----------------------------------------------------------------------------