Skip to content

Commit

Permalink
Save image to buffer (#1105)
Browse files Browse the repository at this point in the history
  • Loading branch information
Meakk authored Dec 19, 2023
1 parent 8aee29d commit 8f6209e
Show file tree
Hide file tree
Showing 6 changed files with 93 additions and 2 deletions.
1 change: 1 addition & 0 deletions doc/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ For F3D users:

For libf3d users:
- Added `scene.animation.autoplay` option.
- Added a `f3d::image::saveBuffer` API to save an image as a file format in memory.

## v2.2.1

Expand Down
9 changes: 9 additions & 0 deletions library/public/image.h
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
#include "export.h"

#include <string>
#include <vector>

namespace f3d
{
Expand Down Expand Up @@ -153,6 +154,14 @@ class F3D_EXPORT image
*/
void save(const std::string& path, SaveFormat format = SaveFormat::PNG) const;

/**
* Save an image to a memory buffer in the specified format.
* Default format is PNG if not specified.
* TIF format is not supported yet.
* Throw an image::write_exception if the type is TIF.
*/
std::vector<unsigned char> saveBuffer(SaveFormat format = SaveFormat::PNG) const;

/**
* An exception that can be thrown by the image when there.
* is an error on write.
Expand Down
35 changes: 34 additions & 1 deletion library/src/image.cxx
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
#include "init.h"

#include <vtkBMPWriter.h>
#include <vtkDataArrayRange.h>
#include <vtkImageData.h>
#include <vtkImageDifference.h>
#include <vtkImageReader2.h>
Expand All @@ -13,17 +14,33 @@
#include <vtkPointData.h>
#include <vtkSmartPointer.h>
#include <vtkTIFFWriter.h>
#include <vtkUnsignedCharArray.h>
#include <vtksys/SystemTools.hxx>

#include <cassert>
#include <vector>

namespace f3d
{
class image::internals
{
public:
vtkSmartPointer<vtkImageData> Image;

template<typename WriterType>
std::vector<unsigned char> SaveBuffer()
{
vtkNew<WriterType> writer;
writer->WriteToMemoryOn();
writer->SetInputData(this->Image);
writer->Write();

std::vector<unsigned char> result;

auto valRange = vtk::DataArrayValueRange(writer->GetResult());
std::copy(valRange.begin(), valRange.end(), std::back_inserter(result));

return result;
}
};

//----------------------------------------------------------------------------
Expand Down Expand Up @@ -290,6 +307,22 @@ void image::save(const std::string& path, SaveFormat format) const
}
}

//----------------------------------------------------------------------------
std::vector<unsigned char> image::saveBuffer(SaveFormat format) const
{
switch (format)
{
case SaveFormat::PNG:
return this->Internals->SaveBuffer<vtkPNGWriter>();
case SaveFormat::JPG:
return this->Internals->SaveBuffer<vtkJPEGWriter>();
case SaveFormat::BMP:
return this->Internals->SaveBuffer<vtkBMPWriter>();
default:
throw write_exception("Cannot save to buffer in the specified format");
}
}

//----------------------------------------------------------------------------
image::write_exception::write_exception(const std::string& what)
: exception(what)
Expand Down
32 changes: 32 additions & 0 deletions library/testing/TestSDKImage.cxx
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,38 @@ int TestSDKImage(int argc, char* argv[])
generated.save(std::string(argv[2]) + "TestSDKImage.tif", f3d::image::SaveFormat::TIF);
generated.save(std::string(argv[2]) + "TestSDKImage.bmp", f3d::image::SaveFormat::BMP);

// test saveBuffer in different formats
std::vector<unsigned char> bufferPNG = generated.saveBuffer();
if (bufferPNG.size() == 0)
{
std::cerr << "PNG buffer empty" << std::endl;
return EXIT_FAILURE;
}

std::vector<unsigned char> bufferJPG = generated.saveBuffer(f3d::image::SaveFormat::JPG);
if (bufferJPG.size() == 0)
{
std::cerr << "JPG buffer empty" << std::endl;
return EXIT_FAILURE;
}

try
{
generated.saveBuffer(f3d::image::SaveFormat::TIF);
std::cerr << "An exception has not been thrown when saving buffer to TIF format" << std::endl;
return EXIT_FAILURE;
}
catch (const f3d::image::write_exception&)
{
}

std::vector<unsigned char> bufferBMP = generated.saveBuffer(f3d::image::SaveFormat::BMP);
if (bufferBMP.size() == 0)
{
std::cerr << "BMP buffer empty" << std::endl;
return EXIT_FAILURE;
}

// test constructor with different channel sizes
f3d::image img16(width, height, channels, f3d::image::ChannelType::SHORT);
f3d::image img32(width, height, channels, f3d::image::ChannelType::FLOAT);
Expand Down
11 changes: 10 additions & 1 deletion python/F3DPythonBindings.cxx
Original file line number Diff line number Diff line change
Expand Up @@ -102,6 +102,12 @@ PYBIND11_MODULE(pyf3d, module)
return py::bytes(static_cast<char*>(img.getContent()), expectedSize);
};

auto getFileBytes = [](const f3d::image& img, f3d::image::SaveFormat format)
{
std::vector<unsigned char> result = img.saveBuffer(format);
return py::bytes(reinterpret_cast<char*>(result.data()), result.size());
};

image //
.def(py::init<>())
.def(py::init<const std::string&>())
Expand All @@ -116,7 +122,10 @@ PYBIND11_MODULE(pyf3d, module)
.def_property("content", getImageBytes, setImageBytes)
.def("compare", &f3d::image::compare)
.def(
"save", &f3d::image::save, py::arg("path"), py::arg("format") = f3d::image::SaveFormat::PNG);
"save", &f3d::image::save, py::arg("path"), py::arg("format") = f3d::image::SaveFormat::PNG)
.def("save_buffer", getFileBytes, py::arg("format") = f3d::image::SaveFormat::PNG)
.def("_repr_png_",
[&](const f3d::image& img) { return getFileBytes(img, f3d::image::SaveFormat::PNG); });

// f3d::options
py::class_<f3d::options> options(module, "Options");
Expand Down
7 changes: 7 additions & 0 deletions python/testing/test_image.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,3 +56,10 @@ def test_save(f3d_engine):

img.save(fn, f3d.Image.SaveFormat.BMP)
assert os.path.isfile(fn)


def test_save_buffer(f3d_engine):
img = f3d_engine.window.render_to_image(True)
buffer = img.save_buffer(f3d.Image.SaveFormat.PNG)
assert buffer.startswith(b"\x89PNG")
assert img._repr_png_() == buffer

0 comments on commit 8f6209e

Please sign in to comment.