| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*  particles_editor_plugin.cpp                                          */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*                       This file is part of:                           */ | 
					
						
							|  |  |  | /*                           GODOT ENGINE                                */ | 
					
						
							| 
									
										
										
										
											2017-08-27 14:16:55 +02:00
										 |  |  | /*                      https://godotengine.org                          */ | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2018-01-01 14:40:08 +01:00
										 |  |  | /* Copyright (c) 2007-2018 Juan Linietsky, Ariel Manzur.                 */ | 
					
						
							|  |  |  | /* Copyright (c) 2014-2018 Godot Engine contributors (cf. AUTHORS.md)    */ | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* 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.                */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2016-10-03 16:33:42 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #include "particles_editor_plugin.h"
 | 
					
						
							| 
									
										
										
										
											2017-03-05 16:44:50 +01:00
										 |  |  | #include "editor/plugins/spatial_editor_plugin.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #include "io/resource_loader.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-07 09:04:22 -03:00
										 |  |  | #include "scene/3d/cpu_particles.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | bool ParticlesEditorBase::_generate(PoolVector<Vector3> &points, PoolVector<Vector3> &normals) { | 
					
						
							| 
									
										
										
										
											2017-04-08 22:38:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	bool use_normals = emission_fill->get_selected() == 1; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	if (emission_fill->get_selected() < 2) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		float area_accum = 0; | 
					
						
							|  |  |  | 		Map<float, int> triangle_area_map; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		for (int i = 0; i < geometry.size(); i++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-14 18:03:38 +01:00
										 |  |  | 			float area = geometry[i].get_area(); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 			if (area < CMP_EPSILON) | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 			triangle_area_map[area_accum] = i; | 
					
						
							|  |  |  | 			area_accum += area; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		if (!triangle_area_map.size() || area_accum == 0) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-03 22:25:37 -03:00
										 |  |  | 			err_dialog->set_text(TTR("Faces contain no area!")); | 
					
						
							| 
									
										
										
										
											2015-04-08 14:02:13 -03:00
										 |  |  | 			err_dialog->popup_centered_minsize(); | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 			return false; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		int emissor_count = emission_amount->get_value(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		for (int i = 0; i < emissor_count; i++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 			float areapos = Math::random(0.0f, area_accum); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 			Map<float, int>::Element *E = triangle_area_map.find_closest(areapos); | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 			ERR_FAIL_COND_V(!E, false) | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			int index = E->get(); | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 			ERR_FAIL_INDEX_V(index, geometry.size(), false); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// ok FINALLY get face
 | 
					
						
							|  |  |  | 			Face3 face = geometry[index]; | 
					
						
							|  |  |  | 			//now compute some position inside the face...
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Vector3 pos = face.get_random_point_inside(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 			points.push_back(pos); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (use_normals) { | 
					
						
							|  |  |  | 				Vector3 normal = face.get_plane().normal; | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 				normals.push_back(normal); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		int gcount = geometry.size(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		if (gcount == 0) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-19 00:08:12 +02:00
										 |  |  | 			err_dialog->set_text(TTR("No faces!")); | 
					
						
							| 
									
										
										
										
											2015-04-08 14:02:13 -03:00
										 |  |  | 			err_dialog->popup_centered_minsize(); | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 			return false; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-07 18:25:37 -03:00
										 |  |  | 		PoolVector<Face3>::Read r = geometry.read(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-16 21:09:00 -05:00
										 |  |  | 		AABB aabb; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		for (int i = 0; i < gcount; i++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 			for (int j = 0; j < 3; j++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 				if (i == 0 && j == 0) | 
					
						
							| 
									
										
										
										
											2017-06-06 20:33:51 +02:00
										 |  |  | 					aabb.position = r[i].vertex[j]; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 				else | 
					
						
							|  |  |  | 					aabb.expand_to(r[i].vertex[j]); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		int emissor_count = emission_amount->get_value(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		for (int i = 0; i < emissor_count; i++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 			int attempts = 5; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 			for (int j = 0; j < attempts; j++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				Vector3 dir; | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 				dir[Math::rand() % 3] = 1.0; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:15:36 -04:00
										 |  |  | 				Vector3 ofs = (Vector3(1, 1, 1) - dir) * Vector3(Math::randf(), Math::randf(), Math::randf()) * aabb.size + aabb.position; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				Vector3 ofsv = ofs + aabb.size * dir; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				//space it a little
 | 
					
						
							|  |  |  | 				ofs -= dir; | 
					
						
							|  |  |  | 				ofsv += dir; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 				float max = -1e7, min = 1e7; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 				for (int k = 0; k < gcount; k++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 					const Face3 &f3 = r[k]; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					Vector3 res; | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 					if (f3.intersects_segment(ofs, ofsv, &res)) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 						res -= ofs; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 						float d = dir.dot(res); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 						if (d < min) | 
					
						
							|  |  |  | 							min = d; | 
					
						
							|  |  |  | 						if (d > max) | 
					
						
							|  |  |  | 							max = d; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 				if (max < min) | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 					continue; //lost attempt
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 				float val = min + (max - min) * Math::randf(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				Vector3 point = ofs + dir * val; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 				points.push_back(point); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditorBase::_node_selected(const NodePath &p_path) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Node *sel = get_node(p_path); | 
					
						
							|  |  |  | 	if (!sel) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	VisualInstance *vi = Object::cast_to<VisualInstance>(sel); | 
					
						
							|  |  |  | 	if (!vi) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		err_dialog->set_text(TTR("Node does not contain geometry.")); | 
					
						
							|  |  |  | 		err_dialog->popup_centered_minsize(); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	geometry = vi->get_faces(VisualInstance::FACES_SOLID); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (geometry.size() == 0) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		err_dialog->set_text(TTR("Node does not contain geometry (faces).")); | 
					
						
							|  |  |  | 		err_dialog->popup_centered_minsize(); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Transform geom_xform = base_node->get_global_transform().affine_inverse() * vi->get_global_transform(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int gc = geometry.size(); | 
					
						
							|  |  |  | 	PoolVector<Face3>::Write w = geometry.write(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (int i = 0; i < gc; i++) { | 
					
						
							|  |  |  | 		for (int j = 0; j < 3; j++) { | 
					
						
							|  |  |  | 			w[i].vertex[j] = geom_xform.xform(w[i].vertex[j]); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	w = PoolVector<Face3>::Write(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	emission_dialog->popup_centered(Size2(300, 130)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditorBase::_bind_methods() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ClassDB::bind_method("_node_selected", &ParticlesEditorBase::_node_selected); | 
					
						
							|  |  |  | 	ClassDB::bind_method("_generate_emission_points", &ParticlesEditorBase::_generate_emission_points); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ParticlesEditorBase::ParticlesEditorBase() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	emission_dialog = memnew(ConfirmationDialog); | 
					
						
							|  |  |  | 	emission_dialog->set_title(TTR("Create Emitter")); | 
					
						
							|  |  |  | 	add_child(emission_dialog); | 
					
						
							|  |  |  | 	VBoxContainer *emd_vb = memnew(VBoxContainer); | 
					
						
							|  |  |  | 	emission_dialog->add_child(emd_vb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	emission_amount = memnew(SpinBox); | 
					
						
							|  |  |  | 	emission_amount->set_min(1); | 
					
						
							|  |  |  | 	emission_amount->set_max(100000); | 
					
						
							|  |  |  | 	emission_amount->set_value(512); | 
					
						
							|  |  |  | 	emd_vb->add_margin_child(TTR("Emission Points:"), emission_amount); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	emission_fill = memnew(OptionButton); | 
					
						
							|  |  |  | 	emission_fill->add_item(TTR("Surface Points")); | 
					
						
							|  |  |  | 	emission_fill->add_item(TTR("Surface Points+Normal (Directed)")); | 
					
						
							|  |  |  | 	emission_fill->add_item(TTR("Volume")); | 
					
						
							|  |  |  | 	emd_vb->add_margin_child(TTR("Emission Source: "), emission_fill); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	emission_dialog->get_ok()->set_text(TTR("Create")); | 
					
						
							|  |  |  | 	emission_dialog->connect("confirmed", this, "_generate_emission_points"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err_dialog = memnew(ConfirmationDialog); | 
					
						
							|  |  |  | 	add_child(err_dialog); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	emission_file_dialog = memnew(EditorFileDialog); | 
					
						
							|  |  |  | 	add_child(emission_file_dialog); | 
					
						
							|  |  |  | 	emission_file_dialog->connect("file_selected", this, "_resource_seleted"); | 
					
						
							|  |  |  | 	emission_tree_dialog = memnew(SceneTreeDialog); | 
					
						
							|  |  |  | 	add_child(emission_tree_dialog); | 
					
						
							|  |  |  | 	emission_tree_dialog->connect("selected", this, "_node_selected"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	List<String> extensions; | 
					
						
							|  |  |  | 	ResourceLoader::get_recognized_extensions_for_type("Mesh", &extensions); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	emission_file_dialog->clear_filters(); | 
					
						
							|  |  |  | 	for (int i = 0; i < extensions.size(); i++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		emission_file_dialog->add_filter("*." + extensions[i] + " ; " + extensions[i].to_upper()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	emission_file_dialog->set_mode(EditorFileDialog::MODE_OPEN_FILE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditor::_node_removed(Node *p_node) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_node == node) { | 
					
						
							|  |  |  | 		node = NULL; | 
					
						
							|  |  |  | 		hide(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditor::_notification(int p_notification) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_notification == NOTIFICATION_ENTER_TREE) { | 
					
						
							|  |  |  | 		options->set_icon(options->get_popup()->get_icon("Particles", "EditorIcons")); | 
					
						
							| 
									
										
										
										
											2018-08-24 08:28:53 -03:00
										 |  |  | 		get_tree()->connect("node_removed", this, "_node_removed"); | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditor::_menu_option(int p_option) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (p_option) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case MENU_OPTION_GENERATE_AABB: { | 
					
						
							| 
									
										
										
										
											2018-08-13 17:14:56 +02:00
										 |  |  | 			float gen_time = node->get_lifetime(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (gen_time < 1.0) | 
					
						
							|  |  |  | 				generate_seconds->set_value(1.0); | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				generate_seconds->set_value(trunc(gen_time) + 1.0); | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 			generate_aabb->popup_centered_minsize(); | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		case MENU_OPTION_CREATE_EMISSION_VOLUME_FROM_MESH: { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Ref<ParticlesMaterial> material = node->get_process_material(); | 
					
						
							|  |  |  | 			if (material.is_null()) { | 
					
						
							|  |  |  | 				EditorNode::get_singleton()->show_warning(TTR("A processor material of type 'ParticlesMaterial' is required.")); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			emission_file_dialog->popup_centered_ratio(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case MENU_OPTION_CREATE_EMISSION_VOLUME_FROM_NODE: { | 
					
						
							|  |  |  | 			Ref<ParticlesMaterial> material = node->get_process_material(); | 
					
						
							|  |  |  | 			if (material.is_null()) { | 
					
						
							|  |  |  | 				EditorNode::get_singleton()->show_warning(TTR("A processor material of type 'ParticlesMaterial' is required.")); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			emission_tree_dialog->popup_centered_ratio(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2018-07-07 09:04:22 -03:00
										 |  |  | 		case MENU_OPTION_CONVERT_TO_CPU_PARTICLES: { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			UndoRedo *undo_redo = EditorNode::get_singleton()->get_undo_redo(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			CPUParticles *cpu_particles = memnew(CPUParticles); | 
					
						
							|  |  |  | 			cpu_particles->convert_from_particles(node); | 
					
						
							| 
									
										
										
										
											2018-07-23 12:28:15 -03:00
										 |  |  | 			cpu_particles->set_name(node->get_name()); | 
					
						
							|  |  |  | 			cpu_particles->set_transform(node->get_transform()); | 
					
						
							|  |  |  | 			cpu_particles->set_visible(node->is_visible()); | 
					
						
							|  |  |  | 			cpu_particles->set_pause_mode(node->get_pause_mode()); | 
					
						
							| 
									
										
										
										
											2018-07-07 09:04:22 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			undo_redo->create_action("Replace Particles by CPUParticles"); | 
					
						
							|  |  |  | 			undo_redo->add_do_method(node, "replace_by", cpu_particles); | 
					
						
							|  |  |  | 			undo_redo->add_undo_method(cpu_particles, "replace_by", node); | 
					
						
							|  |  |  | 			undo_redo->add_do_reference(cpu_particles); | 
					
						
							|  |  |  | 			undo_redo->add_undo_reference(node); | 
					
						
							|  |  |  | 			undo_redo->commit_action(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditor::_generate_aabb() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	float time = generate_seconds->get_value(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	float running = 0.0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	EditorProgress ep("gen_aabb", TTR("Generating AABB"), int(time)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 17:14:56 +02:00
										 |  |  | 	bool was_emitting = node->is_emitting(); | 
					
						
							|  |  |  | 	if (!was_emitting) { | 
					
						
							|  |  |  | 		node->set_emitting(true); | 
					
						
							|  |  |  | 		OS::get_singleton()->delay_usec(1000); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 	AABB rect; | 
					
						
							| 
									
										
										
										
											2018-08-13 17:14:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 	while (running < time) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint64_t ticks = OS::get_singleton()->get_ticks_usec(); | 
					
						
							|  |  |  | 		ep.step("Generating...", int(running), true); | 
					
						
							|  |  |  | 		OS::get_singleton()->delay_usec(1000); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		AABB capture = node->capture_aabb(); | 
					
						
							|  |  |  | 		if (rect == AABB()) | 
					
						
							|  |  |  | 			rect = capture; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			rect.merge_with(capture); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		running += (OS::get_singleton()->get_ticks_usec() - ticks) / 1000000.0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-13 17:14:56 +02:00
										 |  |  | 	if (!was_emitting) { | 
					
						
							|  |  |  | 		node->set_emitting(false); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 	node->set_visibility_aabb(rect); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditor::edit(Particles *p_particles) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	base_node = p_particles; | 
					
						
							|  |  |  | 	node = p_particles; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditor::_generate_emission_points() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/// hacer codigo aca
 | 
					
						
							|  |  |  | 	PoolVector<Vector3> points; | 
					
						
							|  |  |  | 	PoolVector<Vector3> normals; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!_generate(points, normals)) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int point_count = points.size(); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	int w = 2048; | 
					
						
							|  |  |  | 	int h = (point_count / 2048) + 1; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	PoolVector<uint8_t> point_img; | 
					
						
							|  |  |  | 	point_img.resize(w * h * 3 * sizeof(float)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		PoolVector<uint8_t>::Write iw = point_img.write(); | 
					
						
							|  |  |  | 		zeromem(iw.ptr(), w * h * 3 * sizeof(float)); | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 		PoolVector<Vector3>::Read r = points.read(); | 
					
						
							|  |  |  | 		float *wf = (float *)iw.ptr(); | 
					
						
							|  |  |  | 		for (int i = 0; i < point_count; i++) { | 
					
						
							|  |  |  | 			wf[i * 3 + 0] = r[i].x; | 
					
						
							|  |  |  | 			wf[i * 3 + 1] = r[i].y; | 
					
						
							|  |  |  | 			wf[i * 3 + 2] = r[i].z; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-17 07:36:47 -03:00
										 |  |  | 	Ref<Image> image = memnew(Image(w, h, false, Image::FORMAT_RGBF, point_img)); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Ref<ImageTexture> tex; | 
					
						
							|  |  |  | 	tex.instance(); | 
					
						
							|  |  |  | 	tex->create_from_image(image, Texture::FLAG_FILTER); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Ref<ParticlesMaterial> material = node->get_process_material(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(material.is_null()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 	if (normals.size() > 0) { | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		material->set_emission_shape(ParticlesMaterial::EMISSION_SHAPE_DIRECTED_POINTS); | 
					
						
							|  |  |  | 		material->set_emission_point_count(point_count); | 
					
						
							|  |  |  | 		material->set_emission_point_texture(tex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		PoolVector<uint8_t> point_img2; | 
					
						
							|  |  |  | 		point_img2.resize(w * h * 3 * sizeof(float)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			PoolVector<uint8_t>::Write iw = point_img2.write(); | 
					
						
							|  |  |  | 			zeromem(iw.ptr(), w * h * 3 * sizeof(float)); | 
					
						
							| 
									
										
										
										
											2018-07-06 20:21:13 -03:00
										 |  |  | 			PoolVector<Vector3>::Read r = normals.read(); | 
					
						
							|  |  |  | 			float *wf = (float *)iw.ptr(); | 
					
						
							|  |  |  | 			for (int i = 0; i < point_count; i++) { | 
					
						
							|  |  |  | 				wf[i * 3 + 0] = r[i].x; | 
					
						
							|  |  |  | 				wf[i * 3 + 1] = r[i].y; | 
					
						
							|  |  |  | 				wf[i * 3 + 2] = r[i].z; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-17 07:36:47 -03:00
										 |  |  | 		Ref<Image> image2 = memnew(Image(w, h, false, Image::FORMAT_RGBF, point_img2)); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		Ref<ImageTexture> tex2; | 
					
						
							|  |  |  | 		tex2.instance(); | 
					
						
							|  |  |  | 		tex2->create_from_image(image2, Texture::FLAG_FILTER); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		material->set_emission_normal_texture(tex2); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		material->set_emission_shape(ParticlesMaterial::EMISSION_SHAPE_POINTS); | 
					
						
							|  |  |  | 		material->set_emission_point_count(point_count); | 
					
						
							|  |  |  | 		material->set_emission_point_texture(tex); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditor::_bind_methods() { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	ClassDB::bind_method("_menu_option", &ParticlesEditor::_menu_option); | 
					
						
							| 
									
										
										
										
											2017-04-08 22:38:11 -03:00
										 |  |  | 	ClassDB::bind_method("_generate_aabb", &ParticlesEditor::_generate_aabb); | 
					
						
							| 
									
										
										
										
											2018-08-24 08:28:53 -03:00
										 |  |  | 	ClassDB::bind_method("_node_removed", &ParticlesEditor::_node_removed); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ParticlesEditor::ParticlesEditor() { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	particles_editor_hb = memnew(HBoxContainer); | 
					
						
							| 
									
										
										
										
											2014-07-10 10:55:03 +08:00
										 |  |  | 	SpatialEditor::get_singleton()->add_control_to_menu_panel(particles_editor_hb); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	options = memnew(MenuButton); | 
					
						
							| 
									
										
										
										
											2014-07-10 10:55:03 +08:00
										 |  |  | 	particles_editor_hb->add_child(options); | 
					
						
							|  |  |  | 	particles_editor_hb->hide(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-23 22:25:14 +02:00
										 |  |  | 	options->set_text(TTR("Particles")); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	options->get_popup()->add_item(TTR("Generate AABB"), MENU_OPTION_GENERATE_AABB); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	options->get_popup()->add_separator(); | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	options->get_popup()->add_item(TTR("Create Emission Points From Mesh"), MENU_OPTION_CREATE_EMISSION_VOLUME_FROM_MESH); | 
					
						
							|  |  |  | 	options->get_popup()->add_item(TTR("Create Emission Points From Node"), MENU_OPTION_CREATE_EMISSION_VOLUME_FROM_NODE); | 
					
						
							| 
									
										
										
										
											2018-07-07 09:04:22 -03:00
										 |  |  | 	options->get_popup()->add_separator(); | 
					
						
							|  |  |  | 	options->get_popup()->add_item(TTR("Convert to CPUParticles"), MENU_OPTION_CONVERT_TO_CPU_PARTICLES); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	options->get_popup()->connect("id_pressed", this, "_menu_option"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-08 22:38:11 -03:00
										 |  |  | 	generate_aabb = memnew(ConfirmationDialog); | 
					
						
							|  |  |  | 	generate_aabb->set_title(TTR("Generate Visibility AABB")); | 
					
						
							|  |  |  | 	VBoxContainer *genvb = memnew(VBoxContainer); | 
					
						
							|  |  |  | 	generate_aabb->add_child(genvb); | 
					
						
							|  |  |  | 	generate_seconds = memnew(SpinBox); | 
					
						
							|  |  |  | 	genvb->add_margin_child(TTR("Generation Time (sec):"), generate_seconds); | 
					
						
							|  |  |  | 	generate_seconds->set_min(0.1); | 
					
						
							|  |  |  | 	generate_seconds->set_max(25); | 
					
						
							|  |  |  | 	generate_seconds->set_value(2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	add_child(generate_aabb); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	generate_aabb->connect("confirmed", this, "_generate_aabb"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditorPlugin::edit(Object *p_object) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-24 22:58:51 +02:00
										 |  |  | 	particles_editor->edit(Object::cast_to<Particles>(p_object)); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ParticlesEditorPlugin::handles(Object *p_object) const { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	return p_object->is_class("Particles"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ParticlesEditorPlugin::make_visible(bool p_visible) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_visible) { | 
					
						
							|  |  |  | 		particles_editor->show(); | 
					
						
							| 
									
										
										
										
											2014-07-10 10:55:03 +08:00
										 |  |  | 		particles_editor->particles_editor_hb->show(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2014-07-10 10:55:03 +08:00
										 |  |  | 		particles_editor->particles_editor_hb->hide(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		particles_editor->hide(); | 
					
						
							|  |  |  | 		particles_editor->edit(NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ParticlesEditorPlugin::ParticlesEditorPlugin(EditorNode *p_node) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | 	editor = p_node; | 
					
						
							|  |  |  | 	particles_editor = memnew(ParticlesEditor); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	editor->get_viewport()->add_child(particles_editor); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	particles_editor->hide(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-06 23:36:37 -03:00
										 |  |  | ParticlesEditorPlugin::~ParticlesEditorPlugin() { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } |