mirror of
https://github.com/OpenSpace/OpenSpace.git
synced 2026-01-04 18:51:17 -06:00
437 lines
16 KiB
C++
437 lines
16 KiB
C++
/*****************************************************************************************
|
|
* *
|
|
* OpenSpace *
|
|
* *
|
|
* Copyright (c) 2014-2024 *
|
|
* *
|
|
* 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/screenspaceimageonline.h>
|
|
|
|
#include <openspace/documentation/documentation.h>
|
|
#include <openspace/documentation/verifier.h>
|
|
#include <openspace/engine/globals.h>
|
|
#include <openspace/util/timemanager.h>
|
|
#include <openspace/engine/downloadmanager.h>
|
|
#include <ghoul/filesystem/filesystem.h>
|
|
#include <ghoul/io/texture/texturereader.h>
|
|
#include <ghoul/logging/logmanager.h>
|
|
#include <ghoul/opengl/texture.h>
|
|
#include <ghoul/opengl/programobject.h>
|
|
#include <json/json.hpp>
|
|
#include <optional>
|
|
#include <iostream>
|
|
#include <iomanip>
|
|
#include <sstream>
|
|
#include <string_view>
|
|
#include <fstream>
|
|
#include <ctime>
|
|
|
|
namespace {
|
|
constexpr std::string_view _loggerCat = "ScreenSpaceImageOnline";
|
|
|
|
constexpr openspace::properties::Property::PropertyInfo TextureInfo = {
|
|
"URL",
|
|
"Image URL",
|
|
"The URL of the webpage containing the JSON data with image URLs."
|
|
"The image with the closest timestamp to the current time will be displayed.",
|
|
openspace::properties::Property::Visibility::User
|
|
};
|
|
|
|
constexpr openspace::properties::Property::PropertyInfo DataFileInfo = {
|
|
"JsonFilePath",
|
|
"Json File Path",
|
|
"The file path to the JSON data.",
|
|
openspace::properties::Property::Visibility::User
|
|
};
|
|
|
|
struct [[codegen::Dictionary(ScreenSpaceImageOnline)]] Parameters {
|
|
std::optional<std::string> url [[codegen::key("URL")]];
|
|
std::string jsonFilePath;
|
|
|
|
};
|
|
#include "screenspaceimageonline_codegen.cpp"
|
|
} // namespace
|
|
|
|
namespace openspace {
|
|
|
|
documentation::Documentation ScreenSpaceImageOnline::Documentation() {
|
|
return codegen::doc<Parameters>("base_screenspace_image_online");
|
|
}
|
|
|
|
ScreenSpaceImageOnline::ScreenSpaceImageOnline(const ghoul::Dictionary& dictionary)
|
|
: ScreenSpaceRenderable(dictionary)
|
|
, _textureIsDirty(false)
|
|
, _jsonDownloaded(false)
|
|
, _texturePath(TextureInfo)
|
|
{
|
|
const Parameters p = codegen::bake<Parameters>(dictionary);
|
|
|
|
std::string identifier;
|
|
if (dictionary.hasValue<std::string>(KeyIdentifier)) {
|
|
identifier = dictionary.value<std::string>(KeyIdentifier);
|
|
}
|
|
else {
|
|
identifier = "ScreenSpaceImageOnline";
|
|
}
|
|
identifier = makeUniqueIdentifier(identifier);
|
|
setIdentifier(std::move(identifier));
|
|
|
|
_texturePath.onChange([this]() { _textureIsDirty = true; });
|
|
//_texturePath = p.url.value_or(_texturePath);
|
|
_texturePath = absPath(p.jsonFilePath).string();
|
|
addProperty(_texturePath);
|
|
|
|
_lastCheckedSoftwareTimestamp = getCurrentTimestamp();
|
|
|
|
}
|
|
|
|
ScreenSpaceImageOnline::~ScreenSpaceImageOnline() {}
|
|
|
|
bool ScreenSpaceImageOnline::deinitializeGL() {
|
|
_texture = nullptr;
|
|
return ScreenSpaceRenderable::deinitializeGL();
|
|
}
|
|
|
|
//void ScreenSpaceImageOnline::update() {
|
|
// auto currentTime = std::chrono::system_clock::now();
|
|
|
|
// // Check if one minute has passed since the last check
|
|
// if (std::chrono::duration_cast<std::chrono::minutes>(currentTime - _lastCheckedTime).count() >= 1) {
|
|
// _lastCheckedTime = std::chrono::system_clock::now(); // Update the last checked time
|
|
|
|
// // Read the JSON from the local file system
|
|
// std::string jsonFilePath = _texturePath.value();
|
|
// std::ifstream file(jsonFilePath);
|
|
|
|
// if (!file.is_open()) {
|
|
// LERROR("Could not open JSON file at " + jsonFilePath);
|
|
// return;
|
|
// }
|
|
|
|
// // Parse the JSON data
|
|
// std::stringstream buffer;
|
|
// buffer << file.rdbuf();
|
|
// std::string jsonContent = buffer.str();
|
|
// std::string currentTimestamp = getCurrentTimestamp();
|
|
|
|
// // Find the closest image URL based on the current timestamp
|
|
// std::string closestUrl = findClosestTimestampUrl(jsonContent, currentTimestamp);
|
|
|
|
// if (!closestUrl.empty()) {
|
|
// // Update the currently used texture URL and load the new image
|
|
// _currentTextureUrl = closestUrl;
|
|
// _textureIsDirty = true; // Mark texture as dirty to reload
|
|
// }
|
|
// }
|
|
|
|
// // Load the image if the texture is dirty
|
|
// if (_textureIsDirty && !_currentTextureUrl.empty()) {
|
|
// loadImage(_currentTextureUrl); // Load the image based on the current URL
|
|
// _textureIsDirty = false; // Reset the dirty flag
|
|
// }
|
|
//}
|
|
|
|
|
|
|
|
|
|
void ScreenSpaceImageOnline::update() {
|
|
std::string currentTimestamp = getCurrentTimestamp();
|
|
|
|
// Check if we are not on the same update frame
|
|
if (currentTimestamp != _lastCheckedSoftwareTimestamp) {
|
|
|
|
//std::string lastCheckedTime = parseTimestamp(_lastCheckedSoftwareTimestamp);
|
|
//std::string currentTime = parseTimestamp(currentTimestamp);
|
|
|
|
|
|
//double timeDiff = std::abs(std::difftime(currentTime, lastCheckedTime)) / 60.0;
|
|
|
|
//if (timeDiff >= 1) { // If 1 minutes has passed
|
|
_lastCheckedSoftwareTimestamp = currentTimestamp;
|
|
|
|
std::string jsonFilePath = _texturePath.value();
|
|
std::ifstream file(jsonFilePath);
|
|
|
|
if (!file.is_open()) {
|
|
LERROR("Could not open JSON file at " + jsonFilePath);
|
|
return;
|
|
}
|
|
|
|
// Parse the JSON data
|
|
std::stringstream buffer;
|
|
buffer << file.rdbuf();
|
|
std::string jsonContent = buffer.str();
|
|
|
|
std::string closestUrl = findClosestTimestampUrl(jsonContent, currentTimestamp);
|
|
|
|
if (!closestUrl.empty()) {
|
|
// Update the currently used texture URL
|
|
_currentTextureUrl = closestUrl;
|
|
_textureIsDirty = true; // Mark texture as dirty to reload
|
|
}
|
|
//}
|
|
}
|
|
|
|
if (_textureIsDirty && !_currentTextureUrl.empty()) {
|
|
loadImage(_currentTextureUrl);
|
|
_textureIsDirty = false;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*std::future<DownloadManager::MemoryFile> ScreenSpaceImageOnline::downloadJsonToMemory(const std::string& url) {
|
|
if (url.empty()) {
|
|
LERROR("Attempted to download JSON with an empty URL.");
|
|
return std::future<DownloadManager::MemoryFile>();
|
|
}
|
|
|
|
return global::downloadManager->fetchFile(
|
|
url,
|
|
[url](const DownloadManager::MemoryFile&) {
|
|
LDEBUG("Download to memory finished for JSON data");
|
|
},
|
|
[url](const std::string& err) {
|
|
LERROR(std::format("Download to memory failed for JSON data from URL '{}': {}", url, err));
|
|
}
|
|
);
|
|
}*/
|
|
|
|
|
|
std::future<DownloadManager::MemoryFile> ScreenSpaceImageOnline::downloadImageToMemory(
|
|
const std::string& url)
|
|
{
|
|
return global::downloadManager->fetchFile(
|
|
url,
|
|
[url](const DownloadManager::MemoryFile&) {
|
|
LDEBUG("Download to memory finished for screen space image");
|
|
},
|
|
[url](const std::string& err) {
|
|
LDEBUG(std::format(
|
|
"Download to memory failed for screen space image: {}", err
|
|
));
|
|
}
|
|
);
|
|
}
|
|
|
|
|
|
std::string ScreenSpaceImageOnline::findClosestTimestampUrl(const std::string& jsonContent, std::string& currentTimestamp) {
|
|
auto json = nlohmann::json::parse(jsonContent);
|
|
|
|
std::string closestUrl;
|
|
double closestTimeDiff = std::numeric_limits<double>::max();
|
|
|
|
std::time_t currentTime = parseTimestamp(currentTimestamp);
|
|
|
|
for (const auto& fileEntry : json["files"]) {
|
|
std::string timestamp = fileEntry["timestamp"].get<std::string>();
|
|
std::time_t fileTime = parseTimestamp(timestamp);
|
|
|
|
// Calculate time difference
|
|
double timeDiff = std::abs(difftime(fileTime, currentTime));
|
|
|
|
// Check for the closest timestamp
|
|
if (timeDiff < closestTimeDiff) {
|
|
closestTimeDiff = timeDiff;
|
|
closestUrl = fileEntry["url"].get<std::string>();
|
|
//std::cout << "Closest URL: " << closestUrl << " with time difference: " << closestTimeDiff << std::endl;
|
|
}
|
|
|
|
//std::cout << "Current time: " << currentTime << " file time: " << fileTime << std::endl;
|
|
|
|
|
|
//if (currentTime == fileTime)
|
|
//{
|
|
// closestUrl = fileEntry["url"].get<std::string>();
|
|
//}
|
|
}
|
|
|
|
//if (closestUrl.empty()) {
|
|
// LERROR(std::format("No valid URL found for the given timestamps."));
|
|
//}
|
|
|
|
return closestUrl;
|
|
}
|
|
|
|
|
|
|
|
std::time_t ScreenSpaceImageOnline::parseTimestamp(std::string& timestamp) {
|
|
//std::replace(timestamp.begin(), timestamp.end(), '.', ' '); // Remove milliseconds
|
|
|
|
std::tm tm = {};
|
|
std::istringstream ss(timestamp);
|
|
ss >> std::get_time(&tm, "%Y-%m-%d %H:%M:%S");
|
|
|
|
std::ostringstream oss;
|
|
oss << std::put_time(&tm, "%Y-%m-%d %H:%M:00.0");
|
|
return std::mktime(&tm);
|
|
//return oss.str();
|
|
}
|
|
|
|
std::string ScreenSpaceImageOnline::getClosestUrl(const std::string& jsonFilePath, const std::string& currentTime)
|
|
{
|
|
std::ifstream file(jsonFilePath);
|
|
nlohmann::json root;
|
|
file >> root;
|
|
|
|
std::string closestUrl = "";
|
|
double closestTimeDiff = std::numeric_limits<double>::max();
|
|
|
|
for (const auto& fileEntry : root["files"]) {
|
|
std::string timestamp = fileEntry["timestamp"].get<std::string>();
|
|
double timeDiff = std::abs(std::difftime(std::stoi(timestamp.substr(0, 10)), std::stoi(currentTime.substr(0, 10))));
|
|
|
|
if (timeDiff < closestTimeDiff) {
|
|
closestTimeDiff = timeDiff;
|
|
closestUrl = fileEntry["url"].get<std::string>();
|
|
}
|
|
}
|
|
return closestUrl;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ScreenSpaceImageOnline::loadImage(const std::string& imageUrl) {
|
|
// Download or load the image from the local disk or the web
|
|
|
|
|
|
if (!_imageFuture.valid()) {
|
|
std::future<DownloadManager::MemoryFile> future = downloadImageToMemory(
|
|
imageUrl
|
|
);
|
|
if (future.valid()) {
|
|
_imageFuture = std::move(future);
|
|
}
|
|
}
|
|
|
|
if (_imageFuture.valid() && DownloadManager::futureReady(_imageFuture)) {
|
|
const DownloadManager::MemoryFile imageFile = _imageFuture.get();
|
|
|
|
if (imageFile.corrupted) {
|
|
LERROR(std::format(
|
|
"Error loading image from URL '{}'", imageUrl
|
|
));
|
|
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) {
|
|
// Images don't need to start on 4-byte boundaries, for example if the
|
|
// image is only RGB
|
|
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();
|
|
_textureIsDirty = false;
|
|
}
|
|
}
|
|
catch (const ghoul::io::TextureReader::InvalidLoadException& e) {
|
|
_textureIsDirty = false;
|
|
LERRORC(e.component, e.message);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
std::string ScreenSpaceImageOnline::formatTimeForData(std::string_view timeStr) {
|
|
std::string formattedTime(timeStr);
|
|
|
|
// Convert to the format YYYY-MM-DDTHH:MM:00Z
|
|
std::replace(formattedTime.begin(), formattedTime.end(), 'T', ' ');
|
|
|
|
std::tm tm = {};
|
|
std::istringstream ss(formattedTime);
|
|
ss >> std::get_time(&tm, "%Y %b %d %H:%M:%S");
|
|
|
|
std::ostringstream oss;
|
|
oss << std::put_time(&tm, "%Y-%m-%d %H:%M:00.0");
|
|
return oss.str();
|
|
}
|
|
|
|
|
|
std::string ScreenSpaceImageOnline::getCurrentTimestamp() {
|
|
std::string_view currentTimeStr = global::timeManager->time().UTC();
|
|
std::string formattedTime(currentTimeStr);
|
|
|
|
// Parse the formatted time
|
|
std::tm tm = {};
|
|
std::istringstream ss(formattedTime);
|
|
ss >> std::get_time(&tm, "%Y %b %d %H:%M:%S");
|
|
|
|
// Create a new formatted string in JSON timestamp format
|
|
std::ostringstream jsonTimestamp;
|
|
jsonTimestamp << std::put_time(&tm, "%Y-%m-%d %H:%M:%S") << ".0"; // Adding .0
|
|
|
|
std::string formattedTime2 = formatTimeForData(currentTimeStr);
|
|
|
|
|
|
return formattedTime2;
|
|
//return jsonTimestamp.str();
|
|
}
|
|
|
|
|
|
|
|
//std::string ScreenSpaceImageOnline::getCurrentTimestamp() {
|
|
// // Get the current time
|
|
// std::string_view currentTimeStr = global::timeManager->time().UTC();
|
|
// std::string formattedTime(currentTimeStr);
|
|
//
|
|
// // Parse the formatted time
|
|
// std::tm tm = {};
|
|
// std::istringstream ss(formattedTime);
|
|
// ss >> std::get_time(&tm, "%Y %b %d %H:%M:%S");
|
|
//
|
|
// // Convert to time_point
|
|
// auto timePoint = std::chrono::system_clock::from_time_t(std::mktime(&tm));
|
|
//
|
|
// // Create a new formatted string in JSON timestamp format
|
|
// std::ostringstream jsonTimestamp;
|
|
// jsonTimestamp << std::put_time(&tm, "%Y-%m-%d %H:%M:%S") << ".0"; // Adding .0 for milliseconds
|
|
//
|
|
// return jsonTimestamp.str();
|
|
//}
|
|
|
|
void ScreenSpaceImageOnline::bindTexture() {
|
|
if (_texture) {
|
|
_texture->bind();
|
|
}
|
|
}
|
|
|
|
} // namespace openspace
|