| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*  tile_map.cpp                                                         */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*                       This file is part of:                           */ | 
					
						
							|  |  |  | /*                           GODOT ENGINE                                */ | 
					
						
							|  |  |  | /*                    http://www.godotengine.org                         */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2017-01-01 22:01:57 +01:00
										 |  |  | /* Copyright (c) 2007-2017 Juan Linietsky, Ariel Manzur.                 */ | 
					
						
							| 
									
										
										
										
											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.                */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | #include "tile_map.h"
 | 
					
						
							|  |  |  | #include "io/marshalls.h"
 | 
					
						
							|  |  |  | #include "servers/physics_2d_server.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-19 23:07:25 +10:00
										 |  |  | #include "method_bind_ext.inc"
 | 
					
						
							| 
									
										
										
										
											2015-09-20 13:03:46 -03:00
										 |  |  | #include "os/os.h"
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-12 10:45:31 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | int TileMap::_get_quadrant_size() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (y_sort_mode) | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return quadrant_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void TileMap::_notification(int p_what) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch(p_what) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 21:20:42 -03:00
										 |  |  | 		case NOTIFICATION_ENTER_TREE: { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			Node2D *c=this; | 
					
						
							|  |  |  | 			while(c) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				navigation=c->cast_to<Navigation2D>(); | 
					
						
							|  |  |  | 				if (navigation) { | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				c=c->get_parent()->cast_to<Node2D>(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			pending_update=true; | 
					
						
							|  |  |  | 			_update_dirty_quadrants(); | 
					
						
							|  |  |  | 			RID space = get_world_2d()->get_space(); | 
					
						
							|  |  |  | 			_update_quadrant_transform(); | 
					
						
							|  |  |  | 			_update_quadrant_space(space); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							| 
									
										
										
										
											2014-11-05 21:20:42 -03:00
										 |  |  | 		case NOTIFICATION_EXIT_TREE: { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			_update_quadrant_space(RID()); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Quadrant &q=E->get(); | 
					
						
							|  |  |  | 				if (navigation) { | 
					
						
							|  |  |  | 					for(Map<PosKey,Quadrant::NavPoly>::Element *E=q.navpoly_ids.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						navigation->navpoly_remove(E->get().id); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					q.navpoly_ids.clear(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				for(Map<PosKey,Quadrant::Occluder>::Element *E=q.occluder_instances.front();E;E=E->next()) { | 
					
						
							|  |  |  | 					VS::get_singleton()->free(E->get().id); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				q.occluder_instances.clear(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			navigation=NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		case NOTIFICATION_TRANSFORM_CHANGED: { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			//move stuff
 | 
					
						
							|  |  |  | 			_update_quadrant_transform(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::_update_quadrant_space(const RID& p_space) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Quadrant &q=E->get(); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 		Physics2DServer::get_singleton()->body_set_space(q.body,p_space); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::_update_quadrant_transform() { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 21:20:42 -03:00
										 |  |  | 	if (!is_inside_tree()) | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 	Transform2D global_transform = get_global_transform(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 	Transform2D nav_rel; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	if (navigation) | 
					
						
							| 
									
										
										
										
											2015-12-28 21:05:57 -03:00
										 |  |  | 		nav_rel = get_relative_transform_to_parent(navigation); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Quadrant &q=E->get(); | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 		Transform2D xform; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		xform.set_origin( q.pos ); | 
					
						
							|  |  |  | 		xform = global_transform * xform; | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 		Physics2DServer::get_singleton()->body_set_state(q.body,Physics2DServer::BODY_STATE_TRANSFORM,xform); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (navigation) { | 
					
						
							|  |  |  | 			for(Map<PosKey,Quadrant::NavPoly>::Element *E=q.navpoly_ids.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				navigation->navpoly_set_transform(E->get().id,nav_rel * E->get().xform); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for(Map<PosKey,Quadrant::Occluder>::Element *E=q.occluder_instances.front();E;E=E->next()) { | 
					
						
							|  |  |  | 			VS::get_singleton()->canvas_light_occluder_set_transform(E->get().id,global_transform * E->get().xform); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::set_tileset(const Ref<TileSet>& p_tileset) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (tile_set.is_valid()) | 
					
						
							|  |  |  | 		tile_set->disconnect("changed",this,"_recreate_quadrants"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	tile_set=p_tileset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (tile_set.is_valid()) | 
					
						
							|  |  |  | 		tile_set->connect("changed",this,"_recreate_quadrants"); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							| 
									
										
										
										
											2014-04-19 16:46:52 -03:00
										 |  |  | 	emit_signal("settings_changed"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Ref<TileSet> TileMap::get_tileset() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return tile_set; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | void TileMap::set_cell_size(Size2 p_size) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	ERR_FAIL_COND(p_size.x<1 || p_size.y<1); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	cell_size=p_size; | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							| 
									
										
										
										
											2014-04-19 16:46:52 -03:00
										 |  |  | 	emit_signal("settings_changed"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | Size2 TileMap::get_cell_size() const { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return cell_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | void TileMap::set_quadrant_size(int p_size) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ERR_FAIL_COND(p_size<1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	quadrant_size=p_size; | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							| 
									
										
										
										
											2014-04-19 16:46:52 -03:00
										 |  |  | 	emit_signal("settings_changed"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | int TileMap::get_quadrant_size() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return quadrant_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::set_center_x(bool p_enable) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	center_x=p_enable; | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							| 
									
										
										
										
											2014-04-19 16:46:52 -03:00
										 |  |  | 	emit_signal("settings_changed"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | bool TileMap::get_center_x() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return center_x; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | void TileMap::set_center_y(bool p_enable) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	center_y=p_enable; | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							| 
									
										
										
										
											2014-04-19 16:46:52 -03:00
										 |  |  | 	emit_signal("settings_changed"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | bool TileMap::get_center_y() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return center_y; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | void TileMap::_fix_cell_transform(Transform2D& xform,const Cell& p_cell, const Vector2& p_offset, const Size2 &p_sc) { | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 16:28:08 +08:00
										 |  |  | 	Size2 s=p_sc; | 
					
						
							|  |  |  | 	Vector2 offset = p_offset; | 
					
						
							| 
									
										
										
										
											2016-06-18 00:01:00 +02:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if (tile_origin==TILE_ORIGIN_BOTTOM_LEFT) | 
					
						
							|  |  |  | 		offset.y+=cell_size.y; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-04 15:28:11 +01:00
										 |  |  | 	if (s.y > s.x) { | 
					
						
							|  |  |  | 		if ((p_cell.flip_h && (p_cell.flip_v || p_cell.transpose)) || (p_cell.flip_v && !p_cell.transpose)) | 
					
						
							|  |  |  | 			offset.y += s.y - s.x; | 
					
						
							|  |  |  | 	} else if (s.y < s.x) { | 
					
						
							|  |  |  | 		if ((p_cell.flip_v && (p_cell.flip_h || p_cell.transpose)) || (p_cell.flip_h && !p_cell.transpose)) | 
					
						
							|  |  |  | 			offset.x += s.x - s.y; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 16:28:08 +08:00
										 |  |  | 	if (p_cell.transpose) { | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 		SWAP(xform.elements[0].x, xform.elements[0].y); | 
					
						
							|  |  |  | 		SWAP(xform.elements[1].x, xform.elements[1].y); | 
					
						
							|  |  |  | 		SWAP(offset.x, offset.y); | 
					
						
							|  |  |  | 		SWAP(s.x, s.y); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-09 16:28:08 +08:00
										 |  |  | 	if (p_cell.flip_h) { | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 		xform.elements[0].x=-xform.elements[0].x; | 
					
						
							|  |  |  | 		xform.elements[1].x=-xform.elements[1].x; | 
					
						
							| 
									
										
										
										
											2016-06-18 00:01:00 +02:00
										 |  |  | 		if (tile_origin==TILE_ORIGIN_TOP_LEFT || tile_origin==TILE_ORIGIN_BOTTOM_LEFT) | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			offset.x=s.x-offset.x; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-09 16:28:08 +08:00
										 |  |  | 	if (p_cell.flip_v) { | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 		xform.elements[0].y=-xform.elements[0].y; | 
					
						
							|  |  |  | 		xform.elements[1].y=-xform.elements[1].y; | 
					
						
							|  |  |  | 		if (tile_origin==TILE_ORIGIN_TOP_LEFT) | 
					
						
							|  |  |  | 			offset.y=s.y-offset.y; | 
					
						
							| 
									
										
										
										
											2016-06-18 00:01:00 +02:00
										 |  |  | 		else if (tile_origin==TILE_ORIGIN_BOTTOM_LEFT) { | 
					
						
							|  |  |  | 			if(p_cell.transpose) | 
					
						
							|  |  |  | 				offset.y+=s.y; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				offset.y-=s.y; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	xform.elements[2].x+=offset.x; | 
					
						
							|  |  |  | 	xform.elements[2].y+=offset.y; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void TileMap::_update_dirty_quadrants() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!pending_update) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2017-01-01 18:31:29 +01:00
										 |  |  | 	if (!is_inside_tree() || !tile_set.is_valid()) { | 
					
						
							|  |  |  | 		pending_update = false; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2017-01-01 18:31:29 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	VisualServer *vs = VisualServer::get_singleton(); | 
					
						
							|  |  |  | 	Physics2DServer *ps = Physics2DServer::get_singleton(); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	Vector2 tofs = get_cell_draw_offset(); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	Vector2 tcenter = cell_size/2; | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 	Transform2D nav_rel; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	if (navigation) | 
					
						
							| 
									
										
										
										
											2015-12-28 21:05:57 -03:00
										 |  |  | 		nav_rel = get_relative_transform_to_parent(navigation); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Vector2 qofs; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-20 13:03:46 -03:00
										 |  |  | 	SceneTree *st=SceneTree::get_singleton(); | 
					
						
							|  |  |  | 	Color debug_collision_color; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool debug_shapes = st && st->is_debugging_collisions_hint(); | 
					
						
							|  |  |  | 	if (debug_shapes) { | 
					
						
							|  |  |  | 		debug_collision_color=st->get_debug_collisions_color(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	while (dirty_quadrant_list.first()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Quadrant &q = *dirty_quadrant_list.first()->self(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 		for (List<RID>::Element *E=q.canvas_items.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			vs->free(E->get()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		q.canvas_items.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 		ps->body_clear_shapes(q.body); | 
					
						
							| 
									
										
										
										
											2014-10-16 00:06:34 -03:00
										 |  |  | 		int shape_idx=0; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 		if (navigation) { | 
					
						
							|  |  |  | 			for(Map<PosKey,Quadrant::NavPoly>::Element *E=q.navpoly_ids.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				navigation->navpoly_remove(E->get().id); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			q.navpoly_ids.clear(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for(Map<PosKey,Quadrant::Occluder>::Element *E=q.occluder_instances.front();E;E=E->next()) { | 
					
						
							|  |  |  | 			VS::get_singleton()->free(E->get().id); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		q.occluder_instances.clear(); | 
					
						
							|  |  |  | 		Ref<CanvasItemMaterial> prev_material; | 
					
						
							|  |  |  | 		RID prev_canvas_item; | 
					
						
							| 
									
										
										
										
											2015-12-12 08:11:36 -03:00
										 |  |  | 		RID prev_debug_canvas_item; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		for(int i=0;i<q.cells.size();i++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Map<PosKey,Cell>::Element *E=tile_map.find( q.cells[i] ); | 
					
						
							|  |  |  | 			Cell &c=E->get(); | 
					
						
							|  |  |  | 			//moment of truth
 | 
					
						
							|  |  |  | 			if (!tile_set->has_tile(c.id)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			Ref<Texture> tex = tile_set->tile_get_texture(c.id); | 
					
						
							| 
									
										
										
										
											2014-02-19 11:57:14 -03:00
										 |  |  | 			Vector2 tile_ofs = tile_set->tile_get_texture_offset(c.id); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			Vector2 wofs = _map_to_world(E->key().x, E->key().y); | 
					
						
							|  |  |  | 			Vector2 offset = wofs - q.pos + tofs; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (!tex.is_valid()) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			Ref<CanvasItemMaterial> mat = tile_set->tile_get_material(c.id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			RID canvas_item; | 
					
						
							| 
									
										
										
										
											2015-12-12 08:11:36 -03:00
										 |  |  | 			RID debug_canvas_item; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (prev_canvas_item==RID() || prev_material!=mat) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				canvas_item=vs->canvas_item_create(); | 
					
						
							|  |  |  | 				if (mat.is_valid()) | 
					
						
							|  |  |  | 					vs->canvas_item_set_material(canvas_item,mat->get_rid()); | 
					
						
							|  |  |  | 				vs->canvas_item_set_parent( canvas_item, get_canvas_item() ); | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 				Transform2D xform; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 				xform.set_origin( q.pos ); | 
					
						
							|  |  |  | 				vs->canvas_item_set_transform( canvas_item, xform ); | 
					
						
							| 
									
										
										
										
											2015-12-29 14:47:13 -03:00
										 |  |  | 				vs->canvas_item_set_light_mask(canvas_item,get_light_mask()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 				q.canvas_items.push_back(canvas_item); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-12 08:11:36 -03:00
										 |  |  | 				if (debug_shapes) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					debug_canvas_item=vs->canvas_item_create(); | 
					
						
							|  |  |  | 					vs->canvas_item_set_parent( debug_canvas_item, canvas_item ); | 
					
						
							|  |  |  | 					vs->canvas_item_set_z_as_relative_to_parent(debug_canvas_item,false); | 
					
						
							|  |  |  | 					vs->canvas_item_set_z(debug_canvas_item,VS::CANVAS_ITEM_Z_MAX-1); | 
					
						
							|  |  |  | 					q.canvas_items.push_back(debug_canvas_item); | 
					
						
							|  |  |  | 					prev_debug_canvas_item=debug_canvas_item; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 				prev_canvas_item=canvas_item; | 
					
						
							|  |  |  | 				prev_material=mat; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				canvas_item=prev_canvas_item; | 
					
						
							| 
									
										
										
										
											2015-12-12 08:11:36 -03:00
										 |  |  | 				if (debug_shapes) { | 
					
						
							|  |  |  | 					debug_canvas_item=prev_debug_canvas_item; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			Rect2 r = tile_set->tile_get_region(c.id); | 
					
						
							|  |  |  | 			Size2 s = tex->get_size(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (r==Rect2()) | 
					
						
							|  |  |  | 				s = tex->get_size(); | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				s = r.size; | 
					
						
							|  |  |  | 				r.pos.x+=fp_adjust; | 
					
						
							|  |  |  | 				r.pos.y+=fp_adjust; | 
					
						
							|  |  |  | 				r.size.x-=fp_adjust*2.0; | 
					
						
							|  |  |  | 				r.size.y-=fp_adjust*2.0; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Rect2 rect; | 
					
						
							|  |  |  | 			rect.pos=offset.floor(); | 
					
						
							|  |  |  | 			rect.size=s; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-04 15:28:11 +01:00
										 |  |  | 			if (rect.size.y > rect.size.x) { | 
					
						
							|  |  |  | 				if ((c.flip_h && (c.flip_v || c.transpose)) || (c.flip_v && !c.transpose)) | 
					
						
							|  |  |  | 					tile_ofs.y += rect.size.y - rect.size.x; | 
					
						
							|  |  |  | 			} else if (rect.size.y < rect.size.x) { | 
					
						
							|  |  |  | 				if ((c.flip_v && (c.flip_h || c.transpose)) || (c.flip_h && !c.transpose)) | 
					
						
							|  |  |  | 					tile_ofs.x += rect.size.x - rect.size.y; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 16:17:33 -03:00
										 |  |  | 		/*	rect.size.x+=fp_adjust;
 | 
					
						
							|  |  |  | 			rect.size.y+=fp_adjust;*/ | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-02 18:17:20 +01:00
										 |  |  | 			if (c.transpose) | 
					
						
							|  |  |  | 				SWAP(tile_ofs.x, tile_ofs.y); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (c.flip_h) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 				rect.size.x=-rect.size.x; | 
					
						
							| 
									
										
										
										
											2016-01-02 18:17:20 +01:00
										 |  |  | 				tile_ofs.x=-tile_ofs.x; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (c.flip_v) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 				rect.size.y=-rect.size.y; | 
					
						
							| 
									
										
										
										
											2016-01-02 18:17:20 +01:00
										 |  |  | 				tile_ofs.y=-tile_ofs.y; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			Vector2 center_ofs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (tile_origin==TILE_ORIGIN_TOP_LEFT) { | 
					
						
							|  |  |  | 				rect.pos+=tile_ofs; | 
					
						
							| 
									
										
										
										
											2016-06-18 00:01:00 +02:00
										 |  |  | 				 | 
					
						
							|  |  |  | 			} else if (tile_origin==TILE_ORIGIN_BOTTOM_LEFT) { | 
					
						
							|  |  |  | 								 | 
					
						
							|  |  |  | 				rect.pos+=tile_ofs; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if(c.transpose) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					if(c.flip_h) | 
					
						
							|  |  |  | 						rect.pos.x-=cell_size.x; | 
					
						
							|  |  |  | 					else | 
					
						
							|  |  |  | 						rect.pos.x+=cell_size.x; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					if(c.flip_v) | 
					
						
							|  |  |  | 						rect.pos.y-=cell_size.y; | 
					
						
							|  |  |  | 					else | 
					
						
							|  |  |  | 						rect.pos.y+=cell_size.y; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			} else if (tile_origin==TILE_ORIGIN_CENTER) { | 
					
						
							|  |  |  | 				rect.pos+=tcenter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Vector2 center = (s/2) - tile_ofs; | 
					
						
							|  |  |  | 				center_ofs=tcenter-(s/2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (c.flip_h) | 
					
						
							|  |  |  | 					rect.pos.x-=s.x-center.x; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					rect.pos.x-=center.x; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (c.flip_v) | 
					
						
							|  |  |  | 					rect.pos.y-=s.y-center.y; | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					rect.pos.y-=center.y; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (r==Rect2()) { | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 				tex->draw_rect(canvas_item,rect,false,Color(1,1,1),c.transpose); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 				tex->draw_rect_region(canvas_item,rect,r,Color(1,1,1),c.transpose); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Vector< Ref<Shape2D> > shapes = tile_set->tile_get_shapes(c.id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for(int i=0;i<shapes.size();i++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Ref<Shape2D> shape = shapes[i]; | 
					
						
							|  |  |  | 				if (shape.is_valid()) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-19 11:57:14 -03:00
										 |  |  | 					Vector2 shape_ofs = tile_set->tile_get_shape_offset(c.id); | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 					Transform2D xform; | 
					
						
							| 
									
										
										
										
											2014-04-19 16:46:52 -03:00
										 |  |  | 					xform.set_origin(offset.floor()); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 					_fix_cell_transform(xform,c,shape_ofs+center_ofs,s); | 
					
						
							| 
									
										
										
										
											2014-02-19 11:57:14 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-03 16:33:42 -03:00
										 |  |  | 					if (debug_canvas_item.is_valid()) { | 
					
						
							| 
									
										
										
										
											2016-02-20 10:01:34 -03:00
										 |  |  | 						vs->canvas_item_add_set_transform(debug_canvas_item,xform); | 
					
						
							| 
									
										
										
										
											2015-12-12 08:11:36 -03:00
										 |  |  | 						shape->draw(debug_canvas_item,debug_collision_color); | 
					
						
							| 
									
										
										
										
											2015-09-20 13:03:46 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2016-03-09 00:00:52 +01:00
										 |  |  | 					ps->body_add_shape(q.body,shape->get_rid(),xform); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 					ps->body_set_shape_metadata(q.body,shape_idx++,Vector2(E->key().x,E->key().y)); | 
					
						
							| 
									
										
										
										
											2014-10-16 00:06:34 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-03 16:33:42 -03:00
										 |  |  | 			if (debug_canvas_item.is_valid()) { | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 				vs->canvas_item_add_set_transform(debug_canvas_item,Transform2D()); | 
					
						
							| 
									
										
										
										
											2016-02-20 10:01:34 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			if (navigation) { | 
					
						
							|  |  |  | 				Ref<NavigationPolygon> navpoly = tile_set->tile_get_navigation_polygon(c.id); | 
					
						
							| 
									
										
										
										
											2015-04-09 01:18:01 -03:00
										 |  |  | 				if (navpoly.is_valid()) { | 
					
						
							|  |  |  | 					Vector2 npoly_ofs = tile_set->tile_get_navigation_polygon_offset(c.id); | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 					Transform2D xform; | 
					
						
							| 
									
										
										
										
											2015-04-09 01:18:01 -03:00
										 |  |  | 					xform.set_origin(offset.floor()+q.pos); | 
					
						
							|  |  |  | 					_fix_cell_transform(xform,c,npoly_ofs+center_ofs,s); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-20 13:03:46 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-09 01:18:01 -03:00
										 |  |  | 					int pid = navigation->navpoly_create(navpoly,nav_rel * xform); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-09 01:18:01 -03:00
										 |  |  | 					Quadrant::NavPoly np; | 
					
						
							|  |  |  | 					np.id=pid; | 
					
						
							|  |  |  | 					np.xform=xform; | 
					
						
							|  |  |  | 					q.navpoly_ids[E->key()]=np; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Ref<OccluderPolygon2D> occluder=tile_set->tile_get_light_occluder(c.id); | 
					
						
							|  |  |  | 			if (occluder.is_valid()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Vector2 occluder_ofs = tile_set->tile_get_occluder_offset(c.id); | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 				Transform2D xform; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 				xform.set_origin(offset.floor()+q.pos); | 
					
						
							|  |  |  | 				_fix_cell_transform(xform,c,occluder_ofs+center_ofs,s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				RID orid = VS::get_singleton()->canvas_light_occluder_create(); | 
					
						
							|  |  |  | 				VS::get_singleton()->canvas_light_occluder_set_transform(orid,get_global_transform() * xform); | 
					
						
							|  |  |  | 				VS::get_singleton()->canvas_light_occluder_set_polygon(orid,occluder->get_rid()); | 
					
						
							|  |  |  | 				VS::get_singleton()->canvas_light_occluder_attach_to_canvas(orid,get_canvas()); | 
					
						
							| 
									
										
										
										
											2015-12-12 10:45:31 -03:00
										 |  |  | 				VS::get_singleton()->canvas_light_occluder_set_light_mask(orid,occluder_light_mask); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 				Quadrant::Occluder oc; | 
					
						
							|  |  |  | 				oc.xform=xform; | 
					
						
							|  |  |  | 				oc.id=orid; | 
					
						
							|  |  |  | 				q.occluder_instances[E->key()]=oc; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		dirty_quadrant_list.remove( dirty_quadrant_list.first() ); | 
					
						
							| 
									
										
										
										
											2016-03-18 18:57:26 +02:00
										 |  |  | 		quadrant_order_dirty=true; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pending_update=false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 	if (quadrant_order_dirty) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-03 16:33:42 -03:00
										 |  |  | 		int index=-0x80000000; //always must be drawn below children
 | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 		for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Quadrant &q=E->get(); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			for (List<RID>::Element *E=q.canvas_items.front();E;E=E->next()) { | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-03 16:33:42 -03:00
										 |  |  | 				VS::get_singleton()->canvas_item_set_draw_index(E->get(),index++); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		quadrant_order_dirty=false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	_recompute_rect_cache(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::_recompute_rect_cache() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG_ENABLED
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!rect_cache_dirty) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Rect2 r_total; | 
					
						
							|  |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 		Rect2 r; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 		r.pos=_map_to_world(E->key().x*_get_quadrant_size(), E->key().y*_get_quadrant_size()); | 
					
						
							|  |  |  | 		r.expand_to( _map_to_world(E->key().x*_get_quadrant_size()+_get_quadrant_size(), E->key().y*_get_quadrant_size()) ); | 
					
						
							|  |  |  | 		r.expand_to( _map_to_world(E->key().x*_get_quadrant_size()+_get_quadrant_size(), E->key().y*_get_quadrant_size()+_get_quadrant_size()) ); | 
					
						
							|  |  |  | 		r.expand_to( _map_to_world(E->key().x*_get_quadrant_size(), E->key().y*_get_quadrant_size()+_get_quadrant_size()) ); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		if (E==quadrant_map.front()) | 
					
						
							|  |  |  | 			r_total=r; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			r_total=r_total.merge(r); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-10-03 08:58:41 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	if (r_total==Rect2()) { | 
					
						
							|  |  |  | 		rect_cache=Rect2(-10,-10,20,20); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 		rect_cache=r_total.grow(MAX(cell_size.x,cell_size.y)*_get_quadrant_size()); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	item_rect_changed(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rect_cache_dirty=false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Map<TileMap::PosKey,TileMap::Quadrant>::Element *TileMap::_create_quadrant(const PosKey& p_qk) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 	Transform2D xform; | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	//xform.set_origin(Point2(p_qk.x,p_qk.y)*cell_size*quadrant_size);
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	Quadrant q; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	q.pos = _map_to_world(p_qk.x*_get_quadrant_size(),p_qk.y*_get_quadrant_size()); | 
					
						
							|  |  |  | 	q.pos+=get_cell_draw_offset(); | 
					
						
							|  |  |  | 	if (tile_origin==TILE_ORIGIN_CENTER) | 
					
						
							|  |  |  | 		q.pos+=cell_size/2; | 
					
						
							| 
									
										
										
										
											2016-06-18 00:01:00 +02:00
										 |  |  | 	else if (tile_origin==TILE_ORIGIN_BOTTOM_LEFT) | 
					
						
							|  |  |  | 		q.pos.y+=cell_size.y; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	xform.set_origin( q.pos ); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | //	q.canvas_item = VisualServer::get_singleton()->canvas_item_create();
 | 
					
						
							| 
									
										
										
										
											2015-02-14 06:43:50 +10:00
										 |  |  | 	q.body=Physics2DServer::get_singleton()->body_create(use_kinematic?Physics2DServer::BODY_MODE_KINEMATIC:Physics2DServer::BODY_MODE_STATIC); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 	Physics2DServer::get_singleton()->body_attach_object_instance_ID(q.body,get_instance_ID()); | 
					
						
							|  |  |  | 	Physics2DServer::get_singleton()->body_set_layer_mask(q.body,collision_layer); | 
					
						
							| 
									
										
										
										
											2015-05-03 16:47:21 -03:00
										 |  |  | 	Physics2DServer::get_singleton()->body_set_collision_mask(q.body,collision_mask); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 	Physics2DServer::get_singleton()->body_set_param(q.body,Physics2DServer::BODY_PARAM_FRICTION,friction); | 
					
						
							|  |  |  | 	Physics2DServer::get_singleton()->body_set_param(q.body,Physics2DServer::BODY_PARAM_BOUNCE,bounce); | 
					
						
							| 
									
										
										
										
											2014-06-29 22:41:02 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-05 21:20:42 -03:00
										 |  |  | 	if (is_inside_tree()) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		xform = get_global_transform() * xform; | 
					
						
							|  |  |  | 		RID space = get_world_2d()->get_space(); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 		Physics2DServer::get_singleton()->body_set_space(q.body,space); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 	Physics2DServer::get_singleton()->body_set_state(q.body,Physics2DServer::BODY_STATE_TRANSFORM,xform); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	rect_cache_dirty=true; | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 	quadrant_order_dirty=true; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	return quadrant_map.insert(p_qk,q); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::_erase_quadrant(Map<PosKey,Quadrant>::Element *Q) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Quadrant &q=Q->get(); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 	Physics2DServer::get_singleton()->free(q.body); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	for (List<RID>::Element *E=q.canvas_items.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		VisualServer::get_singleton()->free(E->get()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	q.canvas_items.clear(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	if (q.dirty_list.in_list()) | 
					
						
							|  |  |  | 		dirty_quadrant_list.remove(&q.dirty_list); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	if (navigation) { | 
					
						
							|  |  |  | 		for(Map<PosKey,Quadrant::NavPoly>::Element *E=q.navpoly_ids.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			navigation->navpoly_remove(E->get().id); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		q.navpoly_ids.clear(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(Map<PosKey,Quadrant::Occluder>::Element *E=q.occluder_instances.front();E;E=E->next()) { | 
					
						
							|  |  |  | 		VS::get_singleton()->free(E->get().id); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	q.occluder_instances.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	quadrant_map.erase(Q); | 
					
						
							|  |  |  | 	rect_cache_dirty=true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::_make_quadrant_dirty(Map<PosKey,Quadrant>::Element *Q) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Quadrant &q=Q->get(); | 
					
						
							|  |  |  | 	if (!q.dirty_list.in_list()) | 
					
						
							|  |  |  | 		dirty_quadrant_list.add(&q.dirty_list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (pending_update) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	pending_update=true; | 
					
						
							| 
									
										
										
										
											2014-11-05 21:20:42 -03:00
										 |  |  | 	if (!is_inside_tree()) | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	call_deferred("_update_dirty_quadrants"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-02 12:29:37 -03:00
										 |  |  | void TileMap::set_cellv(const Vector2& p_pos,int p_tile,bool p_flip_x,bool p_flip_y,bool p_transpose) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	set_cell(p_pos.x,p_pos.y,p_tile,p_flip_x,p_flip_y,p_transpose); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-19 23:07:25 +10:00
										 |  |  | void TileMap::set_cell(int p_x,int p_y,int p_tile,bool p_flip_x,bool p_flip_y,bool p_transpose) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	PosKey pk(p_x,p_y); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Map<PosKey,Cell>::Element *E=tile_map.find(pk); | 
					
						
							|  |  |  | 	if (!E && p_tile==INVALID_CELL) | 
					
						
							|  |  |  | 		return; //nothing to do
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	PosKey qk(p_x/_get_quadrant_size(),p_y/_get_quadrant_size()); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	if (p_tile==INVALID_CELL) { | 
					
						
							|  |  |  | 		//erase existing
 | 
					
						
							|  |  |  | 		tile_map.erase(pk); | 
					
						
							|  |  |  | 		Map<PosKey,Quadrant>::Element *Q = quadrant_map.find(qk); | 
					
						
							|  |  |  | 		ERR_FAIL_COND(!Q); | 
					
						
							|  |  |  | 		Quadrant &q=Q->get(); | 
					
						
							|  |  |  | 		q.cells.erase(pk); | 
					
						
							|  |  |  | 		if (q.cells.size()==0) | 
					
						
							|  |  |  | 			_erase_quadrant(Q); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			_make_quadrant_dirty(Q); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Map<PosKey,Quadrant>::Element *Q = quadrant_map.find(qk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!E) { | 
					
						
							|  |  |  | 		E=tile_map.insert(pk,Cell()); | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 		if (!Q) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			Q=_create_quadrant(qk); | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		Quadrant &q=Q->get(); | 
					
						
							|  |  |  | 		q.cells.insert(pk); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ERR_FAIL_COND(!Q); // quadrant should exist...
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-19 23:07:25 +10:00
										 |  |  | 		if (E->get().id==p_tile && E->get().flip_h==p_flip_x && E->get().flip_v==p_flip_y && E->get().transpose==p_transpose) | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			return; //nothing changed
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Cell &c = E->get(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c.id=p_tile; | 
					
						
							|  |  |  | 	c.flip_h=p_flip_x; | 
					
						
							|  |  |  | 	c.flip_v=p_flip_y; | 
					
						
							| 
									
										
										
										
											2015-01-19 23:07:25 +10:00
										 |  |  | 	c.transpose=p_transpose; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	_make_quadrant_dirty(Q); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-06 13:27:26 +01:00
										 |  |  | int TileMap::get_cellv(const Vector2& p_pos) const { | 
					
						
							|  |  |  | 	return get_cell(p_pos.x,p_pos.y); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | int TileMap::get_cell(int p_x,int p_y) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	PosKey pk(p_x,p_y); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const Map<PosKey,Cell>::Element *E=tile_map.find(pk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!E) | 
					
						
							|  |  |  | 		return INVALID_CELL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return E->get().id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | bool TileMap::is_cell_x_flipped(int p_x,int p_y) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	PosKey pk(p_x,p_y); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const Map<PosKey,Cell>::Element *E=tile_map.find(pk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!E) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return E->get().flip_h; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | bool TileMap::is_cell_y_flipped(int p_x,int p_y) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	PosKey pk(p_x,p_y); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const Map<PosKey,Cell>::Element *E=tile_map.find(pk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!E) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return E->get().flip_v; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-01-19 23:07:25 +10:00
										 |  |  | bool TileMap::is_cell_transposed(int p_x,int p_y) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	PosKey pk(p_x,p_y); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const Map<PosKey,Cell>::Element *E=tile_map.find(pk); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!E) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return E->get().transpose; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::_recreate_quadrants() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (Map<PosKey,Cell>::Element *E=tile_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 		PosKey qk(E->key().x/_get_quadrant_size(),E->key().y/_get_quadrant_size()); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		Map<PosKey,Quadrant>::Element *Q=quadrant_map.find(qk); | 
					
						
							|  |  |  | 		if (!Q) { | 
					
						
							|  |  |  | 			Q=_create_quadrant(qk); | 
					
						
							|  |  |  | 			dirty_quadrant_list.add(&Q->get().dirty_list); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Q->get().cells.insert(E->key()); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 		_make_quadrant_dirty(Q); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void TileMap::_clear_quadrants() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (quadrant_map.size()) { | 
					
						
							|  |  |  | 		_erase_quadrant( quadrant_map.front() ); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::clear() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	tile_map.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-07 18:25:37 -03:00
										 |  |  | void TileMap::_set_tile_data(const PoolVector<int>& p_data) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	int c=p_data.size(); | 
					
						
							| 
									
										
										
										
											2017-01-07 18:25:37 -03:00
										 |  |  | 	PoolVector<int>::Read r = p_data.read(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for(int i=0;i<c;i+=2) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const uint8_t *ptr=(const uint8_t*)&r[i]; | 
					
						
							|  |  |  | 		uint8_t local[8]; | 
					
						
							|  |  |  | 		for(int j=0;j<8;j++) | 
					
						
							|  |  |  | 			local[j]=ptr[j]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef BIG_ENDIAN_ENABLED
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		SWAP(local[0],local[3]); | 
					
						
							|  |  |  | 		SWAP(local[1],local[2]); | 
					
						
							|  |  |  | 		SWAP(local[4],local[7]); | 
					
						
							|  |  |  | 		SWAP(local[5],local[6]); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-04-19 16:46:52 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		int16_t x = decode_uint16(&local[0]); | 
					
						
							|  |  |  | 		int16_t y = decode_uint16(&local[2]); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		uint32_t v = decode_uint32(&local[4]); | 
					
						
							|  |  |  | 		bool flip_h = v&(1<<29); | 
					
						
							|  |  |  | 		bool flip_v = v&(1<<30); | 
					
						
							| 
									
										
										
										
											2015-01-19 23:07:25 +10:00
										 |  |  | 		bool transpose = v&(1<<31); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		v&=(1<<29)-1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //		if (x<-20 || y <-20 || x>4000 || y>4000)
 | 
					
						
							|  |  |  | //			continue;
 | 
					
						
							| 
									
										
										
										
											2015-01-19 23:07:25 +10:00
										 |  |  | 		set_cell(x,y,v,flip_h,flip_v,transpose); | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-07 18:25:37 -03:00
										 |  |  | PoolVector<int> TileMap::_get_tile_data() const { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-07 18:25:37 -03:00
										 |  |  | 	PoolVector<int> data; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	data.resize(tile_map.size()*2); | 
					
						
							| 
									
										
										
										
											2017-01-07 18:25:37 -03:00
										 |  |  | 	PoolVector<int>::Write w = data.write(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	int idx=0; | 
					
						
							|  |  |  | 	for(const Map<PosKey,Cell>::Element *E=tile_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		uint8_t *ptr = (uint8_t*)&w[idx]; | 
					
						
							|  |  |  | 		encode_uint16(E->key().x,&ptr[0]); | 
					
						
							|  |  |  | 		encode_uint16(E->key().y,&ptr[2]); | 
					
						
							|  |  |  | 		uint32_t val = E->get().id; | 
					
						
							|  |  |  | 		if (E->get().flip_h) | 
					
						
							|  |  |  | 			val|=(1<<29); | 
					
						
							|  |  |  | 		if (E->get().flip_v) | 
					
						
							|  |  |  | 			val|=(1<<30); | 
					
						
							| 
									
										
										
										
											2015-01-19 23:07:25 +10:00
										 |  |  | 		if (E->get().transpose) | 
					
						
							|  |  |  | 			val|=(1<<31); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		encode_uint32(val,&ptr[4]); | 
					
						
							|  |  |  | 		idx+=2; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-07 18:25:37 -03:00
										 |  |  | 	w = PoolVector<int>::Write(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Rect2 TileMap::get_item_rect() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const_cast<TileMap*>(this)->_update_dirty_quadrants(); | 
					
						
							|  |  |  | 	return rect_cache; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-03 16:47:21 -03:00
										 |  |  | void TileMap::set_collision_layer(uint32_t p_layer) { | 
					
						
							| 
									
										
										
										
											2014-05-14 01:22:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	collision_layer=p_layer; | 
					
						
							|  |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Quadrant &q=E->get(); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 		Physics2DServer::get_singleton()->body_set_layer_mask(q.body,collision_layer); | 
					
						
							| 
									
										
										
										
											2014-05-14 01:22:15 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-03 16:47:21 -03:00
										 |  |  | void TileMap::set_collision_mask(uint32_t p_mask) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	collision_mask=p_mask; | 
					
						
							|  |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Quadrant &q=E->get(); | 
					
						
							|  |  |  | 		Physics2DServer::get_singleton()->body_set_collision_mask(q.body,collision_mask); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-14 06:43:50 +10:00
										 |  |  | bool TileMap::get_collision_use_kinematic() const{ | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-14 06:43:50 +10:00
										 |  |  | 	return use_kinematic; | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-14 06:43:50 +10:00
										 |  |  | void TileMap::set_collision_use_kinematic(bool p_use_kinematic) { | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							| 
									
										
										
										
											2015-02-14 06:43:50 +10:00
										 |  |  | 	use_kinematic=p_use_kinematic; | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 	_recreate_quadrants(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-29 22:41:02 -03:00
										 |  |  | void TileMap::set_collision_friction(float p_friction) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	friction=p_friction; | 
					
						
							|  |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Quadrant &q=E->get(); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 		Physics2DServer::get_singleton()->body_set_param(q.body,Physics2DServer::BODY_PARAM_FRICTION,p_friction); | 
					
						
							| 
									
										
										
										
											2014-06-29 22:41:02 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | float TileMap::get_collision_friction() const{ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return friction; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::set_collision_bounce(float p_bounce){ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bounce=p_bounce; | 
					
						
							|  |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Quadrant &q=E->get(); | 
					
						
							| 
									
										
										
										
											2015-02-13 20:49:21 +10:00
										 |  |  | 		Physics2DServer::get_singleton()->body_set_param(q.body,Physics2DServer::BODY_PARAM_BOUNCE,p_bounce); | 
					
						
							| 
									
										
										
										
											2014-06-29 22:41:02 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | float TileMap::get_collision_bounce() const{ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return bounce; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-03 16:47:21 -03:00
										 |  |  | uint32_t TileMap::get_collision_layer() const { | 
					
						
							| 
									
										
										
										
											2014-05-14 01:22:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return collision_layer; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-03 16:47:21 -03:00
										 |  |  | uint32_t TileMap::get_collision_mask() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return collision_mask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | void TileMap::set_mode(Mode p_mode) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	mode=p_mode; | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							|  |  |  | 	emit_signal("settings_changed"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileMap::Mode TileMap::get_mode() const { | 
					
						
							|  |  |  | 	return mode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TileMap::set_half_offset(HalfOffset p_half_offset) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	half_offset=p_half_offset; | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							|  |  |  | 	emit_signal("settings_changed"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | void TileMap::set_tile_origin(TileOrigin p_tile_origin) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	tile_origin=p_tile_origin; | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							|  |  |  | 	emit_signal("settings_changed"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileMap::TileOrigin TileMap::get_tile_origin() const{ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return tile_origin; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | Vector2 TileMap::get_cell_draw_offset() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch(mode) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case MODE_SQUARE: { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return Vector2(); | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		case MODE_ISOMETRIC: { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return Vector2(-cell_size.x*0.5,0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		case MODE_CUSTOM: { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Vector2 min; | 
					
						
							|  |  |  | 			min.x = MIN(custom_transform[0].x,min.x); | 
					
						
							|  |  |  | 			min.y = MIN(custom_transform[0].y,min.y); | 
					
						
							|  |  |  | 			min.x = MIN(custom_transform[1].x,min.x); | 
					
						
							|  |  |  | 			min.y = MIN(custom_transform[1].y,min.y); | 
					
						
							|  |  |  | 			return min; | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return Vector2(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileMap::HalfOffset TileMap::get_half_offset() const { | 
					
						
							|  |  |  | 	return half_offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | Transform2D TileMap::get_cell_transform() const { | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	switch(mode) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case MODE_SQUARE: { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 			Transform2D m; | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 			m[0]*=cell_size.x; | 
					
						
							|  |  |  | 			m[1]*=cell_size.y; | 
					
						
							|  |  |  | 			return m; | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		case MODE_ISOMETRIC: { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			//isometric only makes sense when y is positive in both x and y vectors, otherwise
 | 
					
						
							|  |  |  | 			//the drawing of tiles will overlap
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 			Transform2D m; | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 			m[0]=Vector2(cell_size.x*0.5,cell_size.y*0.5); | 
					
						
							|  |  |  | 			m[1]=Vector2(-cell_size.x*0.5,cell_size.y*0.5); | 
					
						
							|  |  |  | 			return m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		case MODE_CUSTOM: { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return custom_transform; | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 	return Transform2D(); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | void TileMap::set_custom_transform(const Transform2D& p_xform) { | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	custom_transform=p_xform; | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							|  |  |  | 	emit_signal("settings_changed"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | Transform2D TileMap::get_custom_transform() const{ | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return custom_transform; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Vector2 TileMap::_map_to_world(int x,int y,bool p_ignore_ofs) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Vector2 ret = get_cell_transform().xform(Vector2(x,y)); | 
					
						
							|  |  |  | 	if (!p_ignore_ofs) { | 
					
						
							|  |  |  | 		switch(half_offset) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			case HALF_OFFSET_X: { | 
					
						
							|  |  |  | 				if (ABS(y)&1) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					ret+=get_cell_transform()[0]*0.5; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} break; | 
					
						
							|  |  |  | 			case HALF_OFFSET_Y: { | 
					
						
							|  |  |  | 				if (ABS(x)&1) { | 
					
						
							|  |  |  | 					ret+=get_cell_transform()[1]*0.5; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} break; | 
					
						
							|  |  |  | 			default: {} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | Vector2 TileMap::map_to_world(const Vector2& p_pos,bool p_ignore_ofs) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return _map_to_world(p_pos.x,p_pos.y,p_ignore_ofs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | Vector2 TileMap::world_to_map(const Vector2& p_pos) const{ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Vector2 ret = get_cell_transform().affine_inverse().xform(p_pos); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch(half_offset) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case HALF_OFFSET_X: { | 
					
						
							| 
									
										
										
										
											2015-10-24 21:47:48 -04:00
										 |  |  | 			if ( ret.y > 0 ? int(ret.y)&1 : (int(ret.y)-1)&1 ) { | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 				ret.x-=0.5; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		case HALF_OFFSET_Y: { | 
					
						
							| 
									
										
										
										
											2015-10-24 21:47:48 -04:00
										 |  |  | 			if ( ret.x > 0 ? int(ret.x)&1 : (int(ret.x)-1)&1) { | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 				ret.y-=0.5; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} break; | 
					
						
							|  |  |  | 		default: {} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret.floor(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | void TileMap::set_y_sort_mode(bool p_enable) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_clear_quadrants(); | 
					
						
							|  |  |  | 	y_sort_mode=p_enable; | 
					
						
							|  |  |  | 	VS::get_singleton()->canvas_item_set_sort_children_by_y(get_canvas_item(),y_sort_mode); | 
					
						
							|  |  |  | 	_recreate_quadrants(); | 
					
						
							|  |  |  | 	emit_signal("settings_changed"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool TileMap::is_y_sort_mode_enabled() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return y_sort_mode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 14:00:15 -03:00
										 |  |  | Array TileMap::get_used_cells() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Array a; | 
					
						
							|  |  |  | 	a.resize(tile_map.size()); | 
					
						
							|  |  |  | 	int i=0; | 
					
						
							|  |  |  | 	for (Map<PosKey,Cell>::Element *E=tile_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Vector2 p (E->key().x,E->key().y); | 
					
						
							|  |  |  | 		a[i++]=p; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return a; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-14 01:22:15 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-12 10:45:31 -03:00
										 |  |  | void TileMap::set_occluder_light_mask(int p_mask) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	occluder_light_mask=p_mask; | 
					
						
							|  |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (Map<PosKey,Quadrant::Occluder>::Element *F=E->get().occluder_instances.front();F;F=F->next()) { | 
					
						
							|  |  |  | 			VisualServer::get_singleton()->canvas_light_occluder_set_light_mask(F->get().id,occluder_light_mask); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int TileMap::get_occluder_light_mask() const{ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return occluder_light_mask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-29 14:47:13 -03:00
										 |  |  | void TileMap::set_light_mask(int p_light_mask) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	CanvasItem::set_light_mask(p_light_mask); | 
					
						
							|  |  |  | 	for (Map<PosKey,Quadrant>::Element *E=quadrant_map.front();E;E=E->next()) { | 
					
						
							| 
									
										
										
										
											2015-12-12 10:45:31 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-29 14:47:13 -03:00
										 |  |  | 		for (List<RID>::Element *F=E->get().canvas_items.front();F;F=F->next()) { | 
					
						
							|  |  |  | 			VisualServer::get_singleton()->canvas_item_set_light_mask(F->get(),get_light_mask()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-12-12 10:45:31 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void TileMap::_bind_methods() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_tileset","tileset:TileSet"),&TileMap::set_tileset); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_tileset:TileSet"),&TileMap::get_tileset); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_mode","mode"),&TileMap::set_mode); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_mode"),&TileMap::get_mode); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_half_offset","half_offset"),&TileMap::set_half_offset); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_half_offset"),&TileMap::get_half_offset); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_custom_transform","custom_transform"),&TileMap::set_custom_transform); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_custom_transform"),&TileMap::get_custom_transform); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_cell_size","size"),&TileMap::set_cell_size); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_cell_size"),&TileMap::get_cell_size); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("_set_old_cell_size","size"),&TileMap::_set_old_cell_size); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("_get_old_cell_size"),&TileMap::_get_old_cell_size); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_quadrant_size","size"),&TileMap::set_quadrant_size); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_quadrant_size"),&TileMap::get_quadrant_size); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_tile_origin","origin"),&TileMap::set_tile_origin); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_tile_origin"),&TileMap::get_tile_origin); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_center_x","enable"),&TileMap::set_center_x); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_center_x"),&TileMap::get_center_x); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_center_y","enable"),&TileMap::set_center_y); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_center_y"),&TileMap::get_center_y); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_y_sort_mode","enable"),&TileMap::set_y_sort_mode); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("is_y_sort_mode_enabled"),&TileMap::is_y_sort_mode_enabled); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_collision_use_kinematic","use_kinematic"),&TileMap::set_collision_use_kinematic); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_collision_use_kinematic"),&TileMap::get_collision_use_kinematic); | 
					
						
							| 
									
										
										
										
											2015-02-14 06:43:50 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_collision_layer","mask"),&TileMap::set_collision_layer); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_collision_layer"),&TileMap::get_collision_layer); | 
					
						
							| 
									
										
										
										
											2015-05-03 16:47:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_collision_mask","mask"),&TileMap::set_collision_mask); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_collision_mask"),&TileMap::get_collision_mask); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_collision_friction","value"),&TileMap::set_collision_friction); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_collision_friction"),&TileMap::get_collision_friction); | 
					
						
							| 
									
										
										
										
											2014-06-29 22:41:02 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_collision_bounce","value"),&TileMap::set_collision_bounce); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_collision_bounce"),&TileMap::get_collision_bounce); | 
					
						
							| 
									
										
										
										
											2014-06-29 22:41:02 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_occluder_light_mask","mask"),&TileMap::set_occluder_light_mask); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_occluder_light_mask"),&TileMap::get_occluder_light_mask); | 
					
						
							| 
									
										
										
										
											2015-12-12 10:45:31 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("set_cell","x","y","tile","flip_x","flip_y","transpose"),&TileMap::set_cell,DEFVAL(false),DEFVAL(false),DEFVAL(false)); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("set_cellv","pos","tile","flip_x","flip_y","transpose"),&TileMap::set_cellv,DEFVAL(false),DEFVAL(false),DEFVAL(false)); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_cell","x","y"),&TileMap::get_cell); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("get_cellv","pos"),&TileMap::get_cellv); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("is_cell_x_flipped","x","y"),&TileMap::is_cell_x_flipped); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("is_cell_y_flipped","x","y"),&TileMap::is_cell_y_flipped); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("is_cell_transposed","x","y"),&TileMap::is_cell_transposed); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("clear"),&TileMap::clear); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("get_used_cells"),&TileMap::get_used_cells); | 
					
						
							| 
									
										
										
										
											2015-04-18 14:00:15 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("map_to_world","mappos","ignore_half_ofs"),&TileMap::map_to_world,DEFVAL(false)); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("world_to_map","worldpos"),&TileMap::world_to_map); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("_clear_quadrants"),&TileMap::_clear_quadrants); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("_recreate_quadrants"),&TileMap::_recreate_quadrants); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("_update_dirty_quadrants"),&TileMap::_update_dirty_quadrants); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 23:03:46 -03:00
										 |  |  | 	ClassDB::bind_method(_MD("_set_tile_data"),&TileMap::_set_tile_data); | 
					
						
							|  |  |  | 	ClassDB::bind_method(_MD("_get_tile_data"),&TileMap::_get_tile_data); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::INT,"mode",PROPERTY_HINT_ENUM,"Square,Isometric,Custom"),_SCS("set_mode"),_SCS("get_mode")); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::OBJECT,"tile_set",PROPERTY_HINT_RESOURCE_TYPE,"TileSet"),_SCS("set_tileset"),_SCS("get_tileset")); | 
					
						
							| 
									
										
										
										
											2015-05-03 16:47:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-04 01:16:14 -03:00
										 |  |  | 	ADD_GROUP("Cell","cell_"); | 
					
						
							|  |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::VECTOR2,"cell_size",PROPERTY_HINT_RANGE,"1,8192,1"),_SCS("set_cell_size"),_SCS("get_cell_size")); | 
					
						
							|  |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::INT,"cell_quadrant_size",PROPERTY_HINT_RANGE,"1,128,1"),_SCS("set_quadrant_size"),_SCS("get_quadrant_size")); | 
					
						
							| 
									
										
										
										
											2017-01-11 00:52:51 -03:00
										 |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::TRANSFORM2D,"cell_custom_transform"),_SCS("set_custom_transform"),_SCS("get_custom_transform")); | 
					
						
							| 
									
										
										
										
											2017-01-04 01:16:14 -03:00
										 |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::INT,"cell_half_offset",PROPERTY_HINT_ENUM,"Offset X,Offset Y,Disabled"),_SCS("set_half_offset"),_SCS("get_half_offset")); | 
					
						
							|  |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::INT,"cell_tile_origin",PROPERTY_HINT_ENUM,"Top Left,Center,Bottom Left"),_SCS("set_tile_origin"),_SCS("get_tile_origin")); | 
					
						
							|  |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::BOOL,"cell_y_sort"),_SCS("set_y_sort_mode"),_SCS("is_y_sort_mode_enabled")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ADD_GROUP("Collision","collision_"); | 
					
						
							|  |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::BOOL,"collision_use_kinematic",PROPERTY_HINT_NONE,""),_SCS("set_collision_use_kinematic"),_SCS("get_collision_use_kinematic")); | 
					
						
							|  |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::REAL,"collision_friction",PROPERTY_HINT_RANGE,"0,1,0.01"),_SCS("set_collision_friction"),_SCS("get_collision_friction")); | 
					
						
							|  |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::REAL,"collision_bounce",PROPERTY_HINT_RANGE,"0,1,0.01"),_SCS("set_collision_bounce"),_SCS("get_collision_bounce")); | 
					
						
							| 
									
										
										
										
											2017-01-10 22:20:57 -03:00
										 |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::INT,"collision_layers",PROPERTY_HINT_LAYERS_2D_PHYSICS),_SCS("set_collision_layer"),_SCS("get_collision_layer")); | 
					
						
							|  |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::INT,"collision_mask",PROPERTY_HINT_LAYERS_2D_PHYSICS),_SCS("set_collision_mask"),_SCS("get_collision_mask")); | 
					
						
							| 
									
										
										
										
											2017-01-04 01:16:14 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ADD_GROUP("Occluder","occluder_"); | 
					
						
							| 
									
										
										
										
											2017-01-10 22:20:57 -03:00
										 |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::INT,"occluder_light_mask",PROPERTY_HINT_LAYERS_2D_RENDER),_SCS("set_occluder_light_mask"),_SCS("get_occluder_light_mask")); | 
					
						
							| 
									
										
										
										
											2017-01-04 01:16:14 -03:00
										 |  |  | 	ADD_GROUP("",""); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	ADD_PROPERTY( PropertyInfo(Variant::OBJECT,"tile_data",PROPERTY_HINT_NONE,"",PROPERTY_USAGE_NOEDITOR),_SCS("_set_tile_data"),_SCS("_get_tile_data")); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-19 16:46:52 -03:00
										 |  |  | 	ADD_SIGNAL(MethodInfo("settings_changed")); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	BIND_CONSTANT( INVALID_CELL ); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	BIND_CONSTANT( MODE_SQUARE ); | 
					
						
							|  |  |  | 	BIND_CONSTANT( MODE_ISOMETRIC ); | 
					
						
							|  |  |  | 	BIND_CONSTANT( MODE_CUSTOM ); | 
					
						
							|  |  |  | 	BIND_CONSTANT( HALF_OFFSET_X ); | 
					
						
							|  |  |  | 	BIND_CONSTANT( HALF_OFFSET_Y ); | 
					
						
							|  |  |  | 	BIND_CONSTANT( HALF_OFFSET_DISABLED ); | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	BIND_CONSTANT( TILE_ORIGIN_TOP_LEFT ); | 
					
						
							|  |  |  | 	BIND_CONSTANT( TILE_ORIGIN_CENTER ); | 
					
						
							| 
									
										
										
										
											2016-06-18 00:01:00 +02:00
										 |  |  | 	BIND_CONSTANT( TILE_ORIGIN_BOTTOM_LEFT ); | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileMap::TileMap() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rect_cache_dirty=true; | 
					
						
							|  |  |  | 	pending_update=false; | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 	quadrant_order_dirty=false; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	quadrant_size=16; | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	cell_size=Size2(64,64); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	center_x=false; | 
					
						
							|  |  |  | 	center_y=false; | 
					
						
							| 
									
										
										
										
											2014-05-14 01:22:15 -03:00
										 |  |  | 	collision_layer=1; | 
					
						
							| 
									
										
										
										
											2015-05-03 16:47:21 -03:00
										 |  |  | 	collision_mask=1; | 
					
						
							| 
									
										
										
										
											2014-06-29 22:41:02 -03:00
										 |  |  | 	friction=1; | 
					
						
							|  |  |  | 	bounce=0; | 
					
						
							| 
									
										
										
										
											2014-10-03 00:10:51 -03:00
										 |  |  | 	mode=MODE_SQUARE; | 
					
						
							|  |  |  | 	half_offset=HALF_OFFSET_DISABLED; | 
					
						
							| 
									
										
										
										
											2015-02-14 06:43:50 +10:00
										 |  |  | 	use_kinematic=false; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	navigation=NULL; | 
					
						
							|  |  |  | 	y_sort_mode=false; | 
					
						
							| 
									
										
										
										
											2015-12-12 10:45:31 -03:00
										 |  |  | 	occluder_light_mask=1; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-18 16:17:33 -03:00
										 |  |  | 	fp_adjust=0.00001; | 
					
						
							| 
									
										
										
										
											2015-03-09 02:34:56 -03:00
										 |  |  | 	tile_origin=TILE_ORIGIN_TOP_LEFT; | 
					
						
							| 
									
										
										
										
											2017-01-12 20:35:46 -03:00
										 |  |  | 	set_notify_transform(true); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TileMap::~TileMap() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	clear(); | 
					
						
							|  |  |  | } |