| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright 2016, Blender Foundation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Blender Institute | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file overlay_mode.c
 | 
					
						
							|  |  |  |  *  \ingroup draw_engine | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-28 09:01:34 +02:00
										 |  |  | #include "DNA_view3d_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | #include "BKE_object.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | #include "GPU_shader.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-03 15:13:33 +02:00
										 |  |  | #include "GPU_extensions.h"
 | 
					
						
							| 
									
										
										
										
											2018-04-23 14:46:52 +02:00
										 |  |  | #include "DRW_render.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "draw_mode_engines.h"
 | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Structures */ | 
					
						
							|  |  |  | typedef struct OVERLAY_StorageList { | 
					
						
							|  |  |  | 	struct OVERLAY_PrivateData *g_data; | 
					
						
							|  |  |  | } OVERLAY_StorageList; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct OVERLAY_PassList { | 
					
						
							|  |  |  | 	struct DRWPass *face_orientation_pass; | 
					
						
							| 
									
										
										
										
											2018-09-14 18:30:26 +02:00
										 |  |  | 	struct DRWPass *flat_wireframe_pass; | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 	struct DRWPass *face_wireframe_pass; | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 	struct DRWPass *face_wireframe_full_pass; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | } OVERLAY_PassList; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct OVERLAY_Data { | 
					
						
							|  |  |  | 	void *engine_type; | 
					
						
							|  |  |  | 	DRWViewportEmptyList *fbl; | 
					
						
							|  |  |  | 	DRWViewportEmptyList *txl; | 
					
						
							|  |  |  | 	OVERLAY_PassList *psl; | 
					
						
							|  |  |  | 	OVERLAY_StorageList *stl; | 
					
						
							|  |  |  | } OVERLAY_Data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct OVERLAY_PrivateData { | 
					
						
							|  |  |  | 	DRWShadingGroup *face_orientation_shgrp; | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 	DRWShadingGroup *sculpt_wires_full; | 
					
						
							|  |  |  | 	DRWShadingGroup *sculpt_wires; | 
					
						
							| 
									
										
										
										
											2018-05-05 10:45:15 +02:00
										 |  |  | 	View3DOverlay overlay; | 
					
						
							| 
									
										
										
										
											2018-06-05 19:35:08 +02:00
										 |  |  | 	float wire_step_param[2]; | 
					
						
							| 
									
										
										
										
											2018-08-15 17:43:39 +02:00
										 |  |  | 	bool ghost_stencil_test; | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 	bool show_overlays; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | } OVERLAY_PrivateData; /* Transient data */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* *********** STATIC *********** */ | 
					
						
							|  |  |  | static struct { | 
					
						
							| 
									
										
										
										
											2018-04-20 12:46:37 +02:00
										 |  |  | 	/* Face orientation shader */ | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 	struct GPUShader *face_orientation_sh; | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 	/* Wireframe shader */ | 
					
						
							| 
									
										
										
										
											2018-09-04 17:14:46 +02:00
										 |  |  | 	struct GPUShader *select_wireframe_sh; | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 	struct GPUShader *face_wireframe_sh; | 
					
						
							| 
									
										
										
										
											2018-06-05 19:35:08 +02:00
										 |  |  | 	struct GPUShader *face_wireframe_pretty_sh; | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 	struct GPUShader *face_wireframe_sculpt_sh; | 
					
						
							|  |  |  | 	struct GPUShader *face_wireframe_sculpt_pretty_sh; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | } e_data = {NULL}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Shaders */ | 
					
						
							|  |  |  | extern char datatoc_overlay_face_orientation_frag_glsl[]; | 
					
						
							|  |  |  | extern char datatoc_overlay_face_orientation_vert_glsl[]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | extern char datatoc_overlay_face_wireframe_vert_glsl[]; | 
					
						
							| 
									
										
										
										
											2018-06-03 15:13:33 +02:00
										 |  |  | extern char datatoc_overlay_face_wireframe_geom_glsl[]; | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | extern char datatoc_overlay_face_wireframe_frag_glsl[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern struct GlobalsUboStorage ts; /* draw_common.c */ | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Functions */ | 
					
						
							| 
									
										
										
										
											2018-04-23 14:46:52 +02:00
										 |  |  | static void overlay_engine_init(void *vedata) | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-10-11 09:08:30 +11:00
										 |  |  | 	OVERLAY_Data *data = vedata; | 
					
						
							| 
									
										
										
										
											2018-04-23 14:46:52 +02:00
										 |  |  | 	OVERLAY_StorageList *stl = data->stl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!stl->g_data) { | 
					
						
							|  |  |  | 		/* Alloc transient pointers */ | 
					
						
							|  |  |  | 		stl->g_data = MEM_mallocN(sizeof(*stl->g_data), __func__); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-15 17:43:39 +02:00
										 |  |  | 	stl->g_data->ghost_stencil_test = false; | 
					
						
							| 
									
										
										
										
											2018-04-23 14:46:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-20 12:46:37 +02:00
										 |  |  | 	if (!e_data.face_orientation_sh) { | 
					
						
							|  |  |  | 		/* Face orientation */ | 
					
						
							| 
									
										
										
										
											2018-05-26 10:41:25 +02:00
										 |  |  | 		e_data.face_orientation_sh = DRW_shader_create( | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 		        datatoc_overlay_face_orientation_vert_glsl, NULL, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_orientation_frag_glsl, NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!e_data.face_wireframe_sh) { | 
					
						
							| 
									
										
										
										
											2018-06-05 19:35:08 +02:00
										 |  |  | 		bool use_geom = GPU_type_matches(GPU_DEVICE_INTEL, GPU_OS_ANY, GPU_DRIVER_ANY); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:14:46 +02:00
										 |  |  | 		e_data.select_wireframe_sh = DRW_shader_create( | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_vert_glsl, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_geom_glsl, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_frag_glsl, | 
					
						
							|  |  |  | 		        "#define SELECT_EDGES\n" | 
					
						
							|  |  |  | 		        "#define LIGHT_EDGES\n" | 
					
						
							|  |  |  | 		        "#define USE_GEOM_SHADER\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 		e_data.face_wireframe_sh = DRW_shader_create( | 
					
						
							| 
									
										
										
										
											2018-06-03 15:13:33 +02:00
										 |  |  | 		        datatoc_overlay_face_wireframe_vert_glsl, | 
					
						
							| 
									
										
										
										
											2018-06-05 19:35:08 +02:00
										 |  |  | 		        use_geom ? datatoc_overlay_face_wireframe_geom_glsl : NULL, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_frag_glsl, | 
					
						
							|  |  |  | 		        use_geom ? "#define USE_GEOM_SHADER\n" | 
					
						
							|  |  |  | 		                 : NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		e_data.face_wireframe_pretty_sh = DRW_shader_create( | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_vert_glsl, | 
					
						
							|  |  |  | 		        use_geom ? datatoc_overlay_face_wireframe_geom_glsl : NULL, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_frag_glsl, | 
					
						
							|  |  |  | 		        use_geom ? "#define USE_GEOM_SHADER\n" | 
					
						
							|  |  |  | 		                   "#define LIGHT_EDGES\n" | 
					
						
							|  |  |  | 		                 : "#define LIGHT_EDGES\n"); | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		e_data.face_wireframe_sculpt_sh = DRW_shader_create( | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_vert_glsl, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_geom_glsl, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_frag_glsl, | 
					
						
							|  |  |  | 		        "#define USE_SCULPT\n" | 
					
						
							|  |  |  | 		        "#define USE_GEOM_SHADER\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		e_data.face_wireframe_sculpt_pretty_sh = DRW_shader_create( | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_vert_glsl, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_geom_glsl, | 
					
						
							|  |  |  | 		        datatoc_overlay_face_wireframe_frag_glsl, | 
					
						
							|  |  |  | 		        "#define USE_SCULPT\n" | 
					
						
							|  |  |  | 		        "#define USE_GEOM_SHADER\n" | 
					
						
							|  |  |  | 		        "#define LIGHT_EDGES\n"); | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void overlay_cache_init(void *vedata) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-10-11 09:08:30 +11:00
										 |  |  | 	OVERLAY_Data *data = vedata; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 	OVERLAY_PassList *psl = data->psl; | 
					
						
							|  |  |  | 	OVERLAY_StorageList *stl = data->stl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const DRWContextState *DCS = DRW_context_state_get(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	View3D *v3d = DCS->v3d; | 
					
						
							|  |  |  | 	if (v3d) { | 
					
						
							| 
									
										
										
										
											2018-05-05 10:45:15 +02:00
										 |  |  | 		stl->g_data->overlay = v3d->overlay; | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 		stl->g_data->show_overlays = (v3d->flag2 & V3D_RENDER_OVERRIDE) == 0; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2018-05-05 10:45:15 +02:00
										 |  |  | 		memset(&stl->g_data->overlay, 0, sizeof(stl->g_data->overlay)); | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 		stl->g_data->show_overlays = false; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 18:29:05 +02:00
										 |  |  | 	if (stl->g_data->show_overlays == false) { | 
					
						
							|  |  |  | 		stl->g_data->overlay.flag = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (v3d->shading.type == OB_WIRE) { | 
					
						
							|  |  |  | 		stl->g_data->overlay.flag |= V3D_OVERLAY_WIREFRAMES; | 
					
						
							|  |  |  | 		stl->g_data->show_overlays = true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		/* Face Orientation Pass */ | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 		DRWState state = DRW_STATE_WRITE_COLOR | DRW_STATE_DEPTH_EQUAL | DRW_STATE_BLEND; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 		psl->face_orientation_pass = DRW_pass_create("Face Orientation", state); | 
					
						
							| 
									
										
										
										
											2018-05-26 10:41:25 +02:00
										 |  |  | 		stl->g_data->face_orientation_shgrp = DRW_shgroup_create( | 
					
						
							|  |  |  | 		        e_data.face_orientation_sh, psl->face_orientation_pass); | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* Wireframe */ | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 		DRWState state = DRW_STATE_WRITE_COLOR | DRW_STATE_DEPTH_LESS_EQUAL | DRW_STATE_BLEND; | 
					
						
							| 
									
										
										
										
											2018-10-30 19:07:27 +01:00
										 |  |  | 		float wire_size = max_ff(0.0f, U.pixelsize - 1.0f) * 0.5f; | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-14 18:30:26 +02:00
										 |  |  | 		psl->flat_wireframe_pass = DRW_pass_create("Flat Object Wires", state | DRW_STATE_WRITE_DEPTH); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 		psl->face_wireframe_full_pass = DRW_pass_create("All Face Wires", state); | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		stl->g_data->sculpt_wires_full = DRW_shgroup_create(e_data.face_wireframe_sculpt_sh, psl->face_wireframe_full_pass); | 
					
						
							| 
									
										
										
										
											2018-10-30 19:07:27 +01:00
										 |  |  | 		DRW_shgroup_uniform_float_copy(stl->g_data->sculpt_wires_full, "wireSize", wire_size); | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 		DRWShadingGroup *shgrp = DRW_shgroup_create(e_data.face_wireframe_sh, psl->face_wireframe_full_pass); | 
					
						
							| 
									
										
										
										
											2018-10-30 19:07:27 +01:00
										 |  |  | 		DRW_shgroup_uniform_float_copy(shgrp, "wireSize", wire_size); | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 		psl->face_wireframe_pass = DRW_pass_create("Face Wires", state); | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		stl->g_data->sculpt_wires = DRW_shgroup_create(e_data.face_wireframe_sculpt_pretty_sh, psl->face_wireframe_pass); | 
					
						
							|  |  |  | 		DRW_shgroup_uniform_vec2(stl->g_data->sculpt_wires, "wireStepParam", stl->g_data->wire_step_param, 1); | 
					
						
							| 
									
										
										
										
											2018-10-30 19:07:27 +01:00
										 |  |  | 		DRW_shgroup_uniform_float_copy(stl->g_data->sculpt_wires, "wireSize", wire_size); | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 		shgrp = DRW_shgroup_create(e_data.face_wireframe_pretty_sh, psl->face_wireframe_pass); | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 		DRW_shgroup_uniform_vec2(shgrp, "wireStepParam", stl->g_data->wire_step_param, 1); | 
					
						
							| 
									
										
										
										
											2018-10-30 19:07:27 +01:00
										 |  |  | 		DRW_shgroup_uniform_float_copy(shgrp, "wireSize", wire_size); | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 		/**
 | 
					
						
							|  |  |  | 		 * The wireframe threshold ranges from 0.0 to 1.0 | 
					
						
							|  |  |  | 		 * When 1.0 we show all the edges, when 0.5 we show as many as 2.7. | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * If we wanted 0.0 to match 2.7, factor would need to be 0.003f. | 
					
						
							|  |  |  | 		 * The range controls the falloff effect. If range was 0.0f we would get a hard cut (as in 2.7). | 
					
						
							|  |  |  | 		 * That said we are using a different algorithm so the results will always differ. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		const float factor = 0.0045f; | 
					
						
							|  |  |  | 		const float range = 0.00125f; | 
					
						
							|  |  |  | 		stl->g_data->wire_step_param[1] = (1.0f - factor) + stl->g_data->overlay.wireframe_threshold * factor; | 
					
						
							|  |  |  | 		stl->g_data->wire_step_param[0] = stl->g_data->wire_step_param[1] + range; | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void overlay_cache_populate(void *vedata, Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-10-11 09:08:30 +11:00
										 |  |  | 	OVERLAY_Data *data = vedata; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 	OVERLAY_StorageList *stl = data->stl; | 
					
						
							|  |  |  | 	OVERLAY_PrivateData *pd = stl->g_data; | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 	OVERLAY_PassList *psl = data->psl; | 
					
						
							|  |  |  | 	const DRWContextState *draw_ctx = DRW_context_state_get(); | 
					
						
							| 
									
										
										
										
											2018-09-14 18:30:26 +02:00
										 |  |  | 	RegionView3D *rv3d = draw_ctx->rv3d; | 
					
						
							| 
									
										
										
										
											2018-09-04 17:14:46 +02:00
										 |  |  | 	View3D *v3d = draw_ctx->v3d; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-17 13:27:28 +01:00
										 |  |  | 	if ((!stl->g_data->show_overlays) || | 
					
						
							|  |  |  | 		(ob->dt < OB_WIRE) || | 
					
						
							|  |  |  | 		(!DRW_object_is_renderable(ob) && (ob->dt != OB_WIRE))) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2018-11-17 13:27:28 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 13:18:41 +02:00
										 |  |  | 	if (DRW_object_is_renderable(ob) && stl->g_data->overlay.flag & V3D_OVERLAY_FACE_ORIENTATION) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 		struct GPUBatch *geom = DRW_cache_object_surface_get(ob); | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 		if (geom) { | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 			DRW_shgroup_call_add(pd->face_orientation_shgrp, geom, ob->obmat); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-20 13:18:41 +02:00
										 |  |  | 	if ((stl->g_data->overlay.flag & V3D_OVERLAY_WIREFRAMES) || | 
					
						
							| 
									
										
										
										
											2018-09-21 14:10:06 +02:00
										 |  |  | 	    (v3d->shading.type == OB_WIRE) || | 
					
						
							| 
									
										
										
										
											2018-08-20 13:18:41 +02:00
										 |  |  | 	    (ob->dtx & OB_DRAWWIRE) || | 
					
						
							|  |  |  | 	    (ob->dt == OB_WIRE)) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 		/* Don't do that in edit mode. */ | 
					
						
							|  |  |  | 		if ((ob != draw_ctx->object_edit) && !BKE_object_is_in_editmode(ob)) { | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 			const bool is_active = (ob == draw_ctx->obact); | 
					
						
							|  |  |  | 			const bool is_sculpt_mode = is_active && (draw_ctx->object_mode & OB_MODE_SCULPT) != 0; | 
					
						
							|  |  |  | 			const bool all_wires = (stl->g_data->overlay.wireframe_threshold == 1.0f) || | 
					
						
							|  |  |  | 			                       (ob->dtx & OB_DRAW_ALL_EDGES); | 
					
						
							| 
									
										
										
										
											2018-10-22 19:44:45 +02:00
										 |  |  | 			const bool is_wire = (ob->dt < OB_SOLID); | 
					
						
							|  |  |  | 			const int stencil_mask = (ob->dtx & OB_DRAWXRAY) ? 0x00 : 0xFF; | 
					
						
							|  |  |  | 			DRWShadingGroup *shgrp = NULL; | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-14 18:30:26 +02:00
										 |  |  | 			/* This fixes only the biggest case which is a plane in ortho view. */ | 
					
						
							|  |  |  | 			int flat_axis = 0; | 
					
						
							|  |  |  | 			bool is_flat_object_viewed_from_side = (rv3d->persp == RV3D_ORTHO) && | 
					
						
							|  |  |  | 			                                       DRW_object_is_flat(ob, &flat_axis) && | 
					
						
							|  |  |  | 			                                       DRW_object_axis_orthogonal_to_view(ob, flat_axis); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 			if (is_sculpt_mode) { | 
					
						
							| 
									
										
										
										
											2018-10-22 19:44:45 +02:00
										 |  |  | 				shgrp = (all_wires || DRW_object_is_flat_normal(ob)) | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 				                         ? stl->g_data->sculpt_wires_full | 
					
						
							|  |  |  | 				                         : stl->g_data->sculpt_wires; | 
					
						
							| 
									
										
										
										
											2018-10-22 19:44:45 +02:00
										 |  |  | 				if (is_wire) { | 
					
						
							|  |  |  | 					shgrp = DRW_shgroup_create_sub(shgrp); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 				DRW_shgroup_call_sculpt_add(shgrp, ob, ob->obmat); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-09-14 18:30:26 +02:00
										 |  |  | 			else if (is_flat_object_viewed_from_side) { | 
					
						
							| 
									
										
										
										
											2018-09-19 18:19:49 +02:00
										 |  |  | 				/* Avoid losing flat objects when in ortho views (see T56549) */ | 
					
						
							| 
									
										
										
										
											2018-09-14 18:30:26 +02:00
										 |  |  | 				struct GPUBatch *geom = DRW_cache_object_wire_outline_get(ob); | 
					
						
							|  |  |  | 				GPUShader *sh = GPU_shader_get_builtin_shader(GPU_SHADER_3D_UNIFORM_COLOR); | 
					
						
							| 
									
										
										
										
											2018-10-22 19:44:45 +02:00
										 |  |  | 				shgrp = DRW_shgroup_create(sh, psl->flat_wireframe_pass); | 
					
						
							|  |  |  | 				DRW_shgroup_stencil_mask(shgrp, stencil_mask); | 
					
						
							| 
									
										
										
										
											2018-09-14 18:30:26 +02:00
										 |  |  | 				DRW_shgroup_uniform_vec4(shgrp, "color", ts.colorWire, 1); | 
					
						
							|  |  |  | 				DRW_shgroup_call_object_add(shgrp, geom, ob); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 			else { | 
					
						
							|  |  |  | 				int tri_count; | 
					
						
							|  |  |  | 				GPUTexture *verts = NULL, *faceids; | 
					
						
							|  |  |  | 				DRW_cache_object_face_wireframe_get(ob, &verts, &faceids, &tri_count); | 
					
						
							|  |  |  | 				if (verts) { | 
					
						
							|  |  |  | 					float *rim_col = ts.colorWire; | 
					
						
							|  |  |  | 					if ((ob->base_flag & BASE_SELECTED) != 0) { | 
					
						
							|  |  |  | 						rim_col = (ob == draw_ctx->obact) ? ts.colorActive : ts.colorSelect; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					DRWPass *pass = (all_wires) ? psl->face_wireframe_full_pass : psl->face_wireframe_pass; | 
					
						
							|  |  |  | 					GPUShader *sh = (all_wires) ? e_data.face_wireframe_sh : e_data.face_wireframe_pretty_sh; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-01 18:53:23 +02:00
										 |  |  | 					if ((DRW_state_is_select() || DRW_state_is_depth())) { | 
					
						
							| 
									
										
										
										
											2018-09-04 17:14:46 +02:00
										 |  |  | 						static float params[2] = {1.2f, 1.0f}; /* Parameters for all wires */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						sh = e_data.select_wireframe_sh; | 
					
						
							| 
									
										
										
										
											2018-10-22 19:44:45 +02:00
										 |  |  | 						shgrp = DRW_shgroup_create(sh, pass); | 
					
						
							| 
									
										
										
										
											2018-09-04 17:14:46 +02:00
										 |  |  | 						DRW_shgroup_uniform_vec2(shgrp, "wireStepParam", (all_wires) | 
					
						
							|  |  |  | 						                                                 ? params | 
					
						
							|  |  |  | 						                                                 : stl->g_data->wire_step_param, 1); | 
					
						
							|  |  |  | 						DRW_shgroup_uniform_texture(shgrp, "vertData", verts); | 
					
						
							|  |  |  | 						DRW_shgroup_uniform_texture(shgrp, "faceIds", faceids); | 
					
						
							|  |  |  | 						DRW_shgroup_call_object_procedural_triangles_culled_add(shgrp, tri_count, ob); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							| 
									
										
										
										
											2018-10-22 19:44:45 +02:00
										 |  |  | 						shgrp = DRW_shgroup_create(sh, pass); | 
					
						
							|  |  |  | 						DRW_shgroup_stencil_mask(shgrp, stencil_mask); | 
					
						
							| 
									
										
										
										
											2018-09-04 17:14:46 +02:00
										 |  |  | 						DRW_shgroup_uniform_texture(shgrp, "vertData", verts); | 
					
						
							|  |  |  | 						DRW_shgroup_uniform_texture(shgrp, "faceIds", faceids); | 
					
						
							|  |  |  | 						DRW_shgroup_uniform_vec3(shgrp, "wireColor", ts.colorWire, 1); | 
					
						
							|  |  |  | 						DRW_shgroup_uniform_vec3(shgrp, "rimColor", rim_col, 1); | 
					
						
							|  |  |  | 						DRW_shgroup_call_object_procedural_triangles_culled_add(shgrp, tri_count, ob); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-08-15 17:43:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-22 19:44:45 +02:00
										 |  |  | 			if (is_wire && shgrp != NULL) { | 
					
						
							|  |  |  | 				/* If object is wireframe, don't try to use stencil test. */ | 
					
						
							|  |  |  | 				DRW_shgroup_state_disable(shgrp, DRW_STATE_STENCIL_EQUAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (ob->dtx & OB_DRAWXRAY) { | 
					
						
							|  |  |  | 					DRW_shgroup_state_disable(shgrp, DRW_STATE_DEPTH_LESS_EQUAL); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if ((ob->dtx & OB_DRAWXRAY) && shgrp != NULL) { | 
					
						
							|  |  |  | 				stl->g_data->ghost_stencil_test = true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-15 17:43:39 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-15 17:43:39 +02:00
										 |  |  | static void overlay_cache_finish(void *vedata) | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-10-11 09:08:30 +11:00
										 |  |  | 	OVERLAY_Data *data = vedata; | 
					
						
							| 
									
										
										
										
											2018-08-15 17:43:39 +02:00
										 |  |  | 	OVERLAY_PassList *psl = data->psl; | 
					
						
							|  |  |  | 	OVERLAY_StorageList *stl = data->stl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const DRWContextState *ctx = DRW_context_state_get(); | 
					
						
							|  |  |  | 	View3D *v3d = ctx->v3d; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* only in solid mode */ | 
					
						
							| 
									
										
										
										
											2018-09-26 20:31:14 +02:00
										 |  |  | 	if (v3d->shading.type == OB_SOLID && (v3d->shading.flag & XRAY_FLAG(v3d)) == 0) { | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 		if (stl->g_data->ghost_stencil_test) { | 
					
						
							| 
									
										
										
										
											2018-08-15 17:43:39 +02:00
										 |  |  | 			DRW_pass_state_add(psl->face_wireframe_pass, DRW_STATE_STENCIL_EQUAL); | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 			DRW_pass_state_add(psl->face_wireframe_full_pass, DRW_STATE_STENCIL_EQUAL); | 
					
						
							| 
									
										
										
										
											2018-08-16 23:51:15 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-15 17:43:39 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void overlay_draw_scene(void *vedata) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-10-11 09:08:30 +11:00
										 |  |  | 	OVERLAY_Data *data = vedata; | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | 	OVERLAY_PassList *psl = data->psl; | 
					
						
							| 
									
										
										
										
											2018-08-22 14:18:02 +02:00
										 |  |  | 	DefaultFramebufferList *dfbl = DRW_viewport_framebuffer_list_get(); | 
					
						
							| 
									
										
										
										
											2018-04-23 14:46:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 17:14:46 +02:00
										 |  |  | 	if (DRW_state_is_fbo()) { | 
					
						
							|  |  |  | 		GPU_framebuffer_bind(dfbl->default_fb); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 	DRW_draw_pass(psl->face_orientation_pass); | 
					
						
							| 
									
										
										
										
											2018-09-14 18:30:26 +02:00
										 |  |  | 	DRW_draw_pass(psl->flat_wireframe_pass); | 
					
						
							| 
									
										
										
										
											2018-08-20 10:45:02 +02:00
										 |  |  | 	DRW_draw_pass(psl->face_wireframe_pass); | 
					
						
							|  |  |  | 	DRW_draw_pass(psl->face_wireframe_full_pass); | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void overlay_engine_free(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	DRW_SHADER_FREE_SAFE(e_data.face_orientation_sh); | 
					
						
							| 
									
										
										
										
											2018-09-04 17:14:46 +02:00
										 |  |  | 	DRW_SHADER_FREE_SAFE(e_data.select_wireframe_sh); | 
					
						
							| 
									
										
										
										
											2018-05-31 19:01:22 +02:00
										 |  |  | 	DRW_SHADER_FREE_SAFE(e_data.face_wireframe_sh); | 
					
						
							| 
									
										
										
										
											2018-06-05 19:35:08 +02:00
										 |  |  | 	DRW_SHADER_FREE_SAFE(e_data.face_wireframe_pretty_sh); | 
					
						
							| 
									
										
										
										
											2018-08-23 18:05:50 +02:00
										 |  |  | 	DRW_SHADER_FREE_SAFE(e_data.face_wireframe_sculpt_sh); | 
					
						
							|  |  |  | 	DRW_SHADER_FREE_SAFE(e_data.face_wireframe_sculpt_pretty_sh); | 
					
						
							| 
									
										
										
										
											2018-04-20 10:45:46 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const DrawEngineDataSize overlay_data_size = DRW_VIEWPORT_DATA_SIZE(OVERLAY_Data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DrawEngineType draw_engine_overlay_type = { | 
					
						
							|  |  |  | 	NULL, NULL, | 
					
						
							|  |  |  | 	N_("OverlayEngine"), | 
					
						
							|  |  |  | 	&overlay_data_size, | 
					
						
							|  |  |  | 	&overlay_engine_init, | 
					
						
							|  |  |  | 	&overlay_engine_free, | 
					
						
							|  |  |  | 	&overlay_cache_init, | 
					
						
							|  |  |  | 	&overlay_cache_populate, | 
					
						
							|  |  |  | 	&overlay_cache_finish, | 
					
						
							|  |  |  | 	NULL, | 
					
						
							|  |  |  | 	&overlay_draw_scene, | 
					
						
							|  |  |  | 	NULL, | 
					
						
							|  |  |  | 	NULL, | 
					
						
							|  |  |  | 	NULL, | 
					
						
							|  |  |  | }; |