/* Copyright (c) 2010 - Wii Banner Player Project Copyright (c) 2012 - Dimok This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software. Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions: 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. 3. This notice may not be removed or altered from any source distribution. */ #include "Layout.h" #include "WiiFont.h" #include "unzip/U8Archive.h" Layout::Layout(u8 *font1, u8 *font2) : header(0) { Wbf1Font = font1; Wbf2Font = font2; } Layout::~Layout() { for(u32 i = 0; i < panes.size(); ++i) delete panes[i]; for(u32 i = 0; i < resources.materials.size(); ++i) delete resources.materials[i]; for(u32 i = 0; i < resources.textures.size(); ++i) delete resources.textures[i]; for(u32 i = 0; i < resources.fonts.size(); ++i) delete resources.fonts[i]; } bool Layout::Load(const u8 *brlyt) { if(!brlyt) return false; const BRLYT_Header *brlytFile = (const BRLYT_Header *) brlyt; if(brlytFile->magic != BRLYT_MAGIC || brlytFile->version != BRLYT_VERSION) return false; Group* last_group = NULL; std::stack*> group_stack; group_stack.push(&groups); Pane* last_pane = NULL; std::stack*> pane_stack; pane_stack.push(&panes); const u8 *position = brlyt + brlytFile->header_len; for(u32 i = 0; i < brlytFile->section_count; ++i) { section_t *section = (section_t *) position; position += section->size; if(section->magic == Layout::MAGIC) { header = (Layout::Header *) (section + 1); } else if (section->magic == TextureList::MAGIC) { const LytItemList *txl1 = (const LytItemList *) (section+1); const char *nameoffset = ((const char *)(txl1+1)); const LytStringTable *stringTable = (const LytStringTable *) (((const u8 *)(txl1+1))+txl1->offset_to_first); for(u32 i = 0; i < txl1->num_items; ++i) { const char *name = nameoffset+stringTable[i].offset_filename; Texture *texture = new Texture; texture->setName(name); resources.textures.push_back(texture); } } else if (section->magic == MaterialList::MAGIC) { const LytItemList *mat1 = (const LytItemList *) (section+1); const u32 *mat_offsets = (const u32 *) (((const u8 *)(mat1+1))+mat1->offset_to_first); for(u32 i = 0; i < mat1->num_items; ++i) { Material *material = new Material; material->Load((Material::Header *) (((const u8 *) section)+mat_offsets[i])); resources.materials.push_back(material); } } else if (section->magic == FontList::MAGIC) { // load font list const LytItemList *fnl1 = (const LytItemList *) (section+1); const char *nameoffset = ((const char *)(fnl1+1)); const LytStringTable *stringTable = (const LytStringTable *) (((const u8 *)(fnl1+1))+fnl1->offset_to_first); for(u32 i = 0; i < fnl1->num_items; i++) { const char *name = nameoffset+stringTable[i].offset_filename; WiiFont *font; if(strcmp(name, "wbf1.brfna") == 0 || strcmp(name, "RevoIpl_RodinNTLGPro_DB_32_I4.brfnt") == 0) { //! 1st system font or alias for it font = new WiiFont; font->SetName("wbf1.brfna"); font->Load(Wbf1Font); if(!font) continue; } else if(strcmp(name, "wbf2.brfna") == 0) { //! 2nd system font font = new WiiFont; font->SetName("wbf2.brfna"); font->Load(Wbf2Font); if(!font) continue; } else { //! font from banner font = new WiiFont; font->SetName(name); } resources.fonts.push_back(font); } } else if (section->magic == Pane::MAGIC) { Pane* pane = new Pane; pane->Load((Pane::Header *) section); pane_stack.top()->push_back(last_pane = pane); } else if (section->magic == Bounding::MAGIC) { Bounding* pane = new Bounding; pane->Load((Pane::Header *) section); pane_stack.top()->push_back(last_pane = pane); } else if (section->magic == Picture::MAGIC) { Picture* pane = new Picture; pane->Load((Pane::Header *) section); pane_stack.top()->push_back(last_pane = pane); } else if (section->magic == Window::MAGIC) { Window* pane = new Window; pane->Load((Pane::Header *) section); pane_stack.top()->push_back(last_pane = pane); } else if (section->magic == Textbox::MAGIC) { Textbox* pane = new Textbox; pane->Load((Pane::Header *) section); pane_stack.top()->push_back(last_pane = pane); } else if (section->magic == Layout::MAGIC_PANE_PUSH) { if (last_pane) pane_stack.push(&last_pane->panes); } else if (section->magic == Layout::MAGIC_PANE_POP) { if (pane_stack.size() > 1) pane_stack.pop(); } else if (section->magic == Group::MAGIC) { const char *grp = (const char *) (section + 1); std::string group_name(grp, 0, Layout::Group::NAME_LENGTH); Group& group_ref = (*group_stack.top())[group_name]; grp += Layout::Group::NAME_LENGTH; u16 sub_count = *(u16 *) grp; grp += 4; // 2 bytes reserved while (sub_count--) { std::string pane_name(grp, 0, Layout::Group::NAME_LENGTH); group_ref.panes.push_back(pane_name); grp += Layout::Group::NAME_LENGTH; } last_group = &group_ref; } else if (section->magic == Layout::MAGIC_GROUP_PUSH) { if (last_group) group_stack.push(&last_group->groups); } else if (section->magic == Layout::MAGIC_GROUP_POP) { if (group_stack.size() > 1) group_stack.pop(); } else { gprintf("Uknown layout section: %08X\n", section->magic); } } return true; } bool Layout::LoadTextures(const u8 *banner_file) { bool success = false; for(u32 i = 0; i < resources.textures.size(); ++i) { u32 filesize; const u8 *file = u8_get_file(banner_file, resources.textures[i]->getName().c_str(), &filesize); if (file) resources.textures[i]->Load(file); else success = false; } return success; } bool Layout::LoadFonts(const u8 *banner_file) { bool success = false; for(u32 i = 0; i < resources.fonts.size(); ++i) { if(resources.fonts[i]->IsLoaded()) continue; u32 filesize; const u8 *file = u8_get_file(banner_file, resources.fonts[i]->getName().c_str(), &filesize); if (file) resources.fonts[i]->Load(file); else success = false; } return success; } void Layout::Render(Mtx &modelview, const Vec2f &ScreenProps, bool widescreen, u8 render_alpha) const { if(!header) return; Mtx mv; // we draw inverse guMtxScaleApply(modelview, mv, 1.0f, -1.0f, 1.0f); // centered draw if(header->centered) guMtxTransApply(mv, mv, ScreenProps.x * 0.5f, ScreenProps.y * 0.5f, 0.f); // render all panes for(u32 i = 0; i < panes.size(); ++i) panes[i]->Render(resources, render_alpha, mv, widescreen); } void Layout::SetFrame(FrameNumber frame_number) { frame_current = frame_number; const u8 key_set = (frame_current >= frame_loop_start); if (key_set) frame_number -= frame_loop_start; resources.cur_set = key_set; for(u32 i = 0; i < panes.size(); ++i) panes[i]->SetFrame(frame_number, key_set); for(u32 i = 0; i < resources.materials.size(); ++i) resources.materials[i]->SetFrame(frame_number, key_set); } void Layout::AdvanceFrame() { ++frame_current; if (frame_current >= frame_loop_end) frame_current = frame_loop_start; SetFrame(frame_current); } void Layout::SetLanguage(const std::string& language) { if(!header) return; // check if that language is found bool lang_found = false; // hide panes of non-matching languages std::map::iterator itr; for(itr = groups["RootGroup"].groups.begin(); itr != groups["RootGroup"].groups.end(); itr++) { // check if that language exists if(!lang_found && itr->first == language) lang_found = true; // some hax, there are some odd "Rso0" "Rso1" groups that shouldn't be hidden // only the 3 character language groups should be if (itr->first != language && itr->first.length() == 3) { std::list::iterator itr2; for(itr2 = itr->second.panes.begin(); itr2 != itr->second.panes.end(); itr2++) { Pane* const found = FindPane(*itr2); if (found) found->SetHide(true); } } } // show english if langauge is not found if(!lang_found && language != "ENG") { SetLanguage("ENG"); return; } // unhide panes of matching language, some banners list language specific panes in multiple language groups std::list::iterator itr2; for(itr2 = groups["RootGroup"].groups[language].panes.begin(); itr2 != groups["RootGroup"].groups[language].panes.end(); itr2++) { Pane* const found = FindPane(*itr2); if (found) { found->SetHide(false); found->SetVisible(true); // all games with languages start as visible } } } void Layout::AddPalette(const std::string &name, u8 key_set) { resources.palettes[key_set].push_back(name); if(FindTexture(name) != 0) return; Texture *texture = new Texture; texture->setName(name); resources.textures.push_back(texture); } Pane* Layout::FindPane(const std::string& find_name) { for(u32 i = 0; i < panes.size(); ++i) { Pane* found = panes[i]->FindPane(find_name); if(found) return found; } return NULL; } Material* Layout::FindMaterial(const std::string& find_name) { for(u32 i = 0; i < resources.materials.size(); ++i) { if (find_name.compare(0, 20, resources.materials[i]->getName()) == 0) return resources.materials[i]; } return NULL; } Texture* Layout::FindTexture(const std::string& find_name) { for(u32 i = 0; i < resources.textures.size(); ++i) { if (find_name == resources.textures[i]->getName()) return resources.textures[i]; } return NULL; }