|
|
|
|
/*
|
|
|
|
|
This file is part of Magnum.
|
|
|
|
|
|
|
|
|
|
Copyright © 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019
|
|
|
|
|
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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <Corrade/Containers/Optional.h>
|
|
|
|
|
#include <Corrade/Containers/StaticArray.h>
|
|
|
|
|
#include <Corrade/PluginManager/Manager.h>
|
|
|
|
|
#include <Corrade/Utility/Arguments.h>
|
|
|
|
|
#include <Corrade/Utility/ConfigurationGroup.h>
|
|
|
|
|
#include <Corrade/Utility/DebugStl.h>
|
|
|
|
|
#include <Corrade/Utility/Directory.h>
|
|
|
|
|
#include <Corrade/Utility/String.h>
|
|
|
|
|
|
|
|
|
|
#include "Magnum/PixelFormat.h"
|
|
|
|
|
#include "Magnum/Trade/AbstractImporter.h"
|
|
|
|
|
#include "Magnum/Trade/AbstractImageConverter.h"
|
|
|
|
|
#include "Magnum/Trade/ImageData.h"
|
|
|
|
|
|
|
|
|
|
namespace Magnum {
|
|
|
|
|
|
|
|
|
|
/** @page magnum-imageconverter Image conversion utility
|
|
|
|
|
@brief Converts images of different formats
|
|
|
|
|
|
|
|
|
|
@m_footernavigation
|
|
|
|
|
@m_keywords{magnum-imageconverter imageconverter}
|
|
|
|
|
|
|
|
|
|
This utility is built if both `WITH_TRADE` and `WITH_IMAGECONVERTER` is enabled
|
|
|
|
|
when building Magnum. To use this utility with CMake, you need to request the
|
|
|
|
|
`imageconverter` component of the `Magnum` package and use the
|
|
|
|
|
`Magnum::imageconverter` target for example in a custom command:
|
|
|
|
|
|
|
|
|
|
@code{.cmake}
|
|
|
|
|
find_package(Magnum REQUIRED imageconverter)
|
|
|
|
|
|
|
|
|
|
add_custom_command(OUTPUT ... COMMAND Magnum::imageconverter ...)
|
|
|
|
|
@endcode
|
|
|
|
|
|
|
|
|
|
See @ref building, @ref cmake and the @ref Trade namespace for more
|
|
|
|
|
information.
|
|
|
|
|
|
|
|
|
|
@section magnum-imageconverter-usage Usage
|
|
|
|
|
|
|
|
|
|
@code{.sh}
|
|
|
|
|
magnum-imageconverter [-h|--help] [--importer IMPORTER] [--converter CONVERTER]
|
|
|
|
|
[--plugin-dir DIR] [--] input output
|
|
|
|
|
@endcode
|
|
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
|
|
- `input` --- input image
|
|
|
|
|
- `output` --- output image
|
|
|
|
|
- `-h`, `--help` --- display this help message and exit
|
|
|
|
|
- `--importer IMPORTER` --- image importer plugin (default:
|
|
|
|
|
@ref Trade::AnyImageImporter "AnyImageImporter")
|
|
|
|
|
- `--converter CONVERTER` --- image converter plugin (default:
|
|
|
|
|
@ref Trade::AnyImageConverter "AnyImageConverter")
|
|
|
|
|
- `--plugin-dir DIR` --- override base plugin dir
|
|
|
|
|
|
|
|
|
|
@section magnum-imageconverter-example Example usage
|
|
|
|
|
|
|
|
|
|
Converting a JPEG file to a PNG:
|
|
|
|
|
|
|
|
|
|
@code{.sh}
|
|
|
|
|
magnum-imageconverter image.jpg image.png
|
|
|
|
|
@endcode
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using namespace Magnum;
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
void setOptions(PluginManager::AbstractPlugin& plugin, const std::string& options) {
|
|
|
|
|
for(const std::string& option: Utility::String::splitWithoutEmptyParts(options, ',')) {
|
|
|
|
|
auto keyValue = Utility::String::partition(option, '=');
|
|
|
|
|
Utility::String::trimInPlace(keyValue[0]);
|
|
|
|
|
Utility::String::trimInPlace(keyValue[2]);
|
|
|
|
|
|
|
|
|
|
/* Provide a warning message in case the plugin doesn't define given
|
|
|
|
|
option in its default config. The plugin is not *required* to have
|
|
|
|
|
those tho (could be backward compatibility entries, for example), so
|
|
|
|
|
not an error. */
|
|
|
|
|
if(!plugin.configuration().valueCount(keyValue[0]))
|
|
|
|
|
Warning{} << "Option" << keyValue[0] << "not recognized by" << plugin.plugin();
|
|
|
|
|
|
|
|
|
|
/* If the option doesn't have an =, treat it as a boolean flag that's
|
|
|
|
|
set to true. While there's no similar way to do an inverse, it's
|
|
|
|
|
still nicer than causing a fatal error with those. */
|
|
|
|
|
if(keyValue[1].empty())
|
|
|
|
|
plugin.configuration().setValue(keyValue[0], true);
|
|
|
|
|
else
|
|
|
|
|
plugin.configuration().setValue(keyValue[0], keyValue[2]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int main(int argc, char** argv) {
|
|
|
|
|
Utility::Arguments args;
|
|
|
|
|
args.addArgument("input").setHelp("input", "input image")
|
|
|
|
|
.addArgument("output").setHelp("output", "output image")
|
|
|
|
|
.addOption("importer", "AnyImageImporter").setHelp("importer", "image importer plugin")
|
|
|
|
|
.addOption("converter", "AnyImageConverter").setHelp("converter", "image converter plugin")
|
|
|
|
|
.addOption("plugin-dir").setHelp("plugin-dir", "override base plugin dir", "DIR")
|
|
|
|
|
.addOption('i', "importer-options").setHelp("importer-options", "configuration options to pass to the importer", "key=val,key2=val2,…")
|
|
|
|
|
.addOption('c', "converter-options").setHelp("converter-options", "configuration options to pass to the converter", "key=val,key2=val2,…")
|
|
|
|
|
.setGlobalHelp(R"(Converts images of different formats.
|
|
|
|
|
|
|
|
|
|
The -i / --importer-options and -c / --converter-options arguments accept a
|
|
|
|
|
comma-separated list of key/value pairs to set in the importer / converter
|
|
|
|
|
plugin configuration. If the = character is omitted, it's equivalent to saying
|
|
|
|
|
key=true.)")
|
|
|
|
|
.parse(argc, argv);
|
|
|
|
|
|
|
|
|
|
/* Load importer plugin */
|
|
|
|
|
PluginManager::Manager<Trade::AbstractImporter> importerManager{
|
|
|
|
|
args.value("plugin-dir").empty() ? std::string{} :
|
|
|
|
|
Utility::Directory::join(args.value("plugin-dir"), Trade::AbstractImporter::pluginSearchPaths()[0])};
|
|
|
|
|
Containers::Pointer<Trade::AbstractImporter> importer = importerManager.loadAndInstantiate(args.value("importer"));
|
|
|
|
|
if(!importer) return 1;
|
|
|
|
|
|
|
|
|
|
/* Load converter plugin */
|
|
|
|
|
PluginManager::Manager<Trade::AbstractImageConverter> converterManager{
|
|
|
|
|
args.value("plugin-dir").empty() ? std::string{} :
|
|
|
|
|
Utility::Directory::join(args.value("plugin-dir"), Trade::AbstractImageConverter::pluginSearchPaths()[0])};
|
|
|
|
|
Containers::Pointer<Trade::AbstractImageConverter> converter = converterManager.loadAndInstantiate(args.value("converter"));
|
|
|
|
|
if(!converter) return 2;
|
|
|
|
|
|
|
|
|
|
/* Set options, if passed */
|
|
|
|
|
setOptions(*importer, args.value("importer-options"));
|
|
|
|
|
setOptions(*converter, args.value("converter-options"));
|
|
|
|
|
|
|
|
|
|
/* Open input file */
|
Converting APIs returning std::optional to Containers::Optional.
And removing the bundled std::optional implementation. This finally
makes this library compatible with C++17. Since this would be a huge
backwards-incompatible change that would make everyone angry, the
following had to be done in case both CORRADE_BUILD_DEPRECATED and
MAGNUM_BUILD_DEPRECATED is defined:
* Under C++11 and C++14, Containers::Optional / Containers::NullOpt is
aliased to std::optional / std::nullopt. This is no worse than the
state before, when we also provided these symbols.
* Under C++17, where standard <optional> header is available,
Containers::Optional provides implicit conversion to it. Only one-way
conversion is supported, as there was fortunately no Magnum API that
took std::optional via parameter, and there might be some corner
cases that this doesn't cover. The goal is to have all examples
compiling with the old API, at least.
* There's a new test especially for this, which checks that both the
C++11 and C++17 ways of doing things work as they should.
The typedef and conversion is marked as deprecated, so it will spit out
many warnings to push users to upgrade. I hope I can completely remove
this mess soon :/
8 years ago
|
|
|
Containers::Optional<Trade::ImageData2D> image;
|
|
|
|
|
if(!importer->openFile(args.value("input")) || !(image = importer->image2D(0))) {
|
|
|
|
|
Error() << "Cannot open file" << args.value("input");
|
|
|
|
|
return 3;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Debug() << "Converting image of size" << image->size() << "and format" << image->format() << "to" << args.value("output");
|
|
|
|
|
|
|
|
|
|
/* Save output file */
|
|
|
|
|
if(!converter->exportToFile(*image, args.value("output"))) {
|
|
|
|
|
Error() << "Cannot save file" << args.value("output");
|
|
|
|
|
return 4;
|
|
|
|
|
}
|
|
|
|
|
}
|