mirror of
https://github.com/OpenSpace/OpenSpace.git
synced 2026-01-02 09:41:13 -06:00
245 lines
8.9 KiB
C++
245 lines
8.9 KiB
C++
/*****************************************************************************************
|
|
* *
|
|
* OpenSpace *
|
|
* *
|
|
* Copyright (c) 2014-2025 *
|
|
* *
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of this *
|
|
* software and associated documentation files (the "Software"), to deal in the Software *
|
|
* without restriction, including without limitation the rights to use, copy, modify, *
|
|
* merge, publish, distribute, sublicense, and/or sell copies of the Software, and to *
|
|
* permit persons to whom the Software is furnished to do so, subject to the following *
|
|
* conditions: *
|
|
* *
|
|
* The above copyright notice and this permission notice shall be included in all copies *
|
|
* or substantial portions of the Software. *
|
|
* *
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, *
|
|
* INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A *
|
|
* PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT *
|
|
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF *
|
|
* CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE *
|
|
* OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. *
|
|
****************************************************************************************/
|
|
|
|
#include <modules/base/rendering/screenspacetimevaryingimageonline.h>
|
|
|
|
#include <openspace/documentation/documentation.h>
|
|
#include <openspace/documentation/verifier.h>
|
|
#include <openspace/engine/globals.h>
|
|
#include <openspace/json.h>
|
|
#include <openspace/util/timemanager.h>
|
|
#include <ghoul/filesystem/filesystem.h>
|
|
#include <ghoul/io/texture/texturereader.h>
|
|
#include <ghoul/logging/logmanager.h>
|
|
#include <ghoul/opengl/texture.h>
|
|
#include <fstream>
|
|
|
|
namespace {
|
|
constexpr std::string_view _loggerCat = "ScreenSpaceTimeVaryingImageOnline";
|
|
|
|
constexpr openspace::properties::Property::PropertyInfo FileInfo = {
|
|
"FilePath",
|
|
"File path",
|
|
"The file path to the data containing information about when to display which "
|
|
"image.",
|
|
openspace::properties::Property::Visibility::User
|
|
};
|
|
|
|
// This `ScreenSpaceRenderable` displays an image based on the current in-game
|
|
// simulation time. The image shown is selected from a JSON file containing
|
|
// timestamp-URL pairs. The image with the closest timestamp before or equal to the
|
|
// current time is displayed.
|
|
//
|
|
// Example JSON format:
|
|
// {
|
|
// "files": [
|
|
// {"timestamp": "2024-05-10 15:00:00.0","url": "https://example.com/image1.png"},
|
|
// {"timestamp": "2024-05-10 15:02:00.0","url": "https://example.com/image2.png"}
|
|
// ]
|
|
// }
|
|
struct [[codegen::Dictionary(ScreenSpaceTimeVaryingImageOnline)]] Parameters {
|
|
// [[codegen::verbatim(FileInfo.description)]]
|
|
std::filesystem::path filePath;
|
|
};
|
|
#include "screenspacetimevaryingimageonline_codegen.cpp"
|
|
} // namespace
|
|
|
|
namespace openspace {
|
|
|
|
documentation::Documentation ScreenSpaceTimeVaryingImageOnline::Documentation() {
|
|
return codegen::doc<Parameters>("base_screenspace_time_varying_image_online");
|
|
}
|
|
|
|
ScreenSpaceTimeVaryingImageOnline::ScreenSpaceTimeVaryingImageOnline(
|
|
const ghoul::Dictionary& dictionary)
|
|
: ScreenSpaceRenderable(dictionary)
|
|
, _jsonFilePath(FileInfo, "")
|
|
{
|
|
const Parameters p = codegen::bake<Parameters>(dictionary);
|
|
|
|
_jsonFilePath = p.filePath.string();
|
|
_jsonFilePath.onChange([this]() {
|
|
loadJsonData(_jsonFilePath.value());
|
|
});
|
|
addProperty(_jsonFilePath);
|
|
}
|
|
|
|
void ScreenSpaceTimeVaryingImageOnline::initialize() {
|
|
ScreenSpaceRenderable::initialize();
|
|
|
|
loadJsonData(_jsonFilePath.value());
|
|
}
|
|
|
|
void ScreenSpaceTimeVaryingImageOnline::deinitializeGL() {
|
|
_texture = nullptr;
|
|
|
|
ScreenSpaceRenderable::deinitializeGL();
|
|
}
|
|
|
|
void ScreenSpaceTimeVaryingImageOnline::loadJsonData(const std::filesystem::path& path) {
|
|
std::ifstream file = std::ifstream(path);
|
|
if (!file.is_open()) {
|
|
throw ghoul::RuntimeError(std::format("Could not open JSON file at '{}'", path));
|
|
}
|
|
|
|
nlohmann::json json;
|
|
file >> json;
|
|
|
|
if (json.find("files") == json.end()) {
|
|
throw ghoul::RuntimeError(std::format(
|
|
"Error loading JSON file. No 'files' was found in '{}'", path
|
|
));
|
|
}
|
|
|
|
_timestamps.clear();
|
|
_urls.clear();
|
|
|
|
for (const nlohmann::json& entry : json["files"]) {
|
|
const std::string& timestamp = entry["timestamp"].get<std::string>();
|
|
double j2000 = Time::convertTime(timestamp);
|
|
_timestamps.push_back(j2000);
|
|
_urls[j2000] = entry["url"].get<std::string>();
|
|
}
|
|
|
|
std::sort(_timestamps.begin(), _timestamps.end());
|
|
computeSequenceEndTime();
|
|
}
|
|
|
|
void ScreenSpaceTimeVaryingImageOnline::computeSequenceEndTime() {
|
|
if (_timestamps.size() <= 1) {
|
|
return;
|
|
}
|
|
|
|
double first = _timestamps.front();
|
|
double last = _timestamps.back();
|
|
double avg = (last - first) / static_cast<double>(_timestamps.size() - 1);
|
|
// Extend end time so the last value remains visible for one more interval
|
|
_sequenceEndTime = last + avg;
|
|
}
|
|
|
|
void ScreenSpaceTimeVaryingImageOnline::update() {
|
|
if (_timestamps.empty()) {
|
|
return;
|
|
}
|
|
|
|
const double current = global::timeManager->time().j2000Seconds();
|
|
|
|
if (current < _timestamps.front() || current >= _sequenceEndTime) {
|
|
_activeIndex = -1;
|
|
_texture = nullptr;
|
|
_currentUrl.clear();
|
|
return;
|
|
}
|
|
|
|
if (current >= _timestamps.front() && current < _sequenceEndTime) {
|
|
if (int idx = activeIndex(current); idx != _activeIndex) {
|
|
_activeIndex = idx;
|
|
std::string url = _urls[_timestamps[_activeIndex]];
|
|
if (_currentUrl != url) {
|
|
_currentUrl = url;
|
|
loadImage(url);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
_activeIndex = -1;
|
|
}
|
|
|
|
if (_imageFuture.valid() && DownloadManager::futureReady(_imageFuture)) {
|
|
const DownloadManager::MemoryFile imageFile = _imageFuture.get();
|
|
_imageFuture = std::future<DownloadManager::MemoryFile>();
|
|
if (imageFile.corrupted) {
|
|
LERROR(std::format("Error loading image from URL '{}'", _currentUrl));
|
|
return;
|
|
}
|
|
|
|
try {
|
|
std::unique_ptr<ghoul::opengl::Texture> texture =
|
|
ghoul::io::TextureReader::ref().loadTexture(
|
|
reinterpret_cast<void*>(imageFile.buffer),
|
|
imageFile.size,
|
|
2,
|
|
imageFile.format
|
|
);
|
|
|
|
if (texture) {
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
|
|
|
|
if (texture->format() == ghoul::opengl::Texture::Format::Red) {
|
|
texture->setSwizzleMask({ GL_RED, GL_RED, GL_RED, GL_ONE });
|
|
}
|
|
|
|
texture->uploadTexture();
|
|
texture->setFilter(ghoul::opengl::Texture::FilterMode::LinearMipMap);
|
|
texture->purgeFromRAM();
|
|
|
|
_texture = std::move(texture);
|
|
_objectSize = _texture->dimensions();
|
|
}
|
|
}
|
|
catch (const ghoul::io::TextureReader::InvalidLoadException& e) {
|
|
LERRORC(e.component, e.message);
|
|
}
|
|
}
|
|
}
|
|
|
|
int ScreenSpaceTimeVaryingImageOnline::activeIndex(double currentTime) const {
|
|
if (_timestamps.empty()) {
|
|
return -1;
|
|
}
|
|
|
|
auto it = std::upper_bound(_timestamps.begin(), _timestamps.end(), currentTime);
|
|
if (it == _timestamps.begin()) {
|
|
return 0;
|
|
}
|
|
else if (it != _timestamps.end()) {
|
|
return static_cast<int>(std::distance(_timestamps.begin(), it)) - 1;
|
|
}
|
|
else {
|
|
return static_cast<int>(_timestamps.size()) - 1;
|
|
}
|
|
}
|
|
|
|
void ScreenSpaceTimeVaryingImageOnline::loadImage(const std::string& imageUrl) {
|
|
if (_imageFuture.valid()) {
|
|
return;
|
|
}
|
|
|
|
_imageFuture = global::downloadManager->fetchFile(
|
|
imageUrl,
|
|
[](const DownloadManager::MemoryFile&) {},
|
|
[](const std::string& e) {
|
|
LERROR(std::format("Download failed: {}", e));
|
|
}
|
|
);
|
|
}
|
|
|
|
void ScreenSpaceTimeVaryingImageOnline::bindTexture() {
|
|
if (_texture) [[likely]] {
|
|
_texture->bind();
|
|
}
|
|
}
|
|
|
|
} // namespace openspace
|