gpu_particles_collision_3d.cpp 36 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019
  1. /**************************************************************************/
  2. /* gpu_particles_collision_3d.cpp */
  3. /**************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /**************************************************************************/
  8. /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
  9. /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /**************************************************************************/
  30. #include "gpu_particles_collision_3d.h"
  31. #include "core/object/worker_thread_pool.h"
  32. #include "mesh_instance_3d.h"
  33. #include "scene/3d/camera_3d.h"
  34. #include "scene/main/viewport.h"
  35. void GPUParticlesCollision3D::set_cull_mask(uint32_t p_cull_mask) {
  36. cull_mask = p_cull_mask;
  37. RS::get_singleton()->particles_collision_set_cull_mask(collision, p_cull_mask);
  38. }
  39. uint32_t GPUParticlesCollision3D::get_cull_mask() const {
  40. return cull_mask;
  41. }
  42. void GPUParticlesCollision3D::_bind_methods() {
  43. ClassDB::bind_method(D_METHOD("set_cull_mask", "mask"), &GPUParticlesCollision3D::set_cull_mask);
  44. ClassDB::bind_method(D_METHOD("get_cull_mask"), &GPUParticlesCollision3D::get_cull_mask);
  45. ADD_PROPERTY(PropertyInfo(Variant::INT, "cull_mask", PROPERTY_HINT_LAYERS_3D_RENDER), "set_cull_mask", "get_cull_mask");
  46. }
  47. GPUParticlesCollision3D::GPUParticlesCollision3D(RS::ParticlesCollisionType p_type) {
  48. collision = RS::get_singleton()->particles_collision_create();
  49. RS::get_singleton()->particles_collision_set_collision_type(collision, p_type);
  50. set_base(collision);
  51. }
  52. GPUParticlesCollision3D::~GPUParticlesCollision3D() {
  53. ERR_FAIL_NULL(RenderingServer::get_singleton());
  54. RS::get_singleton()->free(collision);
  55. }
  56. /////////////////////////////////
  57. void GPUParticlesCollisionSphere3D::_bind_methods() {
  58. ClassDB::bind_method(D_METHOD("set_radius", "radius"), &GPUParticlesCollisionSphere3D::set_radius);
  59. ClassDB::bind_method(D_METHOD("get_radius"), &GPUParticlesCollisionSphere3D::get_radius);
  60. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "radius", PROPERTY_HINT_RANGE, "0.01,1024,0.01,or_greater,suffix:m"), "set_radius", "get_radius");
  61. }
  62. void GPUParticlesCollisionSphere3D::set_radius(real_t p_radius) {
  63. radius = p_radius;
  64. RS::get_singleton()->particles_collision_set_sphere_radius(_get_collision(), radius);
  65. update_gizmos();
  66. }
  67. real_t GPUParticlesCollisionSphere3D::get_radius() const {
  68. return radius;
  69. }
  70. AABB GPUParticlesCollisionSphere3D::get_aabb() const {
  71. return AABB(Vector3(-radius, -radius, -radius), Vector3(radius * 2, radius * 2, radius * 2));
  72. }
  73. GPUParticlesCollisionSphere3D::GPUParticlesCollisionSphere3D() :
  74. GPUParticlesCollision3D(RS::PARTICLES_COLLISION_TYPE_SPHERE_COLLIDE) {
  75. }
  76. GPUParticlesCollisionSphere3D::~GPUParticlesCollisionSphere3D() {
  77. }
  78. ///////////////////////////
  79. void GPUParticlesCollisionBox3D::_bind_methods() {
  80. ClassDB::bind_method(D_METHOD("set_size", "size"), &GPUParticlesCollisionBox3D::set_size);
  81. ClassDB::bind_method(D_METHOD("get_size"), &GPUParticlesCollisionBox3D::get_size);
  82. ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "size", PROPERTY_HINT_RANGE, "0.01,1024,0.01,or_greater,suffix:m"), "set_size", "get_size");
  83. }
  84. #ifndef DISABLE_DEPRECATED
  85. bool GPUParticlesCollisionBox3D::_set(const StringName &p_name, const Variant &p_value) {
  86. if (p_name == "extents") { // Compatibility with Godot 3.x.
  87. set_size((Vector3)p_value * 2);
  88. return true;
  89. }
  90. return false;
  91. }
  92. bool GPUParticlesCollisionBox3D::_get(const StringName &p_name, Variant &r_property) const {
  93. if (p_name == "extents") { // Compatibility with Godot 3.x.
  94. r_property = size / 2;
  95. return true;
  96. }
  97. return false;
  98. }
  99. #endif // DISABLE_DEPRECATED
  100. void GPUParticlesCollisionBox3D::set_size(const Vector3 &p_size) {
  101. size = p_size;
  102. RS::get_singleton()->particles_collision_set_box_extents(_get_collision(), size / 2);
  103. update_gizmos();
  104. }
  105. Vector3 GPUParticlesCollisionBox3D::get_size() const {
  106. return size;
  107. }
  108. AABB GPUParticlesCollisionBox3D::get_aabb() const {
  109. return AABB(-size / 2, size);
  110. }
  111. GPUParticlesCollisionBox3D::GPUParticlesCollisionBox3D() :
  112. GPUParticlesCollision3D(RS::PARTICLES_COLLISION_TYPE_BOX_COLLIDE) {
  113. }
  114. GPUParticlesCollisionBox3D::~GPUParticlesCollisionBox3D() {
  115. }
  116. ///////////////////////////////
  117. ///////////////////////////
  118. void GPUParticlesCollisionSDF3D::_find_meshes(const AABB &p_aabb, Node *p_at_node, List<PlotMesh> &plot_meshes) {
  119. MeshInstance3D *mi = Object::cast_to<MeshInstance3D>(p_at_node);
  120. if (mi && mi->is_visible_in_tree()) {
  121. if ((mi->get_layer_mask() & bake_mask) == 0) {
  122. return;
  123. }
  124. Ref<Mesh> mesh = mi->get_mesh();
  125. if (mesh.is_valid()) {
  126. AABB aabb = mesh->get_aabb();
  127. Transform3D xf = get_global_transform().affine_inverse() * mi->get_global_transform();
  128. if (p_aabb.intersects(xf.xform(aabb))) {
  129. PlotMesh pm;
  130. pm.local_xform = xf;
  131. pm.mesh = mesh;
  132. plot_meshes.push_back(pm);
  133. }
  134. }
  135. }
  136. Node3D *s = Object::cast_to<Node3D>(p_at_node);
  137. if (s) {
  138. if (s->is_visible_in_tree()) {
  139. Array meshes = p_at_node->call("get_meshes");
  140. for (int i = 0; i < meshes.size(); i += 2) {
  141. Transform3D mxf = meshes[i];
  142. Ref<Mesh> mesh = meshes[i + 1];
  143. if (!mesh.is_valid()) {
  144. continue;
  145. }
  146. AABB aabb = mesh->get_aabb();
  147. Transform3D xf = get_global_transform().affine_inverse() * (s->get_global_transform() * mxf);
  148. if (p_aabb.intersects(xf.xform(aabb))) {
  149. PlotMesh pm;
  150. pm.local_xform = xf;
  151. pm.mesh = mesh;
  152. plot_meshes.push_back(pm);
  153. }
  154. }
  155. }
  156. }
  157. for (int i = 0; i < p_at_node->get_child_count(); i++) {
  158. Node *child = p_at_node->get_child(i);
  159. _find_meshes(p_aabb, child, plot_meshes);
  160. }
  161. }
  162. uint32_t GPUParticlesCollisionSDF3D::_create_bvh(LocalVector<BVH> &bvh_tree, FacePos *p_faces, uint32_t p_face_count, const Face3 *p_triangles, float p_thickness) {
  163. if (p_face_count == 1) {
  164. return BVH::LEAF_BIT | p_faces[0].index;
  165. }
  166. uint32_t index = bvh_tree.size();
  167. {
  168. BVH bvh;
  169. for (uint32_t i = 0; i < p_face_count; i++) {
  170. const Face3 &f = p_triangles[p_faces[i].index];
  171. AABB aabb(f.vertex[0], Vector3());
  172. aabb.expand_to(f.vertex[1]);
  173. aabb.expand_to(f.vertex[2]);
  174. if (p_thickness > 0.0) {
  175. Vector3 normal = p_triangles[p_faces[i].index].get_plane().normal;
  176. aabb.expand_to(f.vertex[0] - normal * p_thickness);
  177. aabb.expand_to(f.vertex[1] - normal * p_thickness);
  178. aabb.expand_to(f.vertex[2] - normal * p_thickness);
  179. }
  180. if (i == 0) {
  181. bvh.bounds = aabb;
  182. } else {
  183. bvh.bounds.merge_with(aabb);
  184. }
  185. }
  186. bvh_tree.push_back(bvh);
  187. }
  188. uint32_t middle = p_face_count / 2;
  189. SortArray<FacePos, FaceSort> s;
  190. s.compare.axis = bvh_tree[index].bounds.get_longest_axis_index();
  191. s.sort(p_faces, p_face_count);
  192. uint32_t left = _create_bvh(bvh_tree, p_faces, middle, p_triangles, p_thickness);
  193. uint32_t right = _create_bvh(bvh_tree, p_faces + middle, p_face_count - middle, p_triangles, p_thickness);
  194. bvh_tree[index].children[0] = left;
  195. bvh_tree[index].children[1] = right;
  196. return index;
  197. }
  198. static _FORCE_INLINE_ real_t Vector3_dot2(const Vector3 &p_vec3) {
  199. return p_vec3.dot(p_vec3);
  200. }
  201. void GPUParticlesCollisionSDF3D::_find_closest_distance(const Vector3 &p_pos, const BVH *p_bvh, uint32_t p_bvh_cell, const Face3 *p_triangles, float p_thickness, float &r_closest_distance) {
  202. if (p_bvh_cell & BVH::LEAF_BIT) {
  203. p_bvh_cell &= BVH::LEAF_MASK; //remove bit
  204. Vector3 point = p_pos;
  205. Plane p = p_triangles[p_bvh_cell].get_plane();
  206. float d = p.distance_to(point);
  207. float inside_d = 1e20;
  208. if (d < 0 && d > -p_thickness) {
  209. //inside planes, do this in 2D
  210. Vector3 x_axis = (p_triangles[p_bvh_cell].vertex[0] - p_triangles[p_bvh_cell].vertex[1]).normalized();
  211. Vector3 y_axis = p.normal.cross(x_axis).normalized();
  212. Vector2 points[3];
  213. for (int i = 0; i < 3; i++) {
  214. points[i] = Vector2(x_axis.dot(p_triangles[p_bvh_cell].vertex[i]), y_axis.dot(p_triangles[p_bvh_cell].vertex[i]));
  215. }
  216. Vector2 p2d = Vector2(x_axis.dot(point), y_axis.dot(point));
  217. {
  218. // https://www.shadertoy.com/view/XsXSz4
  219. Vector2 e0 = points[1] - points[0];
  220. Vector2 e1 = points[2] - points[1];
  221. Vector2 e2 = points[0] - points[2];
  222. Vector2 v0 = p2d - points[0];
  223. Vector2 v1 = p2d - points[1];
  224. Vector2 v2 = p2d - points[2];
  225. Vector2 pq0 = v0 - e0 * CLAMP(v0.dot(e0) / e0.dot(e0), 0.0, 1.0);
  226. Vector2 pq1 = v1 - e1 * CLAMP(v1.dot(e1) / e1.dot(e1), 0.0, 1.0);
  227. Vector2 pq2 = v2 - e2 * CLAMP(v2.dot(e2) / e2.dot(e2), 0.0, 1.0);
  228. float s = SIGN(e0.x * e2.y - e0.y * e2.x);
  229. Vector2 d2 = Vector2(pq0.dot(pq0), s * (v0.x * e0.y - v0.y * e0.x)).min(Vector2(pq1.dot(pq1), s * (v1.x * e1.y - v1.y * e1.x))).min(Vector2(pq2.dot(pq2), s * (v2.x * e2.y - v2.y * e2.x)));
  230. inside_d = -Math::sqrt(d2.x) * SIGN(d2.y);
  231. }
  232. //make sure distance to planes is not shorter if inside
  233. if (inside_d < 0) {
  234. inside_d = MAX(inside_d, d);
  235. inside_d = MAX(inside_d, -(p_thickness + d));
  236. }
  237. r_closest_distance = MIN(r_closest_distance, inside_d);
  238. } else {
  239. if (d < 0) {
  240. point -= p.normal * p_thickness; //flatten
  241. }
  242. // https://iquilezles.org/www/articles/distfunctions/distfunctions.htm
  243. Vector3 a = p_triangles[p_bvh_cell].vertex[0];
  244. Vector3 b = p_triangles[p_bvh_cell].vertex[1];
  245. Vector3 c = p_triangles[p_bvh_cell].vertex[2];
  246. Vector3 ba = b - a;
  247. Vector3 pa = point - a;
  248. Vector3 cb = c - b;
  249. Vector3 pb = point - b;
  250. Vector3 ac = a - c;
  251. Vector3 pc = point - c;
  252. Vector3 nor = ba.cross(ac);
  253. inside_d = Math::sqrt(
  254. (SIGN(ba.cross(nor).dot(pa)) + SIGN(cb.cross(nor).dot(pb)) + SIGN(ac.cross(nor).dot(pc)) < 2.0)
  255. ? MIN(MIN(
  256. Vector3_dot2(ba * CLAMP(ba.dot(pa) / Vector3_dot2(ba), 0.0, 1.0) - pa),
  257. Vector3_dot2(cb * CLAMP(cb.dot(pb) / Vector3_dot2(cb), 0.0, 1.0) - pb)),
  258. Vector3_dot2(ac * CLAMP(ac.dot(pc) / Vector3_dot2(ac), 0.0, 1.0) - pc))
  259. : nor.dot(pa) * nor.dot(pa) / Vector3_dot2(nor));
  260. r_closest_distance = MIN(r_closest_distance, inside_d);
  261. }
  262. } else {
  263. bool pass = true;
  264. if (!p_bvh[p_bvh_cell].bounds.has_point(p_pos)) {
  265. //outside, find closest point
  266. Vector3 he = p_bvh[p_bvh_cell].bounds.size * 0.5;
  267. Vector3 center = p_bvh[p_bvh_cell].bounds.position + he;
  268. Vector3 rel = (p_pos - center).abs();
  269. Vector3 closest(MIN(rel.x, he.x), MIN(rel.y, he.y), MIN(rel.z, he.z));
  270. float d = rel.distance_to(closest);
  271. if (d >= r_closest_distance) {
  272. pass = false; //already closer than this aabb, discard
  273. }
  274. }
  275. if (pass) {
  276. _find_closest_distance(p_pos, p_bvh, p_bvh[p_bvh_cell].children[0], p_triangles, p_thickness, r_closest_distance);
  277. _find_closest_distance(p_pos, p_bvh, p_bvh[p_bvh_cell].children[1], p_triangles, p_thickness, r_closest_distance);
  278. }
  279. }
  280. }
  281. void GPUParticlesCollisionSDF3D::_compute_sdf_z(uint32_t p_z, ComputeSDFParams *params) {
  282. int32_t z_ofs = p_z * params->size.y * params->size.x;
  283. for (int32_t y = 0; y < params->size.y; y++) {
  284. int32_t y_ofs = z_ofs + y * params->size.x;
  285. for (int32_t x = 0; x < params->size.x; x++) {
  286. int32_t x_ofs = y_ofs + x;
  287. float &cell = params->cells[x_ofs];
  288. Vector3 pos = params->cell_offset + Vector3(x, y, p_z) * params->cell_size;
  289. cell = 1e20;
  290. _find_closest_distance(pos, params->bvh, 0, params->triangles, params->thickness, cell);
  291. }
  292. }
  293. }
  294. void GPUParticlesCollisionSDF3D::_compute_sdf(ComputeSDFParams *params) {
  295. WorkerThreadPool::GroupID group_task = WorkerThreadPool::get_singleton()->add_template_group_task(this, &GPUParticlesCollisionSDF3D::_compute_sdf_z, params, params->size.z);
  296. while (!WorkerThreadPool::get_singleton()->is_group_task_completed(group_task)) {
  297. OS::get_singleton()->delay_usec(10000);
  298. if (bake_step_function) {
  299. bake_step_function(WorkerThreadPool::get_singleton()->get_group_processed_element_count(group_task) * 100 / params->size.z, "Baking SDF");
  300. }
  301. }
  302. WorkerThreadPool::get_singleton()->wait_for_group_task_completion(group_task);
  303. }
  304. Vector3i GPUParticlesCollisionSDF3D::get_estimated_cell_size() const {
  305. static const int subdivs[RESOLUTION_MAX] = { 16, 32, 64, 128, 256, 512 };
  306. int subdiv = subdivs[get_resolution()];
  307. AABB aabb(-size / 2, size);
  308. float cell_size = aabb.get_longest_axis_size() / float(subdiv);
  309. Vector3i sdf_size = Vector3i(aabb.size / cell_size);
  310. sdf_size.x = MAX(1, sdf_size.x);
  311. sdf_size.y = MAX(1, sdf_size.y);
  312. sdf_size.z = MAX(1, sdf_size.z);
  313. return sdf_size;
  314. }
  315. Ref<Image> GPUParticlesCollisionSDF3D::bake() {
  316. static const int subdivs[RESOLUTION_MAX] = { 16, 32, 64, 128, 256, 512 };
  317. int subdiv = subdivs[get_resolution()];
  318. AABB aabb(-size / 2, size);
  319. float cell_size = aabb.get_longest_axis_size() / float(subdiv);
  320. Vector3i sdf_size = Vector3i(aabb.size / cell_size);
  321. sdf_size.x = MAX(1, sdf_size.x);
  322. sdf_size.y = MAX(1, sdf_size.y);
  323. sdf_size.z = MAX(1, sdf_size.z);
  324. if (bake_begin_function) {
  325. bake_begin_function(100);
  326. }
  327. aabb.size = Vector3(sdf_size) * cell_size;
  328. List<PlotMesh> plot_meshes;
  329. _find_meshes(aabb, get_parent(), plot_meshes);
  330. LocalVector<Face3> faces;
  331. if (bake_step_function) {
  332. bake_step_function(0, "Finding Meshes");
  333. }
  334. for (const PlotMesh &pm : plot_meshes) {
  335. for (int i = 0; i < pm.mesh->get_surface_count(); i++) {
  336. if (pm.mesh->surface_get_primitive_type(i) != Mesh::PRIMITIVE_TRIANGLES) {
  337. continue; //only triangles
  338. }
  339. Array a = pm.mesh->surface_get_arrays(i);
  340. Vector<Vector3> vertices = a[Mesh::ARRAY_VERTEX];
  341. const Vector3 *vr = vertices.ptr();
  342. Vector<int> index = a[Mesh::ARRAY_INDEX];
  343. if (index.size()) {
  344. int facecount = index.size() / 3;
  345. const int *ir = index.ptr();
  346. for (int j = 0; j < facecount; j++) {
  347. Face3 face;
  348. for (int k = 0; k < 3; k++) {
  349. face.vertex[k] = pm.local_xform.xform(vr[ir[j * 3 + k]]);
  350. }
  351. //test against original bounds
  352. if (!Geometry3D::triangle_box_overlap(aabb.get_center(), aabb.size * 0.5, face.vertex)) {
  353. continue;
  354. }
  355. faces.push_back(face);
  356. }
  357. } else {
  358. int facecount = vertices.size() / 3;
  359. for (int j = 0; j < facecount; j++) {
  360. Face3 face;
  361. for (int k = 0; k < 3; k++) {
  362. face.vertex[k] = pm.local_xform.xform(vr[j * 3 + k]);
  363. }
  364. //test against original bounds
  365. if (!Geometry3D::triangle_box_overlap(aabb.get_center(), aabb.size * 0.5, face.vertex)) {
  366. continue;
  367. }
  368. faces.push_back(face);
  369. }
  370. }
  371. }
  372. }
  373. //compute bvh
  374. ERR_FAIL_COND_V_MSG(faces.size() <= 1, Ref<Image>(), "No faces detected during GPUParticlesCollisionSDF3D bake. Check whether there are visible meshes matching the bake mask within its extents.");
  375. LocalVector<FacePos> face_pos;
  376. face_pos.resize(faces.size());
  377. float th = cell_size * thickness;
  378. for (uint32_t i = 0; i < faces.size(); i++) {
  379. face_pos[i].index = i;
  380. face_pos[i].center = (faces[i].vertex[0] + faces[i].vertex[1] + faces[i].vertex[2]) / 2;
  381. if (th > 0.0) {
  382. face_pos[i].center -= faces[i].get_plane().normal * th * 0.5;
  383. }
  384. }
  385. if (bake_step_function) {
  386. bake_step_function(0, "Creating BVH");
  387. }
  388. LocalVector<BVH> bvh;
  389. _create_bvh(bvh, face_pos.ptr(), face_pos.size(), faces.ptr(), th);
  390. Vector<uint8_t> cells_data;
  391. cells_data.resize(sdf_size.z * sdf_size.y * sdf_size.x * (int)sizeof(float));
  392. if (bake_step_function) {
  393. bake_step_function(0, "Baking SDF");
  394. }
  395. ComputeSDFParams params;
  396. params.cells = (float *)cells_data.ptrw();
  397. params.size = sdf_size;
  398. params.cell_size = cell_size;
  399. params.cell_offset = aabb.position + Vector3(cell_size * 0.5, cell_size * 0.5, cell_size * 0.5);
  400. params.bvh = bvh.ptr();
  401. params.triangles = faces.ptr();
  402. params.thickness = th;
  403. _compute_sdf(&params);
  404. Ref<Image> ret = Image::create_from_data(sdf_size.x, sdf_size.y * sdf_size.z, false, Image::FORMAT_RF, cells_data);
  405. ret->convert(Image::FORMAT_RH); //convert to half, save space
  406. ret->set_meta("depth", sdf_size.z); //hack, make sure to add to the docs of this function
  407. if (bake_end_function) {
  408. bake_end_function();
  409. }
  410. return ret;
  411. }
  412. PackedStringArray GPUParticlesCollisionSDF3D::get_configuration_warnings() const {
  413. PackedStringArray warnings = Node::get_configuration_warnings();
  414. if (bake_mask == 0) {
  415. warnings.push_back(RTR("The Bake Mask has no bits enabled, which means baking will not produce any collision for this GPUParticlesCollisionSDF3D.\nTo resolve this, enable at least one bit in the Bake Mask property."));
  416. }
  417. return warnings;
  418. }
  419. void GPUParticlesCollisionSDF3D::_bind_methods() {
  420. ClassDB::bind_method(D_METHOD("set_size", "size"), &GPUParticlesCollisionSDF3D::set_size);
  421. ClassDB::bind_method(D_METHOD("get_size"), &GPUParticlesCollisionSDF3D::get_size);
  422. ClassDB::bind_method(D_METHOD("set_resolution", "resolution"), &GPUParticlesCollisionSDF3D::set_resolution);
  423. ClassDB::bind_method(D_METHOD("get_resolution"), &GPUParticlesCollisionSDF3D::get_resolution);
  424. ClassDB::bind_method(D_METHOD("set_texture", "texture"), &GPUParticlesCollisionSDF3D::set_texture);
  425. ClassDB::bind_method(D_METHOD("get_texture"), &GPUParticlesCollisionSDF3D::get_texture);
  426. ClassDB::bind_method(D_METHOD("set_thickness", "thickness"), &GPUParticlesCollisionSDF3D::set_thickness);
  427. ClassDB::bind_method(D_METHOD("get_thickness"), &GPUParticlesCollisionSDF3D::get_thickness);
  428. ClassDB::bind_method(D_METHOD("set_bake_mask", "mask"), &GPUParticlesCollisionSDF3D::set_bake_mask);
  429. ClassDB::bind_method(D_METHOD("get_bake_mask"), &GPUParticlesCollisionSDF3D::get_bake_mask);
  430. ClassDB::bind_method(D_METHOD("set_bake_mask_value", "layer_number", "value"), &GPUParticlesCollisionSDF3D::set_bake_mask_value);
  431. ClassDB::bind_method(D_METHOD("get_bake_mask_value", "layer_number"), &GPUParticlesCollisionSDF3D::get_bake_mask_value);
  432. ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "size", PROPERTY_HINT_RANGE, "0.01,1024,0.01,or_greater,suffix:m"), "set_size", "get_size");
  433. ADD_PROPERTY(PropertyInfo(Variant::INT, "resolution", PROPERTY_HINT_ENUM, "16,32,64,128,256,512"), "set_resolution", "get_resolution");
  434. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "thickness", PROPERTY_HINT_RANGE, "0.0,2.0,0.01,suffix:m"), "set_thickness", "get_thickness");
  435. ADD_PROPERTY(PropertyInfo(Variant::INT, "bake_mask", PROPERTY_HINT_LAYERS_3D_RENDER), "set_bake_mask", "get_bake_mask");
  436. ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "texture", PROPERTY_HINT_RESOURCE_TYPE, "Texture3D"), "set_texture", "get_texture");
  437. BIND_ENUM_CONSTANT(RESOLUTION_16);
  438. BIND_ENUM_CONSTANT(RESOLUTION_32);
  439. BIND_ENUM_CONSTANT(RESOLUTION_64);
  440. BIND_ENUM_CONSTANT(RESOLUTION_128);
  441. BIND_ENUM_CONSTANT(RESOLUTION_256);
  442. BIND_ENUM_CONSTANT(RESOLUTION_512);
  443. BIND_ENUM_CONSTANT(RESOLUTION_MAX);
  444. }
  445. #ifndef DISABLE_DEPRECATED
  446. bool GPUParticlesCollisionSDF3D::_set(const StringName &p_name, const Variant &p_value) {
  447. if (p_name == "extents") { // Compatibility with Godot 3.x.
  448. set_size((Vector3)p_value * 2);
  449. return true;
  450. }
  451. return false;
  452. }
  453. bool GPUParticlesCollisionSDF3D::_get(const StringName &p_name, Variant &r_property) const {
  454. if (p_name == "extents") { // Compatibility with Godot 3.x.
  455. r_property = size / 2;
  456. return true;
  457. }
  458. return false;
  459. }
  460. #endif // DISABLE_DEPRECATED
  461. void GPUParticlesCollisionSDF3D::set_thickness(float p_thickness) {
  462. thickness = p_thickness;
  463. }
  464. float GPUParticlesCollisionSDF3D::get_thickness() const {
  465. return thickness;
  466. }
  467. void GPUParticlesCollisionSDF3D::set_size(const Vector3 &p_size) {
  468. size = p_size;
  469. RS::get_singleton()->particles_collision_set_box_extents(_get_collision(), size / 2);
  470. update_gizmos();
  471. }
  472. Vector3 GPUParticlesCollisionSDF3D::get_size() const {
  473. return size;
  474. }
  475. void GPUParticlesCollisionSDF3D::set_resolution(Resolution p_resolution) {
  476. resolution = p_resolution;
  477. update_gizmos();
  478. }
  479. GPUParticlesCollisionSDF3D::Resolution GPUParticlesCollisionSDF3D::get_resolution() const {
  480. return resolution;
  481. }
  482. void GPUParticlesCollisionSDF3D::set_bake_mask(uint32_t p_mask) {
  483. bake_mask = p_mask;
  484. update_configuration_warnings();
  485. }
  486. uint32_t GPUParticlesCollisionSDF3D::get_bake_mask() const {
  487. return bake_mask;
  488. }
  489. void GPUParticlesCollisionSDF3D::set_bake_mask_value(int p_layer_number, bool p_value) {
  490. ERR_FAIL_COND_MSG(p_layer_number < 1 || p_layer_number > 20, vformat("The render layer number (%d) must be between 1 and 20 (inclusive).", p_layer_number));
  491. uint32_t mask = get_bake_mask();
  492. if (p_value) {
  493. mask |= 1 << (p_layer_number - 1);
  494. } else {
  495. mask &= ~(1 << (p_layer_number - 1));
  496. }
  497. set_bake_mask(mask);
  498. }
  499. bool GPUParticlesCollisionSDF3D::get_bake_mask_value(int p_layer_number) const {
  500. ERR_FAIL_COND_V_MSG(p_layer_number < 1 || p_layer_number > 20, false, vformat("The render layer number (%d) must be between 1 and 20 (inclusive).", p_layer_number));
  501. return bake_mask & (1 << (p_layer_number - 1));
  502. }
  503. void GPUParticlesCollisionSDF3D::set_texture(const Ref<Texture3D> &p_texture) {
  504. texture = p_texture;
  505. RID tex = texture.is_valid() ? texture->get_rid() : RID();
  506. RS::get_singleton()->particles_collision_set_field_texture(_get_collision(), tex);
  507. }
  508. Ref<Texture3D> GPUParticlesCollisionSDF3D::get_texture() const {
  509. return texture;
  510. }
  511. AABB GPUParticlesCollisionSDF3D::get_aabb() const {
  512. return AABB(-size / 2, size);
  513. }
  514. GPUParticlesCollisionSDF3D::BakeBeginFunc GPUParticlesCollisionSDF3D::bake_begin_function = nullptr;
  515. GPUParticlesCollisionSDF3D::BakeStepFunc GPUParticlesCollisionSDF3D::bake_step_function = nullptr;
  516. GPUParticlesCollisionSDF3D::BakeEndFunc GPUParticlesCollisionSDF3D::bake_end_function = nullptr;
  517. GPUParticlesCollisionSDF3D::GPUParticlesCollisionSDF3D() :
  518. GPUParticlesCollision3D(RS::PARTICLES_COLLISION_TYPE_SDF_COLLIDE) {
  519. }
  520. GPUParticlesCollisionSDF3D::~GPUParticlesCollisionSDF3D() {
  521. }
  522. ////////////////////////////
  523. ////////////////////////////
  524. void GPUParticlesCollisionHeightField3D::_notification(int p_what) {
  525. switch (p_what) {
  526. case NOTIFICATION_INTERNAL_PROCESS: {
  527. if (update_mode == UPDATE_MODE_ALWAYS) {
  528. RS::get_singleton()->particles_collision_height_field_update(_get_collision());
  529. }
  530. if (follow_camera_mode && get_viewport()) {
  531. Camera3D *cam = get_viewport()->get_camera_3d();
  532. if (cam) {
  533. Transform3D xform = get_global_transform();
  534. Vector3 x_axis = xform.basis.get_column(Vector3::AXIS_X).normalized();
  535. Vector3 z_axis = xform.basis.get_column(Vector3::AXIS_Z).normalized();
  536. float x_len = xform.basis.get_scale().x;
  537. float z_len = xform.basis.get_scale().z;
  538. Vector3 cam_pos = cam->get_global_transform().origin;
  539. Transform3D new_xform = xform;
  540. while (x_axis.dot(cam_pos - new_xform.origin) > x_len) {
  541. new_xform.origin += x_axis * x_len;
  542. }
  543. while (x_axis.dot(cam_pos - new_xform.origin) < -x_len) {
  544. new_xform.origin -= x_axis * x_len;
  545. }
  546. while (z_axis.dot(cam_pos - new_xform.origin) > z_len) {
  547. new_xform.origin += z_axis * z_len;
  548. }
  549. while (z_axis.dot(cam_pos - new_xform.origin) < -z_len) {
  550. new_xform.origin -= z_axis * z_len;
  551. }
  552. if (new_xform != xform) {
  553. set_global_transform(new_xform);
  554. RS::get_singleton()->particles_collision_height_field_update(_get_collision());
  555. }
  556. }
  557. }
  558. } break;
  559. case NOTIFICATION_TRANSFORM_CHANGED: {
  560. RS::get_singleton()->particles_collision_height_field_update(_get_collision());
  561. } break;
  562. }
  563. }
  564. void GPUParticlesCollisionHeightField3D::_bind_methods() {
  565. ClassDB::bind_method(D_METHOD("set_size", "size"), &GPUParticlesCollisionHeightField3D::set_size);
  566. ClassDB::bind_method(D_METHOD("get_size"), &GPUParticlesCollisionHeightField3D::get_size);
  567. ClassDB::bind_method(D_METHOD("set_resolution", "resolution"), &GPUParticlesCollisionHeightField3D::set_resolution);
  568. ClassDB::bind_method(D_METHOD("get_resolution"), &GPUParticlesCollisionHeightField3D::get_resolution);
  569. ClassDB::bind_method(D_METHOD("set_update_mode", "update_mode"), &GPUParticlesCollisionHeightField3D::set_update_mode);
  570. ClassDB::bind_method(D_METHOD("get_update_mode"), &GPUParticlesCollisionHeightField3D::get_update_mode);
  571. ClassDB::bind_method(D_METHOD("set_follow_camera_enabled", "enabled"), &GPUParticlesCollisionHeightField3D::set_follow_camera_enabled);
  572. ClassDB::bind_method(D_METHOD("is_follow_camera_enabled"), &GPUParticlesCollisionHeightField3D::is_follow_camera_enabled);
  573. ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "size", PROPERTY_HINT_RANGE, "0.01,1024,0.01,or_greater,suffix:m"), "set_size", "get_size");
  574. ADD_PROPERTY(PropertyInfo(Variant::INT, "resolution", PROPERTY_HINT_ENUM, "256 (Fastest),512 (Fast),1024 (Average),2048 (Slow),4096 (Slower),8192 (Slowest)"), "set_resolution", "get_resolution");
  575. ADD_PROPERTY(PropertyInfo(Variant::INT, "update_mode", PROPERTY_HINT_ENUM, "When Moved (Fast),Always (Slow)"), "set_update_mode", "get_update_mode");
  576. ADD_PROPERTY(PropertyInfo(Variant::BOOL, "follow_camera_enabled"), "set_follow_camera_enabled", "is_follow_camera_enabled");
  577. BIND_ENUM_CONSTANT(RESOLUTION_256);
  578. BIND_ENUM_CONSTANT(RESOLUTION_512);
  579. BIND_ENUM_CONSTANT(RESOLUTION_1024);
  580. BIND_ENUM_CONSTANT(RESOLUTION_2048);
  581. BIND_ENUM_CONSTANT(RESOLUTION_4096);
  582. BIND_ENUM_CONSTANT(RESOLUTION_8192);
  583. BIND_ENUM_CONSTANT(RESOLUTION_MAX);
  584. BIND_ENUM_CONSTANT(UPDATE_MODE_WHEN_MOVED);
  585. BIND_ENUM_CONSTANT(UPDATE_MODE_ALWAYS);
  586. }
  587. #ifndef DISABLE_DEPRECATED
  588. bool GPUParticlesCollisionHeightField3D::_set(const StringName &p_name, const Variant &p_value) {
  589. if (p_name == "extents") { // Compatibility with Godot 3.x.
  590. set_size((Vector3)p_value * 2);
  591. return true;
  592. }
  593. return false;
  594. }
  595. bool GPUParticlesCollisionHeightField3D::_get(const StringName &p_name, Variant &r_property) const {
  596. if (p_name == "extents") { // Compatibility with Godot 3.x.
  597. r_property = size / 2;
  598. return true;
  599. }
  600. return false;
  601. }
  602. #endif // DISABLE_DEPRECATED
  603. void GPUParticlesCollisionHeightField3D::set_size(const Vector3 &p_size) {
  604. size = p_size;
  605. RS::get_singleton()->particles_collision_set_box_extents(_get_collision(), size / 2);
  606. update_gizmos();
  607. RS::get_singleton()->particles_collision_height_field_update(_get_collision());
  608. }
  609. Vector3 GPUParticlesCollisionHeightField3D::get_size() const {
  610. return size;
  611. }
  612. void GPUParticlesCollisionHeightField3D::set_resolution(Resolution p_resolution) {
  613. resolution = p_resolution;
  614. RS::get_singleton()->particles_collision_set_height_field_resolution(_get_collision(), RS::ParticlesCollisionHeightfieldResolution(resolution));
  615. update_gizmos();
  616. RS::get_singleton()->particles_collision_height_field_update(_get_collision());
  617. }
  618. GPUParticlesCollisionHeightField3D::Resolution GPUParticlesCollisionHeightField3D::get_resolution() const {
  619. return resolution;
  620. }
  621. void GPUParticlesCollisionHeightField3D::set_update_mode(UpdateMode p_update_mode) {
  622. update_mode = p_update_mode;
  623. set_process_internal(follow_camera_mode || update_mode == UPDATE_MODE_ALWAYS);
  624. }
  625. GPUParticlesCollisionHeightField3D::UpdateMode GPUParticlesCollisionHeightField3D::get_update_mode() const {
  626. return update_mode;
  627. }
  628. void GPUParticlesCollisionHeightField3D::set_follow_camera_enabled(bool p_enabled) {
  629. follow_camera_mode = p_enabled;
  630. set_process_internal(follow_camera_mode || update_mode == UPDATE_MODE_ALWAYS);
  631. }
  632. bool GPUParticlesCollisionHeightField3D::is_follow_camera_enabled() const {
  633. return follow_camera_mode;
  634. }
  635. AABB GPUParticlesCollisionHeightField3D::get_aabb() const {
  636. return AABB(-size / 2, size);
  637. }
  638. GPUParticlesCollisionHeightField3D::GPUParticlesCollisionHeightField3D() :
  639. GPUParticlesCollision3D(RS::PARTICLES_COLLISION_TYPE_HEIGHTFIELD_COLLIDE) {
  640. }
  641. GPUParticlesCollisionHeightField3D::~GPUParticlesCollisionHeightField3D() {
  642. }
  643. ////////////////////////////
  644. ////////////////////////////
  645. void GPUParticlesAttractor3D::set_cull_mask(uint32_t p_cull_mask) {
  646. cull_mask = p_cull_mask;
  647. RS::get_singleton()->particles_collision_set_cull_mask(collision, p_cull_mask);
  648. }
  649. uint32_t GPUParticlesAttractor3D::get_cull_mask() const {
  650. return cull_mask;
  651. }
  652. void GPUParticlesAttractor3D::set_strength(real_t p_strength) {
  653. strength = p_strength;
  654. RS::get_singleton()->particles_collision_set_attractor_strength(collision, p_strength);
  655. }
  656. real_t GPUParticlesAttractor3D::get_strength() const {
  657. return strength;
  658. }
  659. void GPUParticlesAttractor3D::set_attenuation(real_t p_attenuation) {
  660. attenuation = p_attenuation;
  661. RS::get_singleton()->particles_collision_set_attractor_attenuation(collision, p_attenuation);
  662. }
  663. real_t GPUParticlesAttractor3D::get_attenuation() const {
  664. return attenuation;
  665. }
  666. void GPUParticlesAttractor3D::set_directionality(real_t p_directionality) {
  667. directionality = p_directionality;
  668. RS::get_singleton()->particles_collision_set_attractor_directionality(collision, p_directionality);
  669. update_gizmos();
  670. }
  671. real_t GPUParticlesAttractor3D::get_directionality() const {
  672. return directionality;
  673. }
  674. void GPUParticlesAttractor3D::_bind_methods() {
  675. ClassDB::bind_method(D_METHOD("set_cull_mask", "mask"), &GPUParticlesAttractor3D::set_cull_mask);
  676. ClassDB::bind_method(D_METHOD("get_cull_mask"), &GPUParticlesAttractor3D::get_cull_mask);
  677. ClassDB::bind_method(D_METHOD("set_strength", "strength"), &GPUParticlesAttractor3D::set_strength);
  678. ClassDB::bind_method(D_METHOD("get_strength"), &GPUParticlesAttractor3D::get_strength);
  679. ClassDB::bind_method(D_METHOD("set_attenuation", "attenuation"), &GPUParticlesAttractor3D::set_attenuation);
  680. ClassDB::bind_method(D_METHOD("get_attenuation"), &GPUParticlesAttractor3D::get_attenuation);
  681. ClassDB::bind_method(D_METHOD("set_directionality", "amount"), &GPUParticlesAttractor3D::set_directionality);
  682. ClassDB::bind_method(D_METHOD("get_directionality"), &GPUParticlesAttractor3D::get_directionality);
  683. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "strength", PROPERTY_HINT_RANGE, "-128,128,0.01,or_greater,or_less"), "set_strength", "get_strength");
  684. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "attenuation", PROPERTY_HINT_EXP_EASING, "0,8,0.01"), "set_attenuation", "get_attenuation");
  685. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "directionality", PROPERTY_HINT_RANGE, "0,1,0.01"), "set_directionality", "get_directionality");
  686. ADD_PROPERTY(PropertyInfo(Variant::INT, "cull_mask", PROPERTY_HINT_LAYERS_3D_RENDER), "set_cull_mask", "get_cull_mask");
  687. }
  688. GPUParticlesAttractor3D::GPUParticlesAttractor3D(RS::ParticlesCollisionType p_type) {
  689. collision = RS::get_singleton()->particles_collision_create();
  690. RS::get_singleton()->particles_collision_set_collision_type(collision, p_type);
  691. set_base(collision);
  692. }
  693. GPUParticlesAttractor3D::~GPUParticlesAttractor3D() {
  694. ERR_FAIL_NULL(RenderingServer::get_singleton());
  695. RS::get_singleton()->free(collision);
  696. }
  697. /////////////////////////////////
  698. void GPUParticlesAttractorSphere3D::_bind_methods() {
  699. ClassDB::bind_method(D_METHOD("set_radius", "radius"), &GPUParticlesAttractorSphere3D::set_radius);
  700. ClassDB::bind_method(D_METHOD("get_radius"), &GPUParticlesAttractorSphere3D::get_radius);
  701. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "radius", PROPERTY_HINT_RANGE, "0.01,1024,0.01,or_greater,suffix:m"), "set_radius", "get_radius");
  702. }
  703. void GPUParticlesAttractorSphere3D::set_radius(real_t p_radius) {
  704. radius = p_radius;
  705. RS::get_singleton()->particles_collision_set_sphere_radius(_get_collision(), radius);
  706. update_gizmos();
  707. }
  708. real_t GPUParticlesAttractorSphere3D::get_radius() const {
  709. return radius;
  710. }
  711. AABB GPUParticlesAttractorSphere3D::get_aabb() const {
  712. return AABB(Vector3(-radius, -radius, -radius), Vector3(radius * 2, radius * 2, radius * 2));
  713. }
  714. GPUParticlesAttractorSphere3D::GPUParticlesAttractorSphere3D() :
  715. GPUParticlesAttractor3D(RS::PARTICLES_COLLISION_TYPE_SPHERE_ATTRACT) {
  716. }
  717. GPUParticlesAttractorSphere3D::~GPUParticlesAttractorSphere3D() {
  718. }
  719. ///////////////////////////
  720. void GPUParticlesAttractorBox3D::_bind_methods() {
  721. ClassDB::bind_method(D_METHOD("set_size", "size"), &GPUParticlesAttractorBox3D::set_size);
  722. ClassDB::bind_method(D_METHOD("get_size"), &GPUParticlesAttractorBox3D::get_size);
  723. ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "size", PROPERTY_HINT_RANGE, "0.01,1024,0.01,or_greater,suffix:m"), "set_size", "get_size");
  724. }
  725. #ifndef DISABLE_DEPRECATED
  726. bool GPUParticlesAttractorBox3D::_set(const StringName &p_name, const Variant &p_value) {
  727. if (p_name == "extents") { // Compatibility with Godot 3.x.
  728. set_size((Vector3)p_value * 2);
  729. return true;
  730. }
  731. return false;
  732. }
  733. bool GPUParticlesAttractorBox3D::_get(const StringName &p_name, Variant &r_property) const {
  734. if (p_name == "extents") { // Compatibility with Godot 3.x.
  735. r_property = size / 2;
  736. return true;
  737. }
  738. return false;
  739. }
  740. #endif // DISABLE_DEPRECATED
  741. void GPUParticlesAttractorBox3D::set_size(const Vector3 &p_size) {
  742. size = p_size;
  743. RS::get_singleton()->particles_collision_set_box_extents(_get_collision(), size / 2);
  744. update_gizmos();
  745. }
  746. Vector3 GPUParticlesAttractorBox3D::get_size() const {
  747. return size;
  748. }
  749. AABB GPUParticlesAttractorBox3D::get_aabb() const {
  750. return AABB(-size / 2, size);
  751. }
  752. GPUParticlesAttractorBox3D::GPUParticlesAttractorBox3D() :
  753. GPUParticlesAttractor3D(RS::PARTICLES_COLLISION_TYPE_BOX_ATTRACT) {
  754. }
  755. GPUParticlesAttractorBox3D::~GPUParticlesAttractorBox3D() {
  756. }
  757. ///////////////////////////
  758. void GPUParticlesAttractorVectorField3D::_bind_methods() {
  759. ClassDB::bind_method(D_METHOD("set_size", "size"), &GPUParticlesAttractorVectorField3D::set_size);
  760. ClassDB::bind_method(D_METHOD("get_size"), &GPUParticlesAttractorVectorField3D::get_size);
  761. ClassDB::bind_method(D_METHOD("set_texture", "texture"), &GPUParticlesAttractorVectorField3D::set_texture);
  762. ClassDB::bind_method(D_METHOD("get_texture"), &GPUParticlesAttractorVectorField3D::get_texture);
  763. ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "size", PROPERTY_HINT_RANGE, "0.01,1024,0.01,or_greater,suffix:m"), "set_size", "get_size");
  764. ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "texture", PROPERTY_HINT_RESOURCE_TYPE, "Texture3D"), "set_texture", "get_texture");
  765. }
  766. #ifndef DISABLE_DEPRECATED
  767. bool GPUParticlesAttractorVectorField3D::_set(const StringName &p_name, const Variant &p_value) {
  768. if (p_name == "extents") { // Compatibility with Godot 3.x.
  769. set_size((Vector3)p_value * 2);
  770. return true;
  771. }
  772. return false;
  773. }
  774. bool GPUParticlesAttractorVectorField3D::_get(const StringName &p_name, Variant &r_property) const {
  775. if (p_name == "extents") { // Compatibility with Godot 3.x.
  776. r_property = size / 2;
  777. return true;
  778. }
  779. return false;
  780. }
  781. #endif // DISABLE_DEPRECATED
  782. void GPUParticlesAttractorVectorField3D::set_size(const Vector3 &p_size) {
  783. size = p_size;
  784. RS::get_singleton()->particles_collision_set_box_extents(_get_collision(), size / 2);
  785. update_gizmos();
  786. }
  787. Vector3 GPUParticlesAttractorVectorField3D::get_size() const {
  788. return size;
  789. }
  790. void GPUParticlesAttractorVectorField3D::set_texture(const Ref<Texture3D> &p_texture) {
  791. texture = p_texture;
  792. RID tex = texture.is_valid() ? texture->get_rid() : RID();
  793. RS::get_singleton()->particles_collision_set_field_texture(_get_collision(), tex);
  794. }
  795. Ref<Texture3D> GPUParticlesAttractorVectorField3D::get_texture() const {
  796. return texture;
  797. }
  798. AABB GPUParticlesAttractorVectorField3D::get_aabb() const {
  799. return AABB(-size / 2, size);
  800. }
  801. GPUParticlesAttractorVectorField3D::GPUParticlesAttractorVectorField3D() :
  802. GPUParticlesAttractor3D(RS::PARTICLES_COLLISION_TYPE_VECTOR_FIELD_ATTRACT) {
  803. }
  804. GPUParticlesAttractorVectorField3D::~GPUParticlesAttractorVectorField3D() {
  805. }