| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |  * Copyright 2019, Blender Foundation. | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file
 | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |  * \ingroup draw_engine | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |  * Utilities to read id buffer created in select_engine. | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "BLI_bitmap.h"
 | 
					
						
							|  |  |  | #include "BLI_bitmap_draw_2d.h"
 | 
					
						
							|  |  |  | #include "BLI_rect.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "DNA_screen_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "GPU_select.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-05 18:02:43 -03:00
										 |  |  | #include "DRW_engine.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | #include "DRW_select_buffer.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-12 12:10:44 -03:00
										 |  |  | #include "draw_manager.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Buffer of select ID's
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Read a block of pixels from the select frame buffer. */ | 
					
						
							|  |  |  | uint *DRW_select_buffer_read(const rcti *rect, uint *r_buf_len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-12 12:10:44 -03:00
										 |  |  |   struct SELECTID_Context *select_ctx = DRW_select_engine_context_get(); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* clamp rect by texture */ | 
					
						
							|  |  |  |   rcti r = { | 
					
						
							|  |  |  |       .xmin = 0, | 
					
						
							|  |  |  |       .xmax = GPU_texture_width(select_ctx->texture_u32), | 
					
						
							|  |  |  |       .ymin = 0, | 
					
						
							|  |  |  |       .ymax = GPU_texture_height(select_ctx->texture_u32), | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   rcti rect_clamp = *rect; | 
					
						
							|  |  |  |   if (BLI_rcti_isect(&r, &rect_clamp, &rect_clamp)) { | 
					
						
							|  |  |  |     uint buf_len = BLI_rcti_size_x(rect) * BLI_rcti_size_y(rect); | 
					
						
							|  |  |  |     uint *r_buf = MEM_mallocN(buf_len * sizeof(*r_buf), __func__); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DRW_opengl_context_enable(); | 
					
						
							|  |  |  |     GPU_framebuffer_bind(select_ctx->framebuffer_select_id); | 
					
						
							|  |  |  |     glReadBuffer(GL_COLOR_ATTACHMENT0); | 
					
						
							|  |  |  |     glReadPixels(rect_clamp.xmin, | 
					
						
							|  |  |  |                  rect_clamp.ymin, | 
					
						
							|  |  |  |                  BLI_rcti_size_x(&rect_clamp), | 
					
						
							|  |  |  |                  BLI_rcti_size_y(&rect_clamp), | 
					
						
							|  |  |  |                  GL_RED_INTEGER, | 
					
						
							|  |  |  |                  GL_UNSIGNED_INT, | 
					
						
							|  |  |  |                  r_buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     GPU_framebuffer_restore(); | 
					
						
							|  |  |  |     DRW_opengl_context_disable(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!BLI_rcti_compare(rect, &rect_clamp)) { | 
					
						
							|  |  |  |       GPU_select_buffer_stride_realign(rect, &rect_clamp, r_buf); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |     if (r_buf_len) { | 
					
						
							|  |  |  |       *r_buf_len = buf_len; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return r_buf; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Select Bitmap from ID's
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Given a buffer of select ID's, fill in a booleans (true/false) per index. | 
					
						
							| 
									
										
										
										
											2019-07-07 15:38:41 +10:00
										 |  |  |  * #BLI_bitmap is used for memory efficiency. | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  |  * | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * \param rect: The rectangle to sample indices from (min/max inclusive). | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |  * \param mask: Specifies the rect pixels (optional). | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |  * \returns a #BLI_bitmap the length of \a bitmap_len or NULL on failure. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | uint *DRW_select_buffer_bitmap_from_rect(const rcti *rect, uint *r_bitmap_len) | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-12 12:10:44 -03:00
										 |  |  |   struct SELECTID_Context *select_ctx = DRW_select_engine_context_get(); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const uint bitmap_len = select_ctx->last_index_drawn; | 
					
						
							| 
									
										
										
										
											2019-08-05 18:02:43 -03:00
										 |  |  |   if (bitmap_len == 0) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-03 18:41:16 -03:00
										 |  |  |   rcti rect_px = *rect; | 
					
						
							|  |  |  |   rect_px.xmax += 1; | 
					
						
							|  |  |  |   rect_px.ymax += 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   uint buf_len; | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |   uint *buf = DRW_select_buffer_read(&rect_px, &buf_len); | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   if (buf == NULL) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_bitmap *bitmap_buf = BLI_BITMAP_NEW(bitmap_len, __func__); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |   const uint *buf_iter = buf; | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   while (buf_len--) { | 
					
						
							|  |  |  |     const uint index = *buf_iter - 1; | 
					
						
							|  |  |  |     if (index < bitmap_len) { | 
					
						
							|  |  |  |       BLI_BITMAP_ENABLE(bitmap_buf, index); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     buf_iter++; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |   MEM_freeN((void *)buf); | 
					
						
							| 
									
										
										
										
											2019-08-08 04:36:21 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (r_bitmap_len) { | 
					
						
							|  |  |  |     *r_bitmap_len = bitmap_len; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   return bitmap_buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * \param bitmap_len: Number of indices in the selection id buffer. | 
					
						
							|  |  |  |  * \param center: Circle center. | 
					
						
							|  |  |  |  * \param radius: Circle radius. | 
					
						
							|  |  |  |  * \returns a #BLI_bitmap the length of \a bitmap_len or NULL on failure. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | uint *DRW_select_buffer_bitmap_from_circle(const int center[2], | 
					
						
							|  |  |  |                                            const int radius, | 
					
						
							|  |  |  |                                            uint *r_bitmap_len) | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-12 12:10:44 -03:00
										 |  |  |   struct SELECTID_Context *select_ctx = DRW_select_engine_context_get(); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const uint bitmap_len = select_ctx->last_index_drawn; | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   if (bitmap_len == 0) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-03 18:41:16 -03:00
										 |  |  |   const rcti rect = { | 
					
						
							|  |  |  |       .xmin = center[0] - radius, | 
					
						
							|  |  |  |       .xmax = center[0] + radius + 1, | 
					
						
							|  |  |  |       .ymin = center[1] - radius, | 
					
						
							|  |  |  |       .ymax = center[1] + radius + 1, | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |   const uint *buf = DRW_select_buffer_read(&rect, NULL); | 
					
						
							| 
									
										
										
										
											2019-08-05 18:02:43 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   if (buf == NULL) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const uint *buf_iter = buf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_bitmap *bitmap_buf = BLI_BITMAP_NEW(bitmap_len, __func__); | 
					
						
							|  |  |  |   const int radius_sq = radius * radius; | 
					
						
							|  |  |  |   for (int yc = -radius; yc <= radius; yc++) { | 
					
						
							|  |  |  |     for (int xc = -radius; xc <= radius; xc++, buf_iter++) { | 
					
						
							|  |  |  |       if (xc * xc + yc * yc < radius_sq) { | 
					
						
							|  |  |  |         /* Intentionally wrap to max value if this is zero. */ | 
					
						
							|  |  |  |         const uint index = *buf_iter - 1; | 
					
						
							|  |  |  |         if (index < bitmap_len) { | 
					
						
							|  |  |  |           BLI_BITMAP_ENABLE(bitmap_buf, index); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   MEM_freeN((void *)buf); | 
					
						
							| 
									
										
										
										
											2019-08-05 18:02:43 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (r_bitmap_len) { | 
					
						
							|  |  |  |     *r_bitmap_len = bitmap_len; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   return bitmap_buf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct PolyMaskData { | 
					
						
							|  |  |  |   BLI_bitmap *px; | 
					
						
							|  |  |  |   int width; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | static void drw_select_mask_px_cb(int x, int x_end, int y, void *user_data) | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | { | 
					
						
							|  |  |  |   struct PolyMaskData *data = user_data; | 
					
						
							|  |  |  |   BLI_bitmap *px = data->px; | 
					
						
							|  |  |  |   int i = (y * data->width) + x; | 
					
						
							|  |  |  |   do { | 
					
						
							|  |  |  |     BLI_BITMAP_ENABLE(px, i); | 
					
						
							|  |  |  |     i++; | 
					
						
							|  |  |  |   } while (++x != x_end); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |  * \param poly: The polygon coordinates. | 
					
						
							|  |  |  |  * \param poly_len: Length of the polygon. | 
					
						
							|  |  |  |  * \param rect: Polygon boundaries. | 
					
						
							|  |  |  |  * \returns a #BLI_bitmap. | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | uint *DRW_select_buffer_bitmap_from_poly(const int poly[][2], const int poly_len, const rcti *rect) | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-12 12:10:44 -03:00
										 |  |  |   struct SELECTID_Context *select_ctx = DRW_select_engine_context_get(); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const uint bitmap_len = select_ctx->last_index_drawn; | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   if (bitmap_len == 0) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-03 18:41:16 -03:00
										 |  |  |   rcti rect_px = *rect; | 
					
						
							|  |  |  |   rect_px.xmax += 1; | 
					
						
							|  |  |  |   rect_px.ymax += 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   uint buf_len; | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |   uint *buf = DRW_select_buffer_read(&rect_px, &buf_len); | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   if (buf == NULL) { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_bitmap *buf_mask = BLI_BITMAP_NEW(buf_len, __func__); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   struct PolyMaskData poly_mask_data; | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   poly_mask_data.px = buf_mask; | 
					
						
							|  |  |  |   poly_mask_data.width = (rect->xmax - rect->xmin) + 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-03 18:41:16 -03:00
										 |  |  |   BLI_bitmap_draw_2d_poly_v2i_n(rect_px.xmin, | 
					
						
							|  |  |  |                                 rect_px.ymin, | 
					
						
							|  |  |  |                                 rect_px.xmax, | 
					
						
							|  |  |  |                                 rect_px.ymax, | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |                                 poly, | 
					
						
							|  |  |  |                                 poly_len, | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |                                 drw_select_mask_px_cb, | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |                                 &poly_mask_data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   BLI_bitmap *bitmap_buf = BLI_BITMAP_NEW(bitmap_len, __func__); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |   const uint *buf_iter = buf; | 
					
						
							| 
									
										
										
										
											2019-05-18 23:12:47 +10:00
										 |  |  |   int i = 0; | 
					
						
							|  |  |  |   while (buf_len--) { | 
					
						
							|  |  |  |     const uint index = *buf_iter - 1; | 
					
						
							|  |  |  |     if (index < bitmap_len && BLI_BITMAP_TEST(buf_mask, i)) { | 
					
						
							|  |  |  |       BLI_BITMAP_ENABLE(bitmap_buf, index); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     buf_iter++; | 
					
						
							|  |  |  |     i++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   MEM_freeN((void *)buf); | 
					
						
							|  |  |  |   MEM_freeN(buf_mask); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return bitmap_buf; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Find Single Select ID's
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Given a buffer of select ID's, find the a single select id. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Samples a single pixel. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | uint DRW_select_buffer_sample_point(const int center[2]) | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-03 18:41:16 -03:00
										 |  |  |   const rcti rect = { | 
					
						
							|  |  |  |       .xmin = center[0], | 
					
						
							|  |  |  |       .xmax = center[0] + 1, | 
					
						
							|  |  |  |       .ymin = center[1], | 
					
						
							|  |  |  |       .ymax = center[1] + 1, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  |   uint buf_len; | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |   uint *buf = DRW_select_buffer_read(&rect, &buf_len); | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  |   BLI_assert(0 != buf_len); | 
					
						
							|  |  |  |   uint ret = buf[0]; | 
					
						
							|  |  |  |   MEM_freeN(buf); | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Find the selection id closest to \a center. | 
					
						
							| 
									
										
										
										
											2019-07-07 15:38:41 +10:00
										 |  |  |  * \param dist[in,out]: Use to initialize the distance, | 
					
						
							| 
									
										
										
										
											2019-08-01 13:53:25 +10:00
										 |  |  |  * when found, this value is set to the distance of the selection that's returned. | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | uint DRW_select_buffer_find_nearest_to_point(const int center[2], | 
					
						
							|  |  |  |                                              const uint id_min, | 
					
						
							|  |  |  |                                              const uint id_max, | 
					
						
							|  |  |  |                                              uint *dist) | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-07-07 15:38:41 +10:00
										 |  |  |   /* Smart function to sample a rect spiraling outside, nice for selection ID. */ | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Create region around center (typically the mouse cursor).
 | 
					
						
							|  |  |  |    * This must be square and have an odd width, | 
					
						
							|  |  |  |    * the spiraling algorithm does not work with arbitrary rectangles. */ | 
					
						
							| 
									
										
										
										
											2019-08-05 18:02:43 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   uint index = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  |   rcti rect; | 
					
						
							|  |  |  |   BLI_rcti_init_pt_radius(&rect, center, *dist); | 
					
						
							|  |  |  |   rect.xmax += 1; | 
					
						
							|  |  |  |   rect.ymax += 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int width = BLI_rcti_size_x(&rect); | 
					
						
							|  |  |  |   int height = width; | 
					
						
							|  |  |  |   BLI_assert(width == height); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Read from selection framebuffer. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint buf_len; | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  |   const uint *buf = DRW_select_buffer_read(&rect, &buf_len); | 
					
						
							| 
									
										
										
										
											2019-08-05 18:02:43 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (buf == NULL) { | 
					
						
							|  |  |  |     return index; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-20 13:16:29 +10:00
										 |  |  |   BLI_assert(width * height == buf_len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Spiral, starting from center of buffer. */ | 
					
						
							|  |  |  |   int spiral_offset = height * (int)(width / 2) + (height / 2); | 
					
						
							|  |  |  |   int spiral_direction = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (int nr = 1; nr <= height; nr++) { | 
					
						
							|  |  |  |     for (int a = 0; a < 2; a++) { | 
					
						
							|  |  |  |       for (int b = 0; b < nr; b++) { | 
					
						
							|  |  |  |         /* Find hit within the specified range. */ | 
					
						
							|  |  |  |         uint hit_id = buf[spiral_offset]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (hit_id && hit_id >= id_min && hit_id < id_max) { | 
					
						
							|  |  |  |           /* Get x/y from spiral offset. */ | 
					
						
							|  |  |  |           int hit_x = spiral_offset % width; | 
					
						
							|  |  |  |           int hit_y = spiral_offset / width; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           int center_x = width / 2; | 
					
						
							|  |  |  |           int center_y = height / 2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* Manhatten distance in keeping with other screen-based selection. */ | 
					
						
							|  |  |  |           *dist = (uint)(abs(hit_x - center_x) + abs(hit_y - center_y)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           /* Indices start at 1 here. */ | 
					
						
							|  |  |  |           index = (hit_id - id_min) + 1; | 
					
						
							|  |  |  |           goto exit; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* Next spiral step. */ | 
					
						
							|  |  |  |         if (spiral_direction == 0) { | 
					
						
							|  |  |  |           spiral_offset += 1; /* right */ | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (spiral_direction == 1) { | 
					
						
							|  |  |  |           spiral_offset -= width; /* down */ | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (spiral_direction == 2) { | 
					
						
							|  |  |  |           spiral_offset -= 1; /* left */ | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |           spiral_offset += width; /* up */ | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* Stop if we are outside the buffer. */ | 
					
						
							|  |  |  |         if (spiral_offset < 0 || spiral_offset >= buf_len) { | 
					
						
							|  |  |  |           goto exit; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       spiral_direction = (spiral_direction + 1) % 4; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | exit: | 
					
						
							|  |  |  |   MEM_freeN((void *)buf); | 
					
						
							|  |  |  |   return index; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Object Utils
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool DRW_select_buffer_elem_get(const uint sel_id, | 
					
						
							|  |  |  |                                 uint *r_elem, | 
					
						
							|  |  |  |                                 uint *r_base_index, | 
					
						
							|  |  |  |                                 char *r_elem_type) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-12 12:10:44 -03:00
										 |  |  |   struct SELECTID_Context *select_ctx = DRW_select_engine_context_get(); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   char elem_type = 0; | 
					
						
							|  |  |  |   uint elem_id; | 
					
						
							|  |  |  |   uint base_index = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (; base_index < select_ctx->objects_len; base_index++) { | 
					
						
							| 
									
										
										
										
											2019-08-12 12:29:18 -03:00
										 |  |  |     struct ObjectOffsets *base_ofs = &select_ctx->index_offsets[base_index]; | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (base_ofs->face > sel_id) { | 
					
						
							|  |  |  |       elem_id = sel_id - base_ofs->face_start; | 
					
						
							|  |  |  |       elem_type = SCE_SELECT_FACE; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (base_ofs->edge > sel_id) { | 
					
						
							|  |  |  |       elem_id = sel_id - base_ofs->edge_start; | 
					
						
							|  |  |  |       elem_type = SCE_SELECT_EDGE; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (base_ofs->vert > sel_id) { | 
					
						
							|  |  |  |       elem_id = sel_id - base_ofs->vert_start; | 
					
						
							|  |  |  |       elem_type = SCE_SELECT_VERTEX; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (base_index == select_ctx->objects_len) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   *r_elem = elem_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (r_base_index) { | 
					
						
							|  |  |  |     *r_base_index = base_index; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (r_elem_type) { | 
					
						
							|  |  |  |     *r_elem_type = elem_type; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint DRW_select_buffer_context_offset_for_object_elem(const uint base_index, char elem_type) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-12 12:10:44 -03:00
										 |  |  |   struct SELECTID_Context *select_ctx = DRW_select_engine_context_get(); | 
					
						
							| 
									
										
										
										
											2019-08-12 12:29:18 -03:00
										 |  |  |   struct ObjectOffsets *base_ofs = &select_ctx->index_offsets[base_index]; | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (elem_type == SCE_SELECT_VERTEX) { | 
					
						
							|  |  |  |     return base_ofs->vert_start - 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (elem_type == SCE_SELECT_EDGE) { | 
					
						
							|  |  |  |     return base_ofs->edge_start - 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (elem_type == SCE_SELECT_FACE) { | 
					
						
							|  |  |  |     return base_ofs->face_start - 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   BLI_assert(0); | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Context
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DRW_select_buffer_context_create(Base **UNUSED(bases), | 
					
						
							|  |  |  |                                       const uint bases_len, | 
					
						
							|  |  |  |                                       short select_mode) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-08-12 12:10:44 -03:00
										 |  |  |   struct SELECTID_Context *select_ctx = DRW_select_engine_context_get(); | 
					
						
							| 
									
										
										
										
											2019-08-07 12:43:04 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |   select_ctx->select_mode = select_mode; | 
					
						
							|  |  |  |   select_ctx->objects_len = bases_len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MEM_SAFE_FREE(select_ctx->index_offsets); | 
					
						
							|  |  |  |   select_ctx->index_offsets = MEM_mallocN(sizeof(*select_ctx->index_offsets) * bases_len, | 
					
						
							|  |  |  |                                           __func__); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Legacy
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DRW_draw_select_id_object(Depsgraph *depsgraph, | 
					
						
							|  |  |  |                                ViewLayer *view_layer, | 
					
						
							|  |  |  |                                ARegion *ar, | 
					
						
							|  |  |  |                                View3D *v3d, | 
					
						
							|  |  |  |                                Object *ob, | 
					
						
							|  |  |  |                                short select_mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Base *base = BKE_view_layer_base_find(view_layer, ob); | 
					
						
							|  |  |  |   DRW_draw_select_id(depsgraph, ar, v3d, &base, 1, select_mode); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ |