Files
OpenSpace/src/rendering/transferfunction.cpp
Alexander Bock 4952f8f977 Code cleanup branch (#618)
* Make height map fallback layer work again
  * Add documentation to joystick button bindings
  * Removed grouped property headers
  * Add new version number constant generated by CMake
  * Make Joystick deadzone work properly
  * Change the startup date on Earth to today
  * Fix key modifier handling
  * Add debugging indices for TreeNodeDebugging
  * Fix script schedule for OsirisRex
  * Do not open Mission schedule automatically
  * Upload default projection texture automatically

  * General code cleanup
  * Fix check_style_guide warnings
  * Remove .clang-format
  * MacOS compile fixes
  * Clang analyzer fixes
2018-06-10 04:47:34 +00:00

253 lines
8.4 KiB
C++

/*****************************************************************************************
* *
* OpenSpace *
* *
* Copyright (c) 2014-2018 *
* *
* 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 <openspace/rendering/transferfunction.h>
#include <ghoul/filesystem/cachemanager.h>
#include <ghoul/filesystem/file.h>
#include <ghoul/filesystem/filesystem.h>
#include <ghoul/filesystem/file.h>
#include <ghoul/io/texture/texturereader.h>
#include <ghoul/logging/logmanager.h>
#include <ghoul/opengl/texture.h>
#include <iterator>
#include <fstream>
#include <string>
namespace {
constexpr const char* _loggerCat = "TransferFunction";
// @TODO Replace with Filesystem::File extension
bool hasExtension(const std::string& filepath, const std::string& extension) {
std::string ending = "." + extension;
if (filepath.length() > ending.length()) {
return (0 == filepath.compare(
filepath.length() - ending.length(),
ending.length(), ending));
} else {
return false;
}
}
} // namespace
namespace openspace {
TransferFunction::TransferFunction(const std::string& filepath,
TfChangedCallback tfChangedCallback)
: _filepath(filepath)
{
setPath(filepath);
setCallback(std::move(tfChangedCallback));
}
TransferFunction::~TransferFunction() {} // NOLINT
void TransferFunction::setPath(const std::string& filepath) {
if (_file) {
_file = nullptr;
}
std::string f = absPath(filepath);
if (!FileSys.fileExists(f)) {
LERROR("Could not find transfer function file.");
_file = nullptr;
return;
}
_filepath = f;
_file = std::make_unique<ghoul::filesystem::File>(
_filepath,
ghoul::filesystem::File::RawPath::Yes
);
_needsUpdate = true;
_file->setCallback([this](const ghoul::filesystem::File&) {
_needsUpdate = true;
});
}
ghoul::opengl::Texture& TransferFunction::texture() {
ghoul_assert(_texture != nullptr, "Transfer function is null");
update();
return *_texture.get();
}
void TransferFunction::update() {
if (_needsUpdate) {
if (hasExtension(_filepath, "txt")) {
setTextureFromTxt();
} else {
setTextureFromImage();
}
_texture->uploadTexture();
_needsUpdate = false;
if (_tfChangedCallback) {
_tfChangedCallback(*this);
}
}
}
void TransferFunction::setCallback(TfChangedCallback callback) {
_tfChangedCallback = std::move(callback);
}
void TransferFunction::setTextureFromTxt(std::shared_ptr<ghoul::opengl::Texture> ptr) {
std::ifstream in;
in.open(_filepath.c_str());
if (!in.is_open()) {
throw ghoul::FileNotFoundError(_filepath);
}
int width = 512;
float lower = 0.f;
float upper = 1.f;
std::vector<MappingKey> mappingKeys;
std::string line;
while (std::getline(in, line)) {
std::istringstream iss(line);
std::string key;
iss >> key;
if (key == "width") {
iss >> width;
} else if (key == "lower") {
iss >> lower;
lower = glm::clamp(lower, 0.f, 1.f);
} else if (key == "upper") {
iss >> upper;
upper = glm::clamp(upper, lower, 1.f);
} else if (key == "mappingkey") {
float intensity;
glm::vec4 rgba = glm::vec4(0.0f);
iss >> intensity;
for(int i = 0; i < 4; ++i) {
iss >> rgba[i];
}
mappingKeys.emplace_back(intensity, rgba);
}
}
in.close();
if (mappingKeys.empty()) {
return;
}
if (mappingKeys.front().position > lower) {
mappingKeys.insert(mappingKeys.begin(), {lower, mappingKeys.front().color});
}
if (mappingKeys.back().position < upper) {
mappingKeys.emplace_back(upper, mappingKeys.back().color);
}
// allocate new float array with zeros
float* transferFunction = new float[width * 4];
for (int i = 0; i < 4 * width; ++i) {
transferFunction[i] = 0.f;
}
size_t lowerIndex = static_cast<size_t>(floorf(lower * static_cast<float>(width-1)));
size_t upperIndex = static_cast<size_t>(floorf(upper * static_cast<float>(width-1)));
auto prevKey = mappingKeys.begin();
auto currentKey = prevKey + 1;
auto lastKey = mappingKeys.end() -1;
for (size_t i = lowerIndex; i <= upperIndex; ++i) {
const float fpos = static_cast<float>(i) / static_cast<float>(width-1);
if (fpos > currentKey->position) {
prevKey = currentKey;
currentKey++;
if (currentKey == mappingKeys.end()) {
currentKey = lastKey;
}
}
const float dist = fpos - prevKey->position;
const float weight = dist / (currentKey->position - prevKey->position);
for (size_t channel = 0; channel < 4; ++channel) {
const size_t position = 4 * i + channel;
// Interpolate linearly between prev and next mapping key
float value = (prevKey->color[channel] * (1.f - weight) +
currentKey->color[channel] * weight) / 255.f;
if (channel < 3) {
// Premultiply with alpha
value *= (prevKey->color[3] * (1.f - weight) +
currentKey->color[3] * weight) / 255.f;
}
transferFunction[position] = value;
}
}
// no need to deallocate transferFunction. Ownership is transferred to the Texture.
_texture = std::make_unique<ghoul::opengl::Texture>(
transferFunction,
glm::size3_t(width, 1, 1),
ghoul::opengl::Texture::Format::RGBA,
GL_RGBA,
GL_FLOAT,
ghoul::opengl::Texture::FilterMode::Linear,
ghoul::opengl::Texture::WrappingMode::ClampToEdge
);
}
void TransferFunction::setTextureFromImage() {
_texture = ghoul::io::TextureReader::ref().loadTexture(_filepath);
_texture->setWrapping(ghoul::opengl::Texture::WrappingMode::ClampToEdge);
}
glm::vec4 TransferFunction::sample(size_t offset) {
if (!_texture) {
return glm::vec4(0.f);
}
const int nPixels = _texture->width();
// Clamp to range.
if (offset >= nPixels) {
offset = nPixels - 1;
}
if (offset < 0) {
offset = 0;
}
return _texture->texelAsFloat(offset);
}
size_t TransferFunction::width() {
update();
return _texture->width();
}
void TransferFunction::bind() {
update();
_texture->bind();
}
} // namespace openspace