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.
422 lines
18 KiB
422 lines
18 KiB
/* |
|
This file is part of Magnum. |
|
|
|
Copyright © 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019, |
|
2020, 2021, 2022 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 <pybind11/operators.h> |
|
#include <pybind11/pybind11.h> |
|
#include <Corrade/Containers/StridedArrayView.h> |
|
#include <Magnum/Image.h> |
|
#include <Magnum/ImageView.h> |
|
#include <Magnum/Mesh.h> |
|
#include <Magnum/PixelFormat.h> |
|
#include <Magnum/PixelStorage.h> |
|
#include <Magnum/Sampler.h> |
|
|
|
#include "Corrade/PythonBindings.h" |
|
#include "Corrade/Containers/PythonBindings.h" |
|
#include "Corrade/Containers/StridedArrayViewPythonBindings.h" |
|
#include "Magnum/PythonBindings.h" |
|
|
|
#include "magnum/bootstrap.h" |
|
|
|
#ifdef MAGNUM_BUILD_STATIC |
|
#include "magnum/staticconfigure.h" |
|
#endif |
|
|
|
namespace py = pybind11; |
|
|
|
namespace magnum { namespace { |
|
|
|
template<class T> void image(py::class_<T>& c) { |
|
c |
|
/* Constructors. Only the ones taking the generic format and *not* |
|
taking an Array, as Python has no way to "move" it in */ |
|
.def(py::init<const PixelStorage&, PixelFormat>(), "Construct an image placeholder") |
|
.def(py::init<PixelFormat>(), "Construct an image placeholder") |
|
|
|
/* Properties */ |
|
.def_property_readonly("storage", &T::storage, "Storage of pixel data") |
|
.def_property_readonly("format", &T::format, "Format of pixel data") |
|
/** @todo formatExtra() */ |
|
.def_property_readonly("pixel_size", &T::pixelSize, "Pixel size (in bytes)") |
|
.def_property_readonly("size", [](T& self) { |
|
return PyDimensionTraits<T::Dimensions, Int>::from(self.size()); |
|
}, "Image size") |
|
.def_property_readonly("data", [](T& self) { |
|
return Containers::pyArrayViewHolder(self.data(), self.data() ? py::cast(self) : py::none{}); |
|
}, "Image data") |
|
.def_property_readonly("pixels", [](T& self) { |
|
return Containers::pyArrayViewHolder(Containers::PyStridedArrayView<T::Dimensions + 1, const char>{self.pixels()}, self.data() ? py::cast(self) : py::none{}); |
|
}, "View on pixel data"); |
|
} |
|
|
|
template<class T> void imageView(py::class_<T, PyImageViewHolder<T>>& c) { |
|
/* |
|
Missing APIs: |
|
|
|
Type, ErasedType, Dimensions |
|
*/ |
|
|
|
py::implicitly_convertible<Image<T::Dimensions>, T>(); |
|
|
|
c |
|
/* Constructors. The variants *not* taking an array view have to be |
|
first, otherwise things fail on systems that don't have numpy |
|
installed: |
|
|
|
=================================================================== |
|
ERROR: test_init_empty (test.test.ImageView) |
|
------------------------------------------------------------------- |
|
Traceback (most recent call last): |
|
File ".../magnum/test/test.py", line 102, in test_init_empty |
|
b = ImageView2D(storage, PixelFormat.R32F, (8, 8)) |
|
ModuleNotFoundError: No module named 'numpy' |
|
|
|
This is because of the order in which pybind processes arguments --- |
|
it would first try to match the (PixelFormat, Vector2i, ArrayView) |
|
variant and *somehow* getting all the way to the third argument, |
|
where, because ArrayView is marked as implicitly convertible from |
|
py::array for numpy compatibility, it ends up doing this in numpy.h: |
|
|
|
module m = module::import("numpy.core.multiarray"); |
|
auto c = m.attr("_ARRAY_API"); |
|
|
|
Wonderful, isn't it. */ |
|
.def(py::init([](const PixelStorage& storage, PixelFormat format, const typename PyDimensionTraits<T::Dimensions, Int>::VectorType& size) { |
|
return T{storage, format, size}; |
|
}), "Construct an empty view") |
|
.def(py::init([](PixelFormat format, const typename PyDimensionTraits<T::Dimensions, Int>::VectorType& size) { |
|
return T{format, size}; |
|
}), "Construct an empty view") |
|
.def(py::init([](const PixelStorage& storage, PixelFormat format, const typename PyDimensionTraits<T::Dimensions, Int>::VectorType& size, const Containers::ArrayView<typename T::Type>& data) { |
|
return pyImageViewHolder(T{storage, format, size, data}, pyObjectHolderFor<Containers::PyArrayViewHolder>(data).owner); |
|
}), "Constructor") |
|
.def(py::init([](PixelFormat format, const typename PyDimensionTraits<T::Dimensions, Int>::VectorType& size, const Containers::ArrayView<typename T::Type>& data) { |
|
return pyImageViewHolder(T{format, size, data}, pyObjectHolderFor<Containers::PyArrayViewHolder>(data).owner); |
|
}), "Constructor") |
|
.def(py::init([](Image<T::Dimensions>& image) { |
|
return pyImageViewHolder(T{image}, image.data() ? py::cast(image) : py::none{}); |
|
}), "Construct a view on an image") |
|
.def(py::init([](const ImageView<T::Dimensions, typename T::Type>& other) { |
|
return pyImageViewHolder(ImageView<T::Dimensions, typename T::Type>(other), pyObjectHolderFor<PyImageViewHolder>(other).owner); |
|
}), "Construct from any type convertible to an image view") |
|
|
|
/* Properties */ |
|
.def_property_readonly("storage", &T::storage, "Storage of pixel data") |
|
.def_property_readonly("format", &T::format, "Format of pixel data") |
|
.def_property_readonly("pixel_size", &T::pixelSize, "Pixel size (in bytes)") |
|
.def_property_readonly("size", [](T& self) { |
|
return PyDimensionTraits<T::Dimensions, Int>::from(self.size()); |
|
}, "Image size") |
|
.def_property("data", [](T& self) { |
|
return Containers::pyArrayViewHolder(self.data(), pyObjectHolderFor<PyImageViewHolder>(self).owner); |
|
}, [](T& self, const Containers::ArrayView<typename T::Type>& data) { |
|
self.setData(data); |
|
pyObjectHolderFor<PyImageViewHolder>(self).owner = |
|
pyObjectHolderFor<Containers::PyArrayViewHolder>(data).owner; |
|
}, "Image data") |
|
.def_property_readonly("pixels", [](T& self) { |
|
return Containers::pyArrayViewHolder(Containers::PyStridedArrayView<T::Dimensions + 1, typename T::Type>{self.pixels()}, pyObjectHolderFor<PyImageViewHolder>(self).owner); |
|
}, "View on pixel data") |
|
|
|
.def_property_readonly("owner", [](T& self) { |
|
return pyObjectHolderFor<PyImageViewHolder>(self).owner; |
|
}, "Memory owner"); |
|
} |
|
|
|
template<class T> void imageViewFromMutable(py::class_<T, PyImageViewHolder<T>>& c) { |
|
py::implicitly_convertible<BasicMutableImageView<T::Dimensions>, T>(); |
|
|
|
c |
|
.def(py::init([](const BasicMutableImageView<T::Dimensions>& other) { |
|
return pyImageViewHolder(BasicImageView<T::Dimensions>(other), pyObjectHolderFor<PyImageViewHolder>(other).owner); |
|
}), "Construct from a mutable view"); |
|
} |
|
|
|
void magnum(py::module_& m) { |
|
m.attr("BUILD_STATIC") = |
|
#ifdef MAGNUM_BUILD_STATIC |
|
true |
|
#else |
|
false |
|
#endif |
|
; |
|
m.attr("TARGET_GL") = |
|
#ifdef MAGNUM_TARGET_GL |
|
true |
|
#else |
|
false |
|
#endif |
|
; |
|
m.attr("TARGET_GLES") = |
|
#ifdef MAGNUM_TARGET_GLES |
|
true |
|
#else |
|
false |
|
#endif |
|
; |
|
m.attr("TARGET_GLES2") = |
|
#ifdef MAGNUM_TARGET_GLES2 |
|
true |
|
#else |
|
false |
|
#endif |
|
; |
|
/** @todo do we need TARGET_GLES3? i hope not */ |
|
m.attr("TARGET_WEBGL") = |
|
#ifdef MAGNUM_TARGET_WEBGL |
|
true |
|
#else |
|
false |
|
#endif |
|
; |
|
/* TARGET_DESKTOP_GLES, TARGET_HEADLESS skipped as they make sense only |
|
on native side (affecting what the builtin utilities use), not really in |
|
Python */ |
|
m.attr("TARGET_VK") = |
|
#ifdef MAGNUM_TARGET_VK |
|
true |
|
#else |
|
false |
|
#endif |
|
; |
|
|
|
py::enum_<MeshPrimitive>{m, "MeshPrimitive", "Mesh primitive type"} |
|
.value("POINTS", MeshPrimitive::Points) |
|
.value("LINES", MeshPrimitive::Lines) |
|
.value("LINE_LOOP", MeshPrimitive::LineLoop) |
|
.value("LINE_STRIP", MeshPrimitive::LineStrip) |
|
.value("TRIANGLES", MeshPrimitive::Triangles) |
|
.value("TRIANGLE_STRIP", MeshPrimitive::TriangleStrip) |
|
.value("TRIANGLE_FAN", MeshPrimitive::TriangleFan); |
|
|
|
py::enum_<MeshIndexType>{m, "MeshIndexType", "Mesh index type"} |
|
.value("UNSIGNED_BYTE", MeshIndexType::UnsignedByte) |
|
.value("UNSIGNED_SHORT", MeshIndexType::UnsignedShort) |
|
.value("UNSIGNED_INT", MeshIndexType::UnsignedInt); |
|
|
|
py::enum_<PixelFormat>{m, "PixelFormat", "Format of pixel data"} |
|
.value("R8_UNORM", PixelFormat::R8Unorm) |
|
.value("RG8_UNORM", PixelFormat::RG8Unorm) |
|
.value("RGB8_UNORM", PixelFormat::RGB8Unorm) |
|
.value("RGBA8_UNORM", PixelFormat::RGBA8Unorm) |
|
.value("R8_SNORM", PixelFormat::R8Snorm) |
|
.value("RG8_SNORM", PixelFormat::RG8Snorm) |
|
.value("RGB8_SNORM", PixelFormat::RGB8Snorm) |
|
.value("RGBA8_SNORM", PixelFormat::RGBA8Snorm) |
|
.value("R8_SRGB", PixelFormat::R8Srgb) |
|
.value("RG8_SRGB", PixelFormat::RG8Srgb) |
|
.value("RGB8_SRGB", PixelFormat::RGB8Srgb) |
|
.value("RGBA8_SRGB", PixelFormat::RGBA8Srgb) |
|
.value("R8UI", PixelFormat::R8UI) |
|
.value("RG8UI", PixelFormat::RG8UI) |
|
.value("RGB8UI", PixelFormat::RGB8UI) |
|
.value("RGBA8UI", PixelFormat::RGBA8UI) |
|
.value("R8I", PixelFormat::R8I) |
|
.value("RG8I", PixelFormat::RG8I) |
|
.value("RGB8I", PixelFormat::RGB8I) |
|
.value("RGBA8I", PixelFormat::RGBA8I) |
|
.value("R16_UNORM", PixelFormat::R16Unorm) |
|
.value("RG16_UNORM", PixelFormat::RG16Unorm) |
|
.value("RGB16_UNORM", PixelFormat::RGB16Unorm) |
|
.value("RGBA16_UNORM", PixelFormat::RGBA16Unorm) |
|
.value("R16_SNORM", PixelFormat::R16Snorm) |
|
.value("RG16_SNORM", PixelFormat::RG16Snorm) |
|
.value("RGB16_SNORM", PixelFormat::RGB16Snorm) |
|
.value("RGBA16_SNORM", PixelFormat::RGBA16Snorm) |
|
.value("R16UI", PixelFormat::R16UI) |
|
.value("RG16UI", PixelFormat::RG16UI) |
|
.value("RGB16UI", PixelFormat::RGB16UI) |
|
.value("RGBA16UI", PixelFormat::RGBA16UI) |
|
.value("R16I", PixelFormat::R16I) |
|
.value("RG16I", PixelFormat::RG16I) |
|
.value("RGB16I", PixelFormat::RGB16I) |
|
.value("RGBA16I", PixelFormat::RGBA16I) |
|
.value("R32UI", PixelFormat::R32UI) |
|
.value("RG32UI", PixelFormat::RG32UI) |
|
.value("RGB32UI", PixelFormat::RGB32UI) |
|
.value("RGBA32UI", PixelFormat::RGBA32UI) |
|
.value("R32I", PixelFormat::R32I) |
|
.value("RG32I", PixelFormat::RG32I) |
|
.value("RGB32I", PixelFormat::RGB32I) |
|
.value("RGBA32I", PixelFormat::RGBA32I) |
|
.value("R16F", PixelFormat::R16F) |
|
.value("RG16F", PixelFormat::RG16F) |
|
.value("RGB16F", PixelFormat::RGB16F) |
|
.value("RGBA16F", PixelFormat::RGBA16F) |
|
.value("R32F", PixelFormat::R32F) |
|
.value("RG32F", PixelFormat::RG32F) |
|
.value("RGB32F", PixelFormat::RGB32F) |
|
.value("RGBA32F", PixelFormat::RGBA32F) |
|
.value("DEPTH16_UNORM", PixelFormat::Depth16Unorm) |
|
.value("DEPTH24_UNORM", PixelFormat::Depth24Unorm) |
|
.value("DEPTH32F", PixelFormat::Depth32F) |
|
.value("STENCIL8UI", PixelFormat::Stencil8UI) |
|
.value("DEPTH16_UNORM_STENCIL8UI", PixelFormat::Depth16UnormStencil8UI) |
|
.value("DEPTH24_UNORM_STENCIL8UI", PixelFormat::Depth24UnormStencil8UI) |
|
.value("DEPTH32F_STENCIL8UI", PixelFormat::Depth32FStencil8UI); |
|
|
|
py::class_<PixelStorage>{m, "PixelStorage", "Pixel storage parameters"} |
|
.def(py::init(), "Default constructor") |
|
|
|
/* Comparison */ |
|
.def(py::self == py::self, "Equality comparison") |
|
.def(py::self != py::self, "Non-equality comparison") |
|
|
|
/* Properties */ |
|
.def_property("alignment", |
|
&PixelStorage::alignment, &PixelStorage::setAlignment, "Row alignment") |
|
.def_property("row_length", |
|
&PixelStorage::rowLength, &PixelStorage::setRowLength, "Row length") |
|
.def_property("image_height", |
|
&PixelStorage::imageHeight, &PixelStorage::setImageHeight, "Image height") |
|
.def_property("skip", |
|
&PixelStorage::skip, &PixelStorage::setSkip, "Pixel, row and image skip"); |
|
|
|
py::class_<Image1D> image1D{m, "Image1D", "One-dimensional image"}; |
|
py::class_<Image2D> image2D{m, "Image2D", "Two-dimensional image"}; |
|
py::class_<Image3D> image3D{m, "Image3D", "Three-dimensional image"}; |
|
image(image1D); |
|
image(image2D); |
|
image(image3D); |
|
|
|
py::class_<ImageView1D, PyImageViewHolder<ImageView1D>> imageView1D{m, "ImageView1D", "One-dimensional image view"}; |
|
py::class_<ImageView2D, PyImageViewHolder<ImageView2D>> imageView2D{m, "ImageView2D", "Two-dimensional image view"}; |
|
py::class_<ImageView3D, PyImageViewHolder<ImageView3D>> imageView3D{m, "ImageView3D", "Three-dimensional image view"}; |
|
py::class_<MutableImageView1D, PyImageViewHolder<MutableImageView1D>> mutableImageView1D{m, "MutableImageView1D", "One-dimensional mutable image view"}; |
|
py::class_<MutableImageView2D, PyImageViewHolder<MutableImageView2D>> mutableImageView2D{m, "MutableImageView2D", "Two-dimensional mutable image view"}; |
|
py::class_<MutableImageView3D, PyImageViewHolder<MutableImageView3D>> mutableImageView3D{m, "MutableImageView3D", "Three-dimensional mutable image view"}; |
|
|
|
imageView(imageView1D); |
|
imageView(imageView2D); |
|
imageView(imageView3D); |
|
imageView(mutableImageView1D); |
|
imageView(mutableImageView2D); |
|
imageView(mutableImageView3D); |
|
|
|
imageViewFromMutable(imageView1D); |
|
imageViewFromMutable(imageView2D); |
|
imageViewFromMutable(imageView3D); |
|
|
|
py::enum_<SamplerFilter>{m, "SamplerFilter", "Texture sampler filtering"} |
|
.value("NEAREST", SamplerFilter::Nearest) |
|
.value("LINEAR", SamplerFilter::Linear); |
|
py::enum_<SamplerMipmap>{m, "SamplerMipmap", "Texture sampler mip level selection"} |
|
.value("BASE", SamplerMipmap::Base) |
|
.value("NEAREST", SamplerMipmap::Nearest) |
|
.value("LINEAR", SamplerMipmap::Linear); |
|
py::enum_<SamplerWrapping>{m, "SamplerWrapping", "Texture sampler wrapping"} |
|
.value("REPEAT", SamplerWrapping::Repeat) |
|
.value("MIRRORED_REPEAT", SamplerWrapping::MirroredRepeat) |
|
.value("CLAMP_TO_EDGE", SamplerWrapping::ClampToEdge) |
|
.value("CLAMP_TO_BORDER", SamplerWrapping::ClampToBorder) |
|
.value("MIRROR_CLAMP_TO_EDGE", SamplerWrapping::MirrorClampToEdge); |
|
} |
|
|
|
}} |
|
|
|
/* TODO: remove declaration when https://github.com/pybind/pybind11/pull/1863 |
|
is released */ |
|
extern "C" PYBIND11_EXPORT PyObject* PyInit__magnum(); |
|
PYBIND11_MODULE(_magnum, m) { |
|
m.doc() = "Root Magnum module"; |
|
|
|
/* We need ArrayView for images */ |
|
py::module_::import("corrade.containers"); |
|
|
|
py::module_ math = m.def_submodule("math"); |
|
magnum::math(m, math); |
|
|
|
/* These need stuff from math, so need to be called after */ |
|
magnum::magnum(m); |
|
|
|
/* In case Magnum is a bunch of static libraries, put everything into a |
|
single shared lib to make it easier to install (which is the point of |
|
static builds) and avoid issues with multiply-defined global symbols. |
|
|
|
These need to be defined in the order they depend on. */ |
|
#ifdef MAGNUM_BUILD_STATIC |
|
#ifdef Magnum_GL_FOUND |
|
py::module_ gl = m.def_submodule("gl"); |
|
magnum::gl(gl); |
|
#endif |
|
|
|
#ifdef Magnum_SceneGraph_FOUND |
|
py::module_ scenegraph = m.def_submodule("scenegraph"); |
|
magnum::scenegraph(scenegraph); |
|
#endif |
|
|
|
#ifdef Magnum_Text_FOUND |
|
py::module_ text = m.def_submodule("text"); |
|
magnum::text(text); |
|
#endif |
|
|
|
#ifdef Magnum_Trade_FOUND |
|
py::module_ trade = m.def_submodule("trade"); |
|
magnum::trade(trade); |
|
#endif |
|
|
|
#ifdef Magnum_MeshTools_FOUND |
|
/* Depends on trade and gl */ |
|
py::module_ meshtools = m.def_submodule("meshtools"); |
|
magnum::meshtools(meshtools); |
|
#endif |
|
|
|
#ifdef Magnum_Primitives_FOUND |
|
/* Depends on trade */ |
|
py::module_ primitives = m.def_submodule("primitives"); |
|
magnum::primitives(primitives); |
|
#endif |
|
|
|
#ifdef Magnum_Shaders_FOUND |
|
/* Depends on gl */ |
|
py::module_ shaders = m.def_submodule("shaders"); |
|
magnum::shaders(shaders); |
|
#endif |
|
|
|
/* Keep the doc in sync with platform/__init__.py */ |
|
py::module_ platform = m.def_submodule("platform"); |
|
platform.doc() = "Platform-specific application and context creation"; |
|
|
|
#ifdef Magnum_GlfwApplication_FOUND |
|
py::module_ glfw = platform.def_submodule("glfw"); |
|
magnum::platform::glfw(glfw); |
|
#endif |
|
|
|
#ifdef Magnum_Sdl2Application_FOUND |
|
py::module_ sdl2 = platform.def_submodule("sdl2"); |
|
magnum::platform::sdl2(sdl2); |
|
#endif |
|
|
|
#ifdef Magnum_WindowlessEglApplication_FOUND |
|
py::module_ egl = platform.def_submodule("egl"); |
|
magnum::platform::egl(egl); |
|
#endif |
|
|
|
#ifdef Magnum_WindowlessGlxApplication_FOUND |
|
py::module_ glx = platform.def_submodule("glx"); |
|
magnum::platform::glx(glx); |
|
#endif |
|
#endif |
|
}
|
|
|