| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /**************************************************************************/ | 
					
						
							| 
									
										
										
										
											2019-02-12 17:18:13 +01:00
										 |  |  | /*  bit_map.cpp                                                           */ | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /*                         This file is part of:                          */ | 
					
						
							|  |  |  | /*                             GODOT ENGINE                               */ | 
					
						
							|  |  |  | /*                        https://godotengine.org                         */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							|  |  |  | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ | 
					
						
							|  |  |  | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                  */ | 
					
						
							|  |  |  | /*                                                                        */ | 
					
						
							|  |  |  | /* 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.                 */ | 
					
						
							|  |  |  | /**************************************************************************/ | 
					
						
							| 
									
										
										
										
											2018-01-05 00:50:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 17:18:13 +01:00
										 |  |  | #include "bit_map.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-14 14:30:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 18:13:45 +02:00
										 |  |  | #include "core/io/image_loader.h"
 | 
					
						
							| 
									
										
										
										
											2022-08-05 03:41:48 +02:00
										 |  |  | #include "core/variant/typed_array.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | void BitMap::create(const Size2i &p_size) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	ERR_FAIL_COND(p_size.width < 1); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(p_size.height < 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	ERR_FAIL_COND(static_cast<int64_t>(p_size.width) * static_cast<int64_t>(p_size.height) > INT32_MAX); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Error err = bitmask.resize((((p_size.width * p_size.height) - 1) / 8) + 1); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(err != OK); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	width = p_size.width; | 
					
						
							|  |  |  | 	height = p_size.height; | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-27 16:19:21 +02:00
										 |  |  | 	memset(bitmask.ptrw(), 0, bitmask.size()); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-21 11:30:55 -05:00
										 |  |  | void BitMap::create_from_image_alpha(const Ref<Image> &p_image, float p_threshold) { | 
					
						
							| 
									
										
										
										
											2020-12-15 12:04:21 +00:00
										 |  |  | 	ERR_FAIL_COND(p_image.is_null() || p_image->is_empty()); | 
					
						
							| 
									
										
										
										
											2017-05-17 07:36:47 -03:00
										 |  |  | 	Ref<Image> img = p_image->duplicate(); | 
					
						
							|  |  |  | 	img->convert(Image::FORMAT_LA8); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(img->get_format() != Image::FORMAT_LA8); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	create(Size2i(img->get_width(), img->get_height())); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 18:06:54 -03:00
										 |  |  | 	const uint8_t *r = img->get_data().ptr(); | 
					
						
							| 
									
										
										
										
											2017-11-25 00:07:54 -03:00
										 |  |  | 	uint8_t *w = bitmask.ptrw(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (int i = 0; i < width * height; i++) { | 
					
						
							|  |  |  | 		int bbyte = i / 8; | 
					
						
							|  |  |  | 		int bbit = i % 8; | 
					
						
							| 
									
										
										
										
											2018-02-21 11:30:55 -05:00
										 |  |  | 		if (r[i * 2 + 1] / 255.0 > p_threshold) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			w[bbyte] |= (1 << bbit); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | void BitMap::set_bit_rect(const Rect2i &p_rect, bool p_value) { | 
					
						
							| 
									
										
										
										
											2020-12-19 12:43:35 +00:00
										 |  |  | 	Rect2i current = Rect2i(0, 0, width, height).intersection(p_rect); | 
					
						
							| 
									
										
										
										
											2017-11-25 00:07:54 -03:00
										 |  |  | 	uint8_t *data = bitmask.ptrw(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-04 00:25:13 +02:00
										 |  |  | 	for (int i = current.position.x; i < current.position.x + current.size.x; i++) { | 
					
						
							|  |  |  | 		for (int j = current.position.y; j < current.position.y + current.size.y; j++) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 			int ofs = width * j + i; | 
					
						
							|  |  |  | 			int bbyte = ofs / 8; | 
					
						
							|  |  |  | 			int bbit = ofs % 8; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			uint8_t b = data[bbyte]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 			if (p_value) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 				b |= (1 << bbit); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2018-01-09 16:31:30 +00:00
										 |  |  | 				b &= ~(1 << bbit); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			data[bbyte] = b; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int BitMap::get_true_bit_count() const { | 
					
						
							|  |  |  | 	int ds = bitmask.size(); | 
					
						
							|  |  |  | 	const uint8_t *d = bitmask.ptr(); | 
					
						
							|  |  |  | 	int c = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	// Fast, almost branchless version.
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (int i = 0; i < ds; i++) { | 
					
						
							|  |  |  | 		c += (d[i] & (1 << 7)) >> 7; | 
					
						
							|  |  |  | 		c += (d[i] & (1 << 6)) >> 6; | 
					
						
							|  |  |  | 		c += (d[i] & (1 << 5)) >> 5; | 
					
						
							|  |  |  | 		c += (d[i] & (1 << 4)) >> 4; | 
					
						
							|  |  |  | 		c += (d[i] & (1 << 3)) >> 3; | 
					
						
							|  |  |  | 		c += (d[i] & (1 << 2)) >> 2; | 
					
						
							| 
									
										
										
										
											2019-04-04 19:11:39 +03:00
										 |  |  | 		c += (d[i] & (1 << 1)) >> 1; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		c += d[i] & 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return c; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | void BitMap::set_bitv(const Point2i &p_pos, bool p_value) { | 
					
						
							|  |  |  | 	set_bit(p_pos.x, p_pos.y, p_value); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | void BitMap::set_bit(int p_x, int p_y, bool p_value) { | 
					
						
							|  |  |  | 	ERR_FAIL_INDEX(p_x, width); | 
					
						
							|  |  |  | 	ERR_FAIL_INDEX(p_y, height); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	int ofs = width * p_y + p_x; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	int bbyte = ofs / 8; | 
					
						
							|  |  |  | 	int bbit = ofs % 8; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	uint8_t b = bitmask[bbyte]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (p_value) { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		b |= (1 << bbit); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2018-01-09 16:31:30 +00:00
										 |  |  | 		b &= ~(1 << bbit); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-25 03:11:03 +02:00
										 |  |  | 	bitmask.write[bbyte] = b; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | bool BitMap::get_bitv(const Point2i &p_pos) const { | 
					
						
							|  |  |  | 	return get_bit(p_pos.x, p_pos.y); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool BitMap::get_bit(int p_x, int p_y) const { | 
					
						
							|  |  |  | 	ERR_FAIL_INDEX_V(p_x, width, false); | 
					
						
							|  |  |  | 	ERR_FAIL_INDEX_V(p_y, height, false); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	int ofs = width * p_y + p_x; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	int bbyte = ofs / 8; | 
					
						
							|  |  |  | 	int bbit = ofs % 8; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return (bitmask[bbyte] & (1 << bbit)) != 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | Size2i BitMap::get_size() const { | 
					
						
							|  |  |  | 	return Size2i(width, height); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BitMap::_set_data(const Dictionary &p_d) { | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!p_d.has("size")); | 
					
						
							|  |  |  | 	ERR_FAIL_COND(!p_d.has("data")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	create(p_d["size"]); | 
					
						
							|  |  |  | 	bitmask = p_d["data"]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Dictionary BitMap::_get_data() const { | 
					
						
							|  |  |  | 	Dictionary d; | 
					
						
							|  |  |  | 	d["size"] = get_size(); | 
					
						
							|  |  |  | 	d["data"] = bitmask; | 
					
						
							|  |  |  | 	return d; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | Vector<Vector<Vector2>> BitMap::_march_square(const Rect2i &p_rect, const Point2i &p_start) const { | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	int stepx = 0; | 
					
						
							|  |  |  | 	int stepy = 0; | 
					
						
							|  |  |  | 	int prevx = 0; | 
					
						
							|  |  |  | 	int prevy = 0; | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	int startx = p_start.x; | 
					
						
							|  |  |  | 	int starty = p_start.y; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	int curx = startx; | 
					
						
							|  |  |  | 	int cury = starty; | 
					
						
							|  |  |  | 	unsigned int count = 0; | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 	HashMap<Point2i, int> cross_map; | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 	Vector<Vector2> _points; | 
					
						
							|  |  |  | 	int points_size = 0; | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Vector<Vector<Vector2>> ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Add starting entry at start of return.
 | 
					
						
							|  |  |  | 	ret.resize(1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	do { | 
					
						
							|  |  |  | 		int sv = 0; | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 		{ // Square value
 | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			/*
 | 
					
						
							| 
									
										
										
										
											2018-08-14 14:30:15 +02:00
										 |  |  | 			checking the 2x2 pixel grid, assigning these values to each pixel, if not transparent | 
					
						
							|  |  |  | 			+---+---+ | 
					
						
							|  |  |  | 			| 1 | 2 | | 
					
						
							|  |  |  | 			+---+---+ | 
					
						
							|  |  |  | 			| 4 | 8 | <- current pixel (curx,cury) | 
					
						
							|  |  |  | 			+---+---+ | 
					
						
							|  |  |  | 			*/ | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 			Point2i tl = Point2i(curx - 1, cury - 1); | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			sv += (p_rect.has_point(tl) && get_bitv(tl)) ? 1 : 0; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 			Point2i tr = Point2i(curx, cury - 1); | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			sv += (p_rect.has_point(tr) && get_bitv(tr)) ? 2 : 0; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 			Point2i bl = Point2i(curx - 1, cury); | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			sv += (p_rect.has_point(bl) && get_bitv(bl)) ? 4 : 0; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 			Point2i br = Point2i(curx, cury); | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			sv += (p_rect.has_point(br) && get_bitv(br)) ? 8 : 0; | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 			ERR_FAIL_COND_V(sv == 0 || sv == 15, Vector<Vector<Vector2>>()); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		switch (sv) { | 
					
						
							|  |  |  | 			case 1: | 
					
						
							|  |  |  | 			case 5: | 
					
						
							|  |  |  | 			case 13: | 
					
						
							|  |  |  | 				/* going UP with these cases:
 | 
					
						
							| 
									
										
										
										
											2018-08-14 14:30:15 +02:00
										 |  |  | 				1          5           13 | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				| 1 |   |  | 1 |   |  | 1 |   | | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				|   |   |  | 4 |   |  | 4 | 8 | | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				*/ | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 				stepx = 0; | 
					
						
							|  |  |  | 				stepy = -1; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			case 8: | 
					
						
							|  |  |  | 			case 10: | 
					
						
							|  |  |  | 			case 11: | 
					
						
							|  |  |  | 				/* going DOWN with these cases:
 | 
					
						
							| 
									
										
										
										
											2018-08-14 14:30:15 +02:00
										 |  |  | 				8          10         11 | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				|   |   |  |   | 2 |  | 1 | 2 | | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				|   | 8 |  |   | 8 |  |   | 8 | | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				*/ | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 				stepx = 0; | 
					
						
							|  |  |  | 				stepy = 1; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			case 4: | 
					
						
							|  |  |  | 			case 12: | 
					
						
							|  |  |  | 			case 14: | 
					
						
							|  |  |  | 				/* going LEFT with these cases:
 | 
					
						
							| 
									
										
										
										
											2018-08-14 14:30:15 +02:00
										 |  |  | 				4          12         14 | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				|   |   |  |   |   |  |   | 2 | | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				| 4 |   |  | 4 | 8 |  | 4 | 8 | | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				*/ | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 				stepx = -1; | 
					
						
							|  |  |  | 				stepy = 0; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			case 2: | 
					
						
							|  |  |  | 			case 3: | 
					
						
							|  |  |  | 			case 7: | 
					
						
							|  |  |  | 				/* going RIGHT with these cases:
 | 
					
						
							| 
									
										
										
										
											2018-08-14 14:30:15 +02:00
										 |  |  | 				2          3          7 | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				|   | 2 |  | 1 | 2 |  | 1 | 2 | | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				|   |   |  |   |   |  | 4 |   | | 
					
						
							|  |  |  | 				+---+---+  +---+---+  +---+---+ | 
					
						
							|  |  |  | 				*/ | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 				stepx = 1; | 
					
						
							|  |  |  | 				stepy = 0; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case 9: | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 				/* Going DOWN if coming from the LEFT, otherwise go UP.
 | 
					
						
							|  |  |  | 				9 | 
					
						
							| 
									
										
										
										
											2018-08-14 14:30:15 +02:00
										 |  |  | 				+---+---+ | 
					
						
							|  |  |  | 				| 1 |   | | 
					
						
							|  |  |  | 				+---+---+ | 
					
						
							|  |  |  | 				|   | 8 | | 
					
						
							|  |  |  | 				+---+---+ | 
					
						
							|  |  |  | 				*/ | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				if (prevx == 1) { | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 					stepx = 0; | 
					
						
							|  |  |  | 					stepy = 1; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					stepx = 0; | 
					
						
							|  |  |  | 					stepy = -1; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case 6: | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 				/* Going RIGHT if coming from BELOW, otherwise go LEFT.
 | 
					
						
							| 
									
										
										
										
											2018-08-14 14:30:15 +02:00
										 |  |  | 				6 | 
					
						
							|  |  |  | 				+---+---+ | 
					
						
							|  |  |  | 				|   | 2 | | 
					
						
							|  |  |  | 				+---+---+ | 
					
						
							|  |  |  | 				| 4 |   | | 
					
						
							|  |  |  | 				+---+---+ | 
					
						
							|  |  |  | 				*/ | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				if (prevy == -1) { | 
					
						
							|  |  |  | 					stepx = 1; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 					stepy = 0; | 
					
						
							|  |  |  | 				} else { | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 					stepx = -1; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 					stepy = 0; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 				ERR_PRINT("this shouldn't happen."); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Handle crossing points.
 | 
					
						
							|  |  |  | 		if (sv == 6 || sv == 9) { | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 			const Point2i cur_pos(curx, cury); | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 12:24:08 +01:00
										 |  |  | 			// Find if this point has occurred before.
 | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 			if (HashMap<Point2i, int>::Iterator found = cross_map.find(cur_pos)) { | 
					
						
							|  |  |  | 				// Add points after the previous crossing to the result.
 | 
					
						
							|  |  |  | 				ret.push_back(_points.slice(found->value + 1, points_size)); | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 				// Remove points after crossing point.
 | 
					
						
							|  |  |  | 				points_size = found->value + 1; | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 				// Erase trailing map elements.
 | 
					
						
							|  |  |  | 				while (cross_map.last() != found) { | 
					
						
							|  |  |  | 					cross_map.remove(cross_map.last()); | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 				cross_map.erase(cur_pos); | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 				// Add crossing point to map.
 | 
					
						
							|  |  |  | 				cross_map.insert(cur_pos, points_size - 1); | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 		// Small optimization:
 | 
					
						
							|  |  |  | 		// If the previous direction is same as the current direction,
 | 
					
						
							|  |  |  | 		// then we should modify the last vector to current.
 | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		curx += stepx; | 
					
						
							|  |  |  | 		cury += stepy; | 
					
						
							|  |  |  | 		if (stepx == prevx && stepy == prevy) { | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 			_points.set(points_size - 1, Vector2(curx, cury) - p_rect.position); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 			_points.resize(MAX(points_size + 1, _points.size())); | 
					
						
							|  |  |  | 			_points.set(points_size, Vector2(curx, cury) - p_rect.position); | 
					
						
							|  |  |  | 			points_size++; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		count++; | 
					
						
							|  |  |  | 		prevx = stepx; | 
					
						
							|  |  |  | 		prevy = stepy; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-28 10:50:54 +02:00
										 |  |  | 		ERR_FAIL_COND_V((int)count > 2 * (width * height + 1), Vector<Vector<Vector2>>()); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	} while (curx != startx || cury != starty); | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 	// Add remaining points to result.
 | 
					
						
							|  |  |  | 	_points.resize(points_size); | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 12:12:47 +01:00
										 |  |  | 	ret.set(0, _points); | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static float perpendicular_distance(const Vector2 &i, const Vector2 &start, const Vector2 &end) { | 
					
						
							|  |  |  | 	float res; | 
					
						
							|  |  |  | 	float slope; | 
					
						
							|  |  |  | 	float intercept; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (start.x == end.x) { | 
					
						
							|  |  |  | 		res = Math::absf(i.x - end.x); | 
					
						
							|  |  |  | 	} else if (start.y == end.y) { | 
					
						
							|  |  |  | 		res = Math::absf(i.y - end.y); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		slope = (end.y - start.y) / (end.x - start.x); | 
					
						
							|  |  |  | 		intercept = start.y - (slope * start.x); | 
					
						
							|  |  |  | 		res = Math::absf(slope * i.x - i.y + intercept) / Math::sqrt(Math::pow(slope, 2.0f) + 1.0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Vector<Vector2> rdp(const Vector<Vector2> &v, float optimization) { | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	if (v.size() < 3) { | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		return v; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	int index = -1; | 
					
						
							| 
									
										
										
										
											2021-02-09 18:24:36 +01:00
										 |  |  | 	float dist = 0.0; | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	// Not looping first and last point.
 | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	for (size_t i = 1, size = v.size(); i < size - 1; ++i) { | 
					
						
							|  |  |  | 		float cdist = perpendicular_distance(v[i], v[0], v[v.size() - 1]); | 
					
						
							|  |  |  | 		if (cdist > dist) { | 
					
						
							|  |  |  | 			dist = cdist; | 
					
						
							|  |  |  | 			index = static_cast<int>(i); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (dist > optimization) { | 
					
						
							|  |  |  | 		Vector<Vector2> left, right; | 
					
						
							|  |  |  | 		left.resize(index); | 
					
						
							|  |  |  | 		for (int i = 0; i < index; i++) { | 
					
						
							| 
									
										
										
										
											2018-07-25 03:11:03 +02:00
										 |  |  | 			left.write[i] = v[i]; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		right.resize(v.size() - index); | 
					
						
							|  |  |  | 		for (int i = 0; i < right.size(); i++) { | 
					
						
							| 
									
										
										
										
											2018-07-25 03:11:03 +02:00
										 |  |  | 			right.write[i] = v[index + i]; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		Vector<Vector2> r1 = rdp(left, optimization); | 
					
						
							|  |  |  | 		Vector<Vector2> r2 = rdp(right, optimization); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		int middle = r1.size(); | 
					
						
							|  |  |  | 		r1.resize(r1.size() + r2.size()); | 
					
						
							|  |  |  | 		for (int i = 0; i < r2.size(); i++) { | 
					
						
							| 
									
										
										
										
											2018-07-25 03:11:03 +02:00
										 |  |  | 			r1.write[middle + i] = r2[i]; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return r1; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		Vector<Vector2> ret; | 
					
						
							|  |  |  | 		ret.push_back(v[0]); | 
					
						
							|  |  |  | 		ret.push_back(v[v.size() - 1]); | 
					
						
							|  |  |  | 		return ret; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Vector<Vector2> reduce(const Vector<Vector2> &points, const Rect2i &rect, float epsilon) { | 
					
						
							|  |  |  | 	int size = points.size(); | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	// If there are less than 3 points, then we have nothing.
 | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	ERR_FAIL_COND_V(size < 3, Vector<Vector2>()); | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	// If there are less than 9 points (but more than 3), then we don't need to reduce it.
 | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	if (size < 9) { | 
					
						
							|  |  |  | 		return points; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	float maxEp = MIN(rect.size.width, rect.size.height); | 
					
						
							|  |  |  | 	float ep = CLAMP(epsilon, 0.0, maxEp / 2); | 
					
						
							|  |  |  | 	Vector<Vector2> result = rdp(points, ep); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Vector2 last = result[result.size() - 1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (last.y > result[0].y && last.distance_to(result[0]) < ep * 0.5f) { | 
					
						
							| 
									
										
										
										
											2018-07-25 03:11:03 +02:00
										 |  |  | 		result.write[0].y = last.y; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 		result.resize(result.size() - 1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | struct FillBitsStackEntry { | 
					
						
							|  |  |  | 	Point2i pos; | 
					
						
							| 
									
										
										
										
											2021-02-09 18:24:36 +01:00
										 |  |  | 	int i = 0; | 
					
						
							|  |  |  | 	int j = 0; | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | static void fill_bits(const BitMap *p_src, Ref<BitMap> &p_map, const Point2i &p_pos, const Rect2i &rect) { | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	// Using a custom stack to work iteratively to avoid stack overflow on big bitmaps.
 | 
					
						
							| 
									
										
										
										
											2020-02-17 18:06:54 -03:00
										 |  |  | 	Vector<FillBitsStackEntry> stack; | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	// Tracking size since we won't be shrinking the stack vector.
 | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 	int stack_size = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Point2i pos = p_pos; | 
					
						
							| 
									
										
										
										
											2018-12-17 12:42:24 +01:00
										 |  |  | 	int next_i = 0; | 
					
						
							|  |  |  | 	int next_j = 0; | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	bool reenter = true; | 
					
						
							|  |  |  | 	bool popped = false; | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		if (reenter) { | 
					
						
							|  |  |  | 			next_i = pos.x - 1; | 
					
						
							|  |  |  | 			next_j = pos.y - 1; | 
					
						
							|  |  |  | 			reenter = false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (int i = next_i; i <= pos.x + 1; i++) { | 
					
						
							|  |  |  | 			for (int j = next_j; j <= pos.y + 1; j++) { | 
					
						
							|  |  |  | 				if (popped) { | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 					// The next loop over j must start normally.
 | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 					next_j = pos.y - 1; | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 					popped = false; | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 					// Skip because an iteration was already executed with current counter values.
 | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 					continue; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 				if (i < rect.position.x || i >= rect.position.x + rect.size.x) { | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 					continue; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				if (j < rect.position.y || j >= rect.position.y + rect.size.y) { | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 					continue; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 				if (p_map->get_bit(i, j)) { | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 					continue; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 				} else if (p_src->get_bit(i, j)) { | 
					
						
							|  |  |  | 					p_map->set_bit(i, j, true); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 					FillBitsStackEntry se = { pos, i, j }; | 
					
						
							|  |  |  | 					stack.resize(MAX(stack_size + 1, stack.size())); | 
					
						
							|  |  |  | 					stack.set(stack_size, se); | 
					
						
							|  |  |  | 					stack_size++; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 					pos = Point2i(i, j); | 
					
						
							|  |  |  | 					reenter = true; | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (reenter) { | 
					
						
							|  |  |  | 				break; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 		if (!reenter) { | 
					
						
							|  |  |  | 			if (stack_size) { | 
					
						
							|  |  |  | 				FillBitsStackEntry se = stack.get(stack_size - 1); | 
					
						
							|  |  |  | 				stack_size--; | 
					
						
							|  |  |  | 				pos = se.pos; | 
					
						
							|  |  |  | 				next_i = se.i; | 
					
						
							|  |  |  | 				next_j = se.j; | 
					
						
							|  |  |  | 				popped = true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} while (reenter || popped); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-10 01:37:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | Vector<Vector<Vector2>> BitMap::clip_opaque_to_polygons(const Rect2i &p_rect, float p_epsilon) const { | 
					
						
							| 
									
										
										
										
											2020-12-19 12:43:35 +00:00
										 |  |  | 	Rect2i r = Rect2i(0, 0, width, height).intersection(p_rect); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Point2i from; | 
					
						
							|  |  |  | 	Ref<BitMap> fill; | 
					
						
							| 
									
										
										
										
											2021-06-17 16:03:09 -06:00
										 |  |  | 	fill.instantiate(); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	fill->create(get_size()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-17 07:33:00 +01:00
										 |  |  | 	Vector<Vector<Vector2>> polygons; | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	for (int i = r.position.y; i < r.position.y + r.size.height; i++) { | 
					
						
							|  |  |  | 		for (int j = r.position.x; j < r.position.x + r.size.width; j++) { | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			if (!fill->get_bit(j, i) && get_bit(j, i)) { | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 				fill_bits(this, fill, Point2i(j, i), r); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 				for (Vector<Vector2> polygon : _march_square(r, Point2i(j, i))) { | 
					
						
							|  |  |  | 					polygon = reduce(polygon, r, p_epsilon); | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 					if (polygon.size() < 3) { | 
					
						
							|  |  |  | 						print_verbose("Invalid polygon, skipped"); | 
					
						
							|  |  |  | 						continue; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 13:15:31 +01:00
										 |  |  | 					polygons.push_back(polygon); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return polygons; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | void BitMap::grow_mask(int p_pixels, const Rect2i &p_rect) { | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 	if (p_pixels == 0) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-14 11:40:55 +02:00
										 |  |  | 	bool bit_value = p_pixels > 0; | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 	p_pixels = Math::abs(p_pixels); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-19 12:43:35 +00:00
										 |  |  | 	Rect2i r = Rect2i(0, 0, width, height).intersection(p_rect); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Ref<BitMap> copy; | 
					
						
							| 
									
										
										
										
											2021-06-17 16:03:09 -06:00
										 |  |  | 	copy.instantiate(); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 	copy->create(get_size()); | 
					
						
							|  |  |  | 	copy->bitmask = bitmask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (int i = r.position.y; i < r.position.y + r.size.height; i++) { | 
					
						
							|  |  |  | 		for (int j = r.position.x; j < r.position.x + r.size.width; j++) { | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			if (bit_value == get_bit(j, i)) { | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			bool found = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			for (int y = i - p_pixels; y <= i + p_pixels; y++) { | 
					
						
							|  |  |  | 				for (int x = j - p_pixels; x <= j + p_pixels; x++) { | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 					bool outside = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					if ((x < p_rect.position.x) || (x >= p_rect.position.x + p_rect.size.x) || (y < p_rect.position.y) || (y >= p_rect.position.y + p_rect.size.y)) { | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 						// Outside of rectangle counts as bit not set.
 | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 						if (!bit_value) { | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 							outside = true; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 						} else { | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 							continue; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 						} | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					float d = Point2(j, i).distance_to(Point2(x, y)) - CMP_EPSILON; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 					if (d > p_pixels) { | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 						continue; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 					if (outside || (bit_value == copy->get_bit(x, y))) { | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 						found = true; | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 				if (found) { | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 					break; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (found) { | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 				set_bit(j, i, bit_value); | 
					
						
							| 
									
										
										
										
											2018-02-21 09:38:21 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | void BitMap::shrink_mask(int p_pixels, const Rect2i &p_rect) { | 
					
						
							| 
									
										
										
										
											2019-10-06 18:45:24 +02:00
										 |  |  | 	grow_mask(-p_pixels, p_rect); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | TypedArray<PackedVector2Array> BitMap::_opaque_to_polygons_bind(const Rect2i &p_rect, float p_epsilon) const { | 
					
						
							| 
									
										
										
										
											2020-03-17 07:33:00 +01:00
										 |  |  | 	Vector<Vector<Vector2>> result = clip_opaque_to_polygons(p_rect, p_epsilon); | 
					
						
							| 
									
										
										
										
											2018-08-08 21:11:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	// Convert result to bindable types.
 | 
					
						
							| 
									
										
										
										
											2018-08-08 21:11:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 03:41:48 +02:00
										 |  |  | 	TypedArray<PackedVector2Array> result_array; | 
					
						
							| 
									
										
										
										
											2018-08-08 21:11:54 +02:00
										 |  |  | 	result_array.resize(result.size()); | 
					
						
							|  |  |  | 	for (int i = 0; i < result.size(); i++) { | 
					
						
							|  |  |  | 		const Vector<Vector2> &polygon = result[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 18:06:54 -03:00
										 |  |  | 		PackedVector2Array polygon_array; | 
					
						
							| 
									
										
										
										
											2018-08-08 21:11:54 +02:00
										 |  |  | 		polygon_array.resize(polygon.size()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2020-02-17 18:06:54 -03:00
										 |  |  | 			Vector2 *w = polygon_array.ptrw(); | 
					
						
							| 
									
										
										
										
											2018-08-08 21:11:54 +02:00
										 |  |  | 			for (int j = 0; j < polygon.size(); j++) { | 
					
						
							|  |  |  | 				w[j] = polygon[j]; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		result_array[i] = polygon_array; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return result_array; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | void BitMap::resize(const Size2i &p_new_size) { | 
					
						
							|  |  |  | 	ERR_FAIL_COND(p_new_size.width < 0 || p_new_size.height < 0); | 
					
						
							|  |  |  | 	if (p_new_size == get_size()) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-19 15:54:33 -03:00
										 |  |  | 	Ref<BitMap> new_bitmap; | 
					
						
							| 
									
										
										
										
											2021-06-17 16:03:09 -06:00
										 |  |  | 	new_bitmap.instantiate(); | 
					
						
							| 
									
										
										
										
											2019-04-19 15:54:33 -03:00
										 |  |  | 	new_bitmap->create(p_new_size); | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	// also allow for upscaling
 | 
					
						
							|  |  |  | 	int lw = (width == 0) ? 0 : p_new_size.width; | 
					
						
							|  |  |  | 	int lh = (height == 0) ? 0 : p_new_size.height; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	float scale_x = ((float)width / p_new_size.width); | 
					
						
							|  |  |  | 	float scale_y = ((float)height / p_new_size.height); | 
					
						
							| 
									
										
										
										
											2019-05-20 13:51:51 +02:00
										 |  |  | 	for (int x = 0; x < lw; x++) { | 
					
						
							|  |  |  | 		for (int y = 0; y < lh; y++) { | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			bool new_bit = get_bit(x * scale_x, y * scale_y); | 
					
						
							|  |  |  | 			new_bitmap->set_bit(x, y, new_bit); | 
					
						
							| 
									
										
										
										
											2019-04-19 15:54:33 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	width = new_bitmap->width; | 
					
						
							|  |  |  | 	height = new_bitmap->height; | 
					
						
							|  |  |  | 	bitmask = new_bitmap->bitmask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Ref<Image> BitMap::convert_to_image() const { | 
					
						
							| 
									
										
										
										
											2022-07-22 20:06:19 +02:00
										 |  |  | 	Ref<Image> image = Image::create_empty(width, height, false, Image::FORMAT_L8); | 
					
						
							| 
									
										
										
										
											2020-02-17 18:06:54 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 13:51:51 +02:00
										 |  |  | 	for (int i = 0; i < width; i++) { | 
					
						
							|  |  |  | 		for (int j = 0; j < height; j++) { | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			image->set_pixel(i, j, get_bit(i, j) ? Color(1, 1, 1) : Color(0, 0, 0)); | 
					
						
							| 
									
										
										
										
											2019-04-19 15:54:33 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return image; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-05-14 14:29:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | void BitMap::blit(const Vector2i &p_pos, const Ref<BitMap> &p_bitmap) { | 
					
						
							|  |  |  | 	ERR_FAIL_COND_MSG(p_bitmap.is_null(), "It's not a reference to a valid BitMap object."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-19 15:54:33 -03:00
										 |  |  | 	int x = p_pos.x; | 
					
						
							|  |  |  | 	int y = p_pos.y; | 
					
						
							|  |  |  | 	int w = p_bitmap->get_size().width; | 
					
						
							|  |  |  | 	int h = p_bitmap->get_size().height; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 13:51:51 +02:00
										 |  |  | 	for (int i = 0; i < w; i++) { | 
					
						
							|  |  |  | 		for (int j = 0; j < h; j++) { | 
					
						
							|  |  |  | 			int px = x + i; | 
					
						
							|  |  |  | 			int py = y + j; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 			if (px < 0 || px >= width) { | 
					
						
							| 
									
										
										
										
											2019-04-19 15:54:33 -03:00
										 |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if (py < 0 || py >= height) { | 
					
						
							| 
									
										
										
										
											2019-04-19 15:54:33 -03:00
										 |  |  | 				continue; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:41:43 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 			if (p_bitmap->get_bit(i, j)) { | 
					
						
							|  |  |  | 				set_bit(px, py, true); | 
					
						
							| 
									
										
										
										
											2019-04-19 15:54:33 -03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void BitMap::_bind_methods() { | 
					
						
							| 
									
										
										
										
											2017-02-13 12:47:24 +01:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("create", "size"), &BitMap::create); | 
					
						
							| 
									
										
										
										
											2018-02-21 11:30:55 -05:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("create_from_image_alpha", "image", "threshold"), &BitMap::create_from_image_alpha, DEFVAL(0.1)); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 18:39:17 +02:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("set_bitv", "position", "bit"), &BitMap::set_bitv); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("set_bit", "x", "y", "bit"), &BitMap::set_bit); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("get_bitv", "position"), &BitMap::get_bitv); | 
					
						
							|  |  |  | 	ClassDB::bind_method(D_METHOD("get_bit", "x", "y"), &BitMap::get_bit); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 11:10:09 +01:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("set_bit_rect", "rect", "bit"), &BitMap::set_bit_rect); | 
					
						
							| 
									
										
										
										
											2017-02-13 12:47:24 +01:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("get_true_bit_count"), &BitMap::get_true_bit_count); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-13 12:47:24 +01:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("get_size"), &BitMap::get_size); | 
					
						
							| 
									
										
										
										
											2021-12-12 01:05:25 +02:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("resize", "new_size"), &BitMap::resize); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-08 15:18:26 +03:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("_set_data", "data"), &BitMap::_set_data); | 
					
						
							| 
									
										
										
										
											2017-02-13 12:47:24 +01:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("_get_data"), &BitMap::_get_data); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-08 21:11:54 +02:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("grow_mask", "pixels", "rect"), &BitMap::grow_mask); | 
					
						
							| 
									
										
										
										
											2021-12-12 01:05:25 +02:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("convert_to_image"), &BitMap::convert_to_image); | 
					
						
							| 
									
										
										
										
											2018-08-08 21:11:54 +02:00
										 |  |  | 	ClassDB::bind_method(D_METHOD("opaque_to_polygons", "rect", "epsilon"), &BitMap::_opaque_to_polygons_bind, DEFVAL(2.0)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-03 23:06:17 +01:00
										 |  |  | 	ADD_PROPERTY(PropertyInfo(Variant::DICTIONARY, "data", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_NO_EDITOR | PROPERTY_USAGE_INTERNAL), "_set_data", "_get_data"); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-09 18:24:36 +01:00
										 |  |  | BitMap::BitMap() {} |