/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * 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 2 * 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "common/debug.h" #include "zvision/search_manager.h" #include "zvision/zfs_archive.h" #include "common/fs.h" #include "common/stream.h" namespace ZVision { sManager::sManager(const Common::String &root_path, int depth) { _root = root_path; if (_root[_root.size() - 1] == '\\' || _root[_root.size() - 1] == '/') _root.deleteLastChar(); Common::FSNode fs_node(_root); list_dir_recursive(dir_list, fs_node, depth); for (Common::List::iterator it = dir_list.begin(); it != dir_list.end();) if (it->size() == _root.size()) it = dir_list.erase(it); else if (it->size() > _root.size()) { *it = Common::String(it->c_str() + _root.size() + 1); it++; } else it++; } sManager::~sManager() { Common::List::iterator it = archList.begin(); while (it != archList.end()) { delete *it; it++; } archList.clear(); } void sManager::addPatch(const Common::String &src, const Common::String &dst) { Common::String lw_name = dst; lw_name.toLowercase(); sManager::MatchList::iterator it = files.find(lw_name); if (it != files.end()) { lw_name = src; lw_name.toLowercase(); files[lw_name] = it->_value; } } void sManager::addFile(const Common::String &name, Common::Archive *arch) { bool addArch = true; Common::List::iterator it = archList.begin(); while (it != archList.end()) { if (*it == arch) { addArch = false; break; } it++; } if (addArch) archList.push_back(arch); Common::String lw_name = name; lw_name.toLowercase(); sManager::Node nod; nod.name = lw_name; nod.arch = arch; sManager::MatchList::iterator fit = files.find(lw_name); if (fit == files.end()) { files[lw_name] = nod; } else { Common::SeekableReadStream *stream = fit->_value.arch->createReadStreamForMember(fit->_value.name); if (stream) { if (stream->size() < 10) fit->_value.arch = arch; delete stream; } else { files[lw_name] = nod; } } } Common::File *sManager::openFile(const Common::String &name) { Common::String lw_name = name; lw_name.toLowercase(); sManager::MatchList::iterator fit = files.find(lw_name); if (fit != files.end()) { Common::File *tmp = new Common::File(); tmp->open(fit->_value.name, *fit->_value.arch); return tmp; } return NULL; } bool sManager::openFile(Common::File &file, const Common::String &name) { Common::String lw_name = name; lw_name.toLowercase(); sManager::MatchList::iterator fit = files.find(lw_name); if (fit != files.end()) return file.open(fit->_value.name, *fit->_value.arch); return false; } bool sManager::hasFile(const Common::String &name) { Common::String lw_name = name; lw_name.toLowercase(); sManager::MatchList::iterator fit = files.find(lw_name); if (fit != files.end()) return true; return false; } void sManager::loadZix(const Common::String &name) { Common::File file; file.open(name); Common::String line; while (!file.eos()) { line = file.readLine(); if (line.matchString("----------*", true)) break; } if (file.eos()) return; Common::Array archives; while (!file.eos()) { line = file.readLine(); line.trim(); if (line.matchString("----------*", true)) break; else if (line.matchString("DIR:*", true)) { Common::String path(line.c_str() + 5); Common::Archive *arc; char n_path[128]; strcpy(n_path, path.c_str()); for (uint i = 0; i < path.size(); i++) if (n_path[i] == '\\') n_path[i] = '/'; path = Common::String(n_path); if (path.size() && path[0] == '.') path.deleteChar(0); if (path.size() && path[0] == '/') path.deleteChar(0); if (path.matchString("*.zfs", true)) arc = new ZfsArchive(path); else { if (path.size()) { if (path[path.size() - 1] == '\\' || path[path.size() - 1] == '/') path.deleteLastChar(); if (path.size()) for (Common::List::iterator it = dir_list.begin(); it != dir_list.end(); ++it) if (path.equalsIgnoreCase(*it)) { path = *it; break; } } path = Common::String::format("%s/%s", _root.c_str(), path.c_str()); arc = new Common::FSDirectory(path); } archives.push_back(arc); } } if (file.eos()) return; while (!file.eos()) { line = file.readLine(); line.trim(); uint dr = 0; char buf[32]; if (sscanf(line.c_str(), "%u %s", &dr, buf) == 2) { if (dr < archives.size() && dr > 0) { addFile(Common::String(buf), archives[dr - 1]); } } } } void sManager::addDir(const Common::String &name) { Common::String path; for (Common::List::iterator it = dir_list.begin(); it != dir_list.end(); ++it) if (name.equalsIgnoreCase(*it)) { path = *it; break; } if (path.size() == 0) return; path = Common::String::format("%s/%s", _root.c_str(), path.c_str()); Common::FSDirectory *dir = new Common::FSDirectory(path); Common::ArchiveMemberList list; dir->listMatchingMembers(list, "*.zfs"); for (Common::ArchiveMemberList::iterator iter = list.begin(); iter != list.end(); ++iter) { Common::String flname = (*iter)->getName(); ZfsArchive *zfs = new ZfsArchive(Common::String::format("%s/%s", name.c_str(), flname.c_str())); Common::ArchiveMemberList zfslist; zfs->listMembers(zfslist); for (Common::ArchiveMemberList::iterator ziter = zfslist.begin(); ziter != zfslist.end(); ++ziter) { Common::String z_name = (*ziter)->getName(); addFile(z_name, zfs); } } list.clear(); dir->listMembers(list); for (Common::ArchiveMemberList::iterator iter = list.begin(); iter != list.end(); ++iter) { Common::String flname = (*iter)->getName(); addFile(flname, dir); } } void sManager::list_dir_recursive(Common::List &_list, const Common::FSNode &fs_node, int depth) { Common::FSList fs_list; fs_node.getChildren(fs_list); _list.push_back(fs_node.getPath()); if (depth > 1) for (Common::FSList::const_iterator it = fs_list.begin(); it != fs_list.end(); ++it) list_dir_recursive(_list, *it, depth - 1); } } // End of namespace ZVision