You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

395 lines
16 KiB

#ifndef Magnum_Trade_AbstractImageConverter_h
#define Magnum_Trade_AbstractImageConverter_h
/*
This file is part of Magnum.
Copyright © 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019,
2020 Vladimír Vondruš <mosra@centrum.cz>
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.
*/
/** @file
* @brief Class @ref Magnum::Trade::AbstractImageConverter, enum @ref Magnum::Trade::ImageConverterFeature, enum set @ref Magnum::Trade::ImageConverterFeatures
*/
#include <Corrade/PluginManager/AbstractManagingPlugin.h>
#include "Magnum/Magnum.h"
#include "Magnum/Trade/Trade.h"
#include "Magnum/Trade/visibility.h"
namespace Magnum { namespace Trade {
/**
@brief Features supported by an image converter
@m_since{2020,06}
@see @ref ImageConverterFeatures, @ref AbstractImageConverter::features()
*/
enum class ImageConverterFeature: UnsignedByte {
/**
* Conversion to image with different format with
* @ref AbstractImageConverter::exportToImage()
*/
ConvertImage = 1 << 0,
/**
* Conversion to compressed image with
* @ref AbstractImageConverter::exportToCompressedImage()
*/
ConvertCompressedImage = 1 << 1,
/**
* Exporting to file with
* @ref AbstractImageConverter::exportToFile(const ImageView2D&, const std::string&)
*/
ConvertFile = 1 << 2,
/**
* Exporting to file with
* @ref AbstractImageConverter::exportToFile(const CompressedImageView2D&, const std::string&)
*/
ConvertCompressedFile = 1 << 3,
/**
* Exporting to raw data with
* @ref AbstractImageConverter::exportToData(const ImageView2D&).
* Implies @ref ImageConverterFeature::ConvertFile.
*/
ConvertData = ConvertFile|(1 << 4),
/**
* Exporting compressed image to raw data with
* @ref AbstractImageConverter::exportToData(const CompressedImageView2D&).
* Implies @ref ImageConverterFeature::ConvertCompressedFile.
*/
ConvertCompressedData = ConvertCompressedFile|(1 << 4)
};
/**
@brief Features supported by an image converter
@m_since{2020,06}
@see @ref AbstractImageConverter::features()
*/
typedef Containers::EnumSet<ImageConverterFeature> ImageConverterFeatures;
CORRADE_ENUMSET_OPERATORS(ImageConverterFeatures)
/** @debugoperatorenum{ImageConverterFeature} */
MAGNUM_TRADE_EXPORT Debug& operator<<(Debug& debug, ImageConverterFeature value);
/** @debugoperatorenum{ImageConverterFeatures} */
MAGNUM_TRADE_EXPORT Debug& operator<<(Debug& debug, ImageConverterFeatures value);
/**
@brief Image converter flag
@m_since{2020,06}
@see @ref ImageConverterFlags, @ref AbstractImageConverter::setFlags()
*/
enum class ImageConverterFlag: UnsignedByte {
/**
* Print verbose diagnostic during conversion. By default the converter
* only prints messages on error or when some operation might cause
* unexpected data modification or loss.
*/
Verbose = 1 << 0
/** @todo Y flip */
};
/**
@brief Image converter flags
@m_since{2020,06}
@see @ref AbstractImageConverter::setFlags()
*/
typedef Containers::EnumSet<ImageConverterFlag> ImageConverterFlags;
CORRADE_ENUMSET_OPERATORS(ImageConverterFlags)
/**
@debugoperatorenum{ImageConverterFlag}
@m_since{2020,06}
*/
MAGNUM_TRADE_EXPORT Debug& operator<<(Debug& debug, ImageConverterFlag value);
/**
@debugoperatorenum{ImageConverterFlags}
@m_since{2020,06}
*/
MAGNUM_TRADE_EXPORT Debug& operator<<(Debug& debug, ImageConverterFlags value);
/**
@brief Base for image converter plugins
Provides functionality for converting images between various internal formats
or compressing them. See @ref plugins for more information and `*ImageConverter`
classes in @ref Trade namespace for available image converter plugins.
@section Trade-AbstractImageConverter-data-dependency Data dependency
The instances returned from various functions *by design* have no dependency on
the converter instance and neither on the dynamic plugin module. In other
words, you don't need to keep the converter instance (or the plugin manager
instance) around in order to have the `*Data` instances valid. Moreover, all
@ref Corrade::Containers::Array instances returned through @ref Image,
@ref CompressedImage, @ref MeshData, @ref MaterialData, @ref AnimationData and
others are only allowed to have default deleters --- this is to avoid potential
dangling function pointer calls when destructing such instances after the
plugin module has been unloaded.
@section Trade-AbstractImageConverter-subclassing Subclassing
The plugin needs to implement the @ref doFeatures() function and one or more of
@ref doExportToImage(), @ref doExportToCompressedImage(), @ref doExportToData()
or @ref doExportToFile() functions based on what features are supported.
You don't need to do most of the redundant sanity checks, these things are
checked by the implementation:
- The function @ref doExportToImage() is called only if
@ref ImageConverterFeature::ConvertImage is supported.
- The function @ref doExportToCompressedImage() is called only if
@ref ImageConverterFeature::ConvertCompressedImage is supported.
- The function @ref doExportToData(const ImageView2D&) is called only if
@ref ImageConverterFeature::ConvertData is supported.
- The function @ref doExportToData(const CompressedImageView2D&) is called
only if @ref ImageConverterFeature::ConvertCompressedData is supported.
- The function @ref doExportToFile(const ImageView2D&, const std::string&) is
called only if @ref ImageConverterFeature::ConvertFile is supported.
- The function @ref doExportToFile(const CompressedImageView2D&, const std::string&)
is called only if @ref ImageConverterFeature::ConvertCompressedFile is
supported.
@m_class{m-block m-warning}
@par Dangling function pointers on plugin unload
As @ref Trade-AbstractImageConverter-data-dependency "mentioned above",
@ref Corrade::Containers::Array instances returned from plugin
implementations are not allowed to use anything else than the default
deleter, otherwise this could cause dangling function pointer call on array
destruction if the plugin gets unloaded before the array is destroyed. This
is asserted by the base implementation on return.
*/
class MAGNUM_TRADE_EXPORT AbstractImageConverter: public PluginManager::AbstractManagingPlugin<AbstractImageConverter> {
public:
#ifdef MAGNUM_BUILD_DEPRECATED
/** @brief @copybrief ImageConverterFeature
* @m_deprecated_since{2020,06} Use @ref ImageConverterFeature instead.
*/
typedef CORRADE_DEPRECATED("use ImageConverterFeature instead") ImageConverterFeature Feature;
/** @brief @copybrief ImageConverterFeatures
* @m_deprecated_since{2020,06} Use @ref ImageConverterFeatures instead.
*/
typedef CORRADE_DEPRECATED("use ImageConverterFeatures instead") ImageConverterFeatures Features;
#endif
/**
* @brief Plugin interface
*
* @snippet Magnum/Trade/AbstractImageConverter.cpp interface
*/
static std::string pluginInterface();
#ifndef CORRADE_PLUGINMANAGER_NO_DYNAMIC_PLUGIN_SUPPORT
/**
* @brief Plugin search paths
*
* Looks into `magnum/imageconverters/` or `magnum-d/imageconverters/`
* next to the dynamic @ref Trade library, next to the executable and
* elsewhere according to the rules documented in
* @ref Corrade::PluginManager::implicitPluginSearchPaths(). The search
* directory can be also hardcoded using the `MAGNUM_PLUGINS_DIR` CMake
* variables, see @ref building for more information.
*
* Not defined on platforms without
* @ref CORRADE_PLUGINMANAGER_NO_DYNAMIC_PLUGIN_SUPPORT "dynamic plugin support".
*/
static std::vector<std::string> pluginSearchPaths();
#endif
/** @brief Default constructor */
explicit AbstractImageConverter();
/** @brief Constructor with access to plugin manager */
explicit AbstractImageConverter(PluginManager::Manager<AbstractImageConverter>& manager);
/** @brief Plugin manager constructor */
explicit AbstractImageConverter(PluginManager::AbstractManager& manager, const std::string& plugin);
/** @brief Features supported by this converter */
ImageConverterFeatures features() const { return doFeatures(); }
/**
* @brief Converter flags
* @m_since{2020,06}
*/
ImageConverterFlags flags() const { return _flags; }
/**
* @brief Set converter flags
* @m_since{2020,06}
*
* Some flags can be set only if the converter supports particular
* features, see documentation of each @ref ImageConverterFlag for more
* information. By default no flags are set.
*/
void setFlags(ImageConverterFlags flags);
/**
* @brief Convert image to different format
*
* Available only if @ref ImageConverterFeature::ConvertImage is
* supported. Returns converted image on success,
* @ref Containers::NullOpt otherwise.
* @see @ref features(), @ref exportToData(), @ref exportToFile()
*/
Containers::Optional<Image2D> exportToImage(const ImageView2D& image);
/**
* @brief Convert image to compressed format
*
* Available only if @ref ImageConverterFeature::ConvertCompressedImage
* is supported. Returns converted image on success,
* @ref Containers::NullOpt otherwise.
* @see @ref features(), @ref exportToData(), @ref exportToFile()
*/
Containers::Optional<CompressedImage2D> exportToCompressedImage(const ImageView2D& image);
/**
* @brief Export image to raw data
*
* Available only if @ref ImageConverterFeature::ConvertData is
* supported. Returns data on success, zero-sized array otherwise.
* @see @ref features(), @ref exportToImage(),
* @ref exportToFile(const ImageView2D&, const std::string&)
*/
Containers::Array<char> exportToData(const ImageView2D& image);
/**
* @brief Export compressed image to raw data
*
* Available only if @ref ImageConverterFeature::ConvertCompressedData
* is supported. Returns data on success, zero-sized array otherwise.
* @see @ref features(), @ref exportToCompressedImage(),
* @ref exportToFile(const CompressedImageView2D&, const std::string&)
*/
Containers::Array<char> exportToData(const CompressedImageView2D& image);
/**
* @brief Export image to raw data
*
* Based on whether the image is compressed or not, calls either
* @ref exportToData(const ImageView2D&) or
* @ref exportToData(const CompressedImageView2D&). See documentation
* of those two functions for details.
*/
Containers::Array<char> exportToData(const ImageData2D& image);
/**
* @brief Export image to file
*
* Available only if @ref ImageConverterFeature::ConvertFile or
* @ref ImageConverterFeature::ConvertData is supported. Returns
* @cpp true @ce on success, @cpp false @ce otherwise.
* @see @ref features(), @ref exportToImage(),
* @ref exportToData(const ImageView2D&)
*/
bool exportToFile(const ImageView2D& image, const std::string& filename);
/**
* @brief Export compressed image to file
*
* Available only if @ref ImageConverterFeature::ConvertCompressedFile
* or @ref ImageConverterFeature::ConvertCompressedData is supported.
* Returns @cpp true @ce on success, @cpp false @ce otherwise.
* @see @ref features(), @ref exportToCompressedImage(),
* @ref exportToData(const CompressedImageView2D&)
*/
bool exportToFile(const CompressedImageView2D& image, const std::string& filename);
/**
* @brief Export image to file
*
* Based on whether the image is compressed or not, calls either
* @ref exportToFile(const ImageView2D&, const std::string&) or
* @ref exportToFile(const CompressedImageView2D&, const std::string&).
* See documentation of those two functions for details.
*/
bool exportToFile(const ImageData2D& image, const std::string& filename);
private:
/** @brief Implementation for @ref features() */
virtual ImageConverterFeatures doFeatures() const = 0;
/**
* @brief Implementation for @ref setFlags()
*
* Useful when the converter needs to modify some internal state on
* flag setup. Default implementation does nothing and this
* function doesn't need to be implemented --- the flags are available
* through @ref flags().
*
* To reduce the amount of error checking on user side, this function
* isn't expected to fail --- if a flag combination is invalid /
* unsuported, error reporting should be delayed to various conversion
* functions, where the user is expected to do error handling anyway.
*/
virtual void doSetFlags(ImageConverterFlags flags);
/** @brief Implementation for @ref exportToImage() */
virtual Containers::Optional<Image2D> doExportToImage(const ImageView2D& image);
/** @brief Implementation for @ref exportToCompressedImage() */
virtual Containers::Optional<CompressedImage2D> doExportToCompressedImage(const ImageView2D& image);
/** @brief Implementation for @ref exportToData(const ImageView2D&) */
virtual Containers::Array<char> doExportToData(const ImageView2D& image);
/** @brief Implementation for @ref exportToData(const CompressedImageView2D&) */
virtual Containers::Array<char> doExportToData(const CompressedImageView2D& image);
/**
* @brief Implementation for @ref exportToFile(const ImageView2D&, const std::string&)
*
* If @ref ImageConverterFeature::ConvertData is supported, default
* implementation calls @ref doExportToData(const ImageView2D&) and
* saves the result to given file.
*/
virtual bool doExportToFile(const ImageView2D& image, const std::string& filename);
/**
* @brief Implementation for @ref exportToFile(const CompressedImageView2D&, const std::string&)
*
* If @ref ImageConverterFeature::ConvertCompressedData is supported,
* default implementation calls @ref doExportToData(const CompressedImageView2D&)
* and saves the result to given file.
*/
virtual bool doExportToFile(const CompressedImageView2D& image, const std::string& filename);
ImageConverterFlags _flags;
};
}}
#endif