Files
scummvm-cursorfix/engines/tetraedge/te/te_images_sequence.cpp
2026-02-02 04:50:13 +01:00

170 lines
4.8 KiB
C++

/* 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 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/file.h"
#include "image/png.h"
#include "graphics/surface.h"
#include "graphics/managed_surface.h"
#include "tetraedge/tetraedge.h"
#include "tetraedge/te/te_core.h"
#include "tetraedge/te/te_images_sequence.h"
namespace Tetraedge {
TeImagesSequence::TeImagesSequence() : _width(0), _height(0), _curFrame(0), _frameRate(0) {
}
TeImagesSequence::~TeImagesSequence() {
for (auto surf : _cachedSurfaces) {
if (surf)
delete surf;
}
}
/*static*/
bool TeImagesSequence::matchExtension(const Common::String &extn) {
return extn == "anim" || extn == "animcached";
}
static bool compareNodes(const TetraedgeFSNode &left, const TetraedgeFSNode &right) {
return left.getPath().toString('/') < right.getPath().toString('/');
}
bool TeImagesSequence::load(const TetraedgeFSNode &dir) {
if (!dir.isDirectory()) {
warning("TeImagesSequence::load:: not a directory %s", dir.toString().c_str());
return false;
}
TetraedgeFSList children;
if (!dir.getChildren(children, Common::FSNode::kListFilesOnly) || children.empty()) {
warning("TeImagesSequence::load:: couldn't get children of %s", dir.toString().c_str());
return false;
}
Common::sort(children.begin(), children.end(), compareNodes);
dir.maybeAddToSearchMan();
for (TetraedgeFSNode &child : children) {
const Common::String fileName = child.getName();
if (fileName.size() <= 10 || fileName.substr(fileName.size() - 7) != "fps.png")
continue;
Common::String fstart = fileName.substr(0, fileName.size() - 7);
int frameno = 0;
int fps = 0;
if (sscanf(fstart.c_str(), "%d-%d", &frameno, &fps) != 2) {
warning("TeImagesSequence::load can't match %s", fileName.c_str());
continue;
}
Common::SeekableReadStream *stream = child.createReadStream();
if (!stream) {
warning("TeImagesSequence::load can't open %s", child.toString().c_str());
continue;
}
// Only do the deep check for the first file to get dimensions.
// If the images are small then cache them to avoid reloading each frame each time.
if (!_width || (_width < 100 && _height < 100)) {
Image::PNGDecoder png;
if (!png.loadStream(*stream)) {
warning("Image sequence failed to load png %s", child.toString().c_str());
delete stream;
return false;
}
_frameRate = fps;
const Graphics::Surface *pngsurf = png.getSurface();
assert(pngsurf);
_width = pngsurf->w;
_height = pngsurf->h;
if (_width < 100 && _height < 100) {
Graphics::ManagedSurface *surf = new Graphics::ManagedSurface();
surf->copyFrom(*pngsurf);
_cachedSurfaces.push_back(surf);
} else {
_cachedSurfaces.push_back(nullptr);
}
} else {
_cachedSurfaces.push_back(nullptr);
}
_files.push_back(child);
delete stream;
}
return true;
}
bool TeImagesSequence::update(uint i, TeImage &imgout) {
_curFrame = i;
if (i >= _files.size())
return false;
if (_cachedSurfaces[i] == nullptr) {
Common::SeekableReadStream *stream = _files[i].createReadStream();
if (!stream)
error("Open %s failed.. it was ok before?", _files[i].getName().c_str());
Image::PNGDecoder png;
if (!png.loadStream(*stream)) {
warning("Image sequence failed to load png %s", _files[i].getName().c_str());
delete stream;
return false;
}
const Graphics::Surface *surf = png.getSurface();
assert(surf);
imgout.setAccessName(_files[i].getPath());
if (imgout.w == surf->w && imgout.h == surf->h && imgout.format == surf->format) {
imgout.copyFrom(*surf);
delete stream;
return true;
}
} else {
const Graphics::ManagedSurface *surf = _cachedSurfaces[i];
if (imgout.w == surf->w && imgout.h == surf->h && imgout.format == surf->format) {
imgout.setAccessName(_files[i].getPath());
imgout.copyFrom(*surf);
return true;
}
}
error("TODO: Implement TeImagesSequence::update for different sizes");
}
bool TeImagesSequence::isAtEnd() {
return _curFrame >= _files.size();
}
Graphics::PixelFormat TeImagesSequence::pixelFormat() {
return Graphics::PixelFormat::createFormatRGBA32();
}
} // end namespace Tetraedge