15#include <pxr/base/vt/array.h>
16#include <pxr/usd/usdGeom/points.h>
17#include <pxr/usd/usdGeom/primvarsAPI.h>
31 const pxr::UsdGeomPoints usd_points = pxr::UsdGeomPoints::Define(stage,
usd_path);
33 pxr::VtArray<pxr::GfVec3f> usd_positions;
34 usd_positions.assign(positions.
begin(), positions.
end());
36 pxr::UsdAttribute attr_positions = usd_points.CreatePointsAttr(pxr::VtValue(),
true);
37 if (!attr_positions.HasValue()) {
38 attr_positions.Set(usd_positions, pxr::UsdTimeCode::Default());
42 if (!radii.is_empty()) {
43 pxr::VtArray<float> usd_widths;
44 usd_widths.resize(radii.size());
45 for (
const int i : radii.index_range()) {
46 usd_widths[
i] = radii[
i] * 2.0f;
49 pxr::UsdAttribute attr_widths = usd_points.CreateWidthsAttr(pxr::VtValue(),
true);
50 if (!attr_widths.HasValue()) {
51 attr_widths.Set(usd_widths, pxr::UsdTimeCode::Default());
56 this->write_velocities(points, usd_points, time);
57 this->write_custom_data(points, usd_points, time);
59 this->
author_extent(usd_points, points->bounds_min_max(), time);
65 switch (blender_domain) {
67 return pxr::UsdGeomTokens->varying;
75 const pxr::UsdGeomPoints &usd_points,
76 const pxr::UsdTimeCode time)
81 if (!pv_interp || !pv_type) {
84 "Attribute '%s' (Blender domain %d, type %d) cannot be converted to USD",
96 const pxr::TfToken pv_name(
98 const pxr::UsdGeomPrimvarsAPI pv_api = pxr::UsdGeomPrimvarsAPI(usd_points);
100 pxr::UsdGeomPrimvar pv_attr = pv_api.CreatePrimvar(pv_name, *pv_type, *pv_interp);
105void USDPointsWriter::write_custom_data(
const PointCloud *points,
106 const pxr::UsdGeomPoints &usd_points,
107 const pxr::UsdTimeCode time)
109 const bke::AttributeAccessor attributes = points->
attributes();
111 attributes.foreach_attribute([&](
const bke::AttributeIter &iter) {
114 ELEM(iter.name,
"position",
"radius",
"id",
"velocity"))
119 this->write_generic_data(iter, usd_points, time);
123void USDPointsWriter::write_velocities(
const PointCloud *points,
124 const pxr::UsdGeomPoints &usd_points,
125 const pxr::UsdTimeCode time)
129 if (velocity.is_empty()) {
133 Span<pxr::GfVec3f>
data = velocity.cast<pxr::GfVec3f>();
134 pxr::VtArray<pxr::GfVec3f> usd_velocities;
135 usd_velocities.assign(
data.begin(),
data.end());
137 pxr::UsdAttribute attr_vel = usd_points.CreateVelocitiesAttr(pxr::VtValue(),
true);
138 if (!attr_vel.HasValue()) {
139 attr_vel.Set(usd_velocities, pxr::UsdTimeCode::Default());
void BKE_reportf(ReportList *reports, eReportType type, const char *format,...) ATTR_PRINTF_FORMAT(3
struct PointCloud PointCloud
BMesh const char void * data
constexpr const T * end() const
constexpr const T * begin() const
constexpr const char * c_str() const
GAttributeReader get() const
void author_extent(const pxr::UsdGeomBoundable &boundable, const pxr::UsdTimeCode time)
const pxr::SdfPath & usd_path() const
ReportList * reports() const
pxr::UsdTimeCode get_export_time_code() const
pxr::UsdUtilsSparseValueWriter usd_value_writer_
const USDExporterContext usd_export_context_
void do_write(HierarchyContext &context) override
bool attribute_name_is_anonymous(const StringRef name)
void copy_blender_attribute_to_primvar(const GVArray &attribute, const bke::AttrType data_type, const pxr::UsdTimeCode time, const pxr::UsdGeomPrimvar &primvar, pxr::UsdUtilsSparseValueWriter &value_writer)
std::optional< pxr::SdfValueTypeName > convert_blender_type_to_usd(const bke::AttrType blender_type, bool use_color3f_type)
static std::optional< pxr::TfToken > convert_blender_domain_to_usd(const bke::AttrDomain blender_domain, bool is_bezier)
std::string make_safe_name(const StringRef name, bool allow_unicode)
VecBase< float, 3 > float3
const USDExportParams & export_params