| 
									
										
										
										
											2017-03-12 21:16:03 +01: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 blender/draw/modes/paint_texture_mode.c
 | 
					
						
							|  |  |  |  *  \ingroup draw | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "DRW_engine.h"
 | 
					
						
							|  |  |  | #include "DRW_render.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | #include "BIF_gl.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | /* If builtin shaders are needed */ | 
					
						
							|  |  |  | #include "GPU_shader.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | #include "GPU_texture.h"
 | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "draw_common.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "draw_mode_engines.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern char datatoc_common_globals_lib_glsl[]; | 
					
						
							|  |  |  | extern char datatoc_paint_texture_vert_glsl[]; | 
					
						
							|  |  |  | extern char datatoc_paint_texture_frag_glsl[]; | 
					
						
							|  |  |  | extern char datatoc_paint_wire_vert_glsl[]; | 
					
						
							|  |  |  | extern char datatoc_paint_wire_frag_glsl[]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 22:14:53 +01:00
										 |  |  | /* If needed, contains all global/Theme colors
 | 
					
						
							|  |  |  |  * Add needed theme colors / values to DRW_globals_update() and update UBO | 
					
						
							|  |  |  |  * Not needed for constant color. */ | 
					
						
							|  |  |  | extern struct GPUUniformBuffer *globals_ubo; /* draw_common.c */ | 
					
						
							|  |  |  | extern struct GlobalsUboStorage ts; /* draw_common.c */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | /* *********** LISTS *********** */ | 
					
						
							|  |  |  | /* All lists are per viewport specific datas.
 | 
					
						
							|  |  |  |  * They are all free when viewport changes engines | 
					
						
							|  |  |  |  * or is free itself. Use PAINT_TEXTURE_engine_init() to | 
					
						
							|  |  |  |  * initialize most of them and PAINT_TEXTURE_cache_init() | 
					
						
							|  |  |  |  * for PAINT_TEXTURE_PassList */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct PAINT_TEXTURE_PassList { | 
					
						
							|  |  |  | 	/* Declare all passes here and init them in
 | 
					
						
							|  |  |  | 	 * PAINT_TEXTURE_cache_init(). | 
					
						
							|  |  |  | 	 * Only contains (DRWPass *) */ | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	struct DRWPass *image_faces; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct DRWPass *wire_overlay; | 
					
						
							|  |  |  | 	struct DRWPass *face_overlay; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | } PAINT_TEXTURE_PassList; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct PAINT_TEXTURE_FramebufferList { | 
					
						
							|  |  |  | 	/* Contains all framebuffer objects needed by this engine.
 | 
					
						
							|  |  |  | 	 * Only contains (GPUFrameBuffer *) */ | 
					
						
							| 
									
										
										
										
											2017-03-13 11:39:41 +01:00
										 |  |  | 	struct GPUFrameBuffer *fb; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | } PAINT_TEXTURE_FramebufferList; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct PAINT_TEXTURE_TextureList { | 
					
						
							|  |  |  | 	/* Contains all framebuffer textures / utility textures
 | 
					
						
							|  |  |  | 	 * needed by this engine. Only viewport specific textures | 
					
						
							|  |  |  | 	 * (not per object). Only contains (GPUTexture *) */ | 
					
						
							| 
									
										
										
										
											2017-03-13 11:39:41 +01:00
										 |  |  | 	struct GPUTexture *texture; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | } PAINT_TEXTURE_TextureList; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct PAINT_TEXTURE_StorageList { | 
					
						
							|  |  |  | 	/* Contains any other memory block that the engine needs.
 | 
					
						
							|  |  |  | 	 * Only directly MEM_(m/c)allocN'ed blocks because they are | 
					
						
							|  |  |  | 	 * free with MEM_freeN() when viewport is freed. | 
					
						
							|  |  |  | 	 * (not per object) */ | 
					
						
							| 
									
										
										
										
											2017-03-13 11:39:41 +01:00
										 |  |  | 	struct CustomStruct *block; | 
					
						
							| 
									
										
										
										
											2017-04-29 16:52:12 +10:00
										 |  |  | 	struct PAINT_TEXTURE_PrivateData *g_data; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | } PAINT_TEXTURE_StorageList; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct PAINT_TEXTURE_Data { | 
					
						
							| 
									
										
										
										
											2017-11-29 10:48:42 +01:00
										 |  |  | 	/* Struct returned by DRW_viewport_engine_data_ensure.
 | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	 * If you don't use one of these, just make it a (void *) */ | 
					
						
							|  |  |  | 	// void *fbl;
 | 
					
						
							| 
									
										
										
										
											2017-04-03 19:32:05 +02:00
										 |  |  | 	void *engine_type; /* Required */ | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	PAINT_TEXTURE_FramebufferList *fbl; | 
					
						
							|  |  |  | 	PAINT_TEXTURE_TextureList *txl; | 
					
						
							|  |  |  | 	PAINT_TEXTURE_PassList *psl; | 
					
						
							|  |  |  | 	PAINT_TEXTURE_StorageList *stl; | 
					
						
							|  |  |  | } PAINT_TEXTURE_Data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* *********** STATIC *********** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-18 22:14:53 +01:00
										 |  |  | static struct { | 
					
						
							|  |  |  | 	/* Custom shaders :
 | 
					
						
							|  |  |  | 	 * Add sources to source/blender/draw/modes/shaders | 
					
						
							|  |  |  | 	 * init in PAINT_TEXTURE_engine_init(); | 
					
						
							|  |  |  | 	 * free in PAINT_TEXTURE_engine_free(); */ | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	struct GPUShader *fallback_sh; | 
					
						
							|  |  |  | 	struct GPUShader *image_sh; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct GPUShader *wire_overlay_shader; | 
					
						
							|  |  |  | 	struct GPUShader *face_overlay_shader; | 
					
						
							| 
									
										
										
										
											2017-03-18 22:14:53 +01:00
										 |  |  | } e_data = {NULL}; /* Engine data */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-29 16:52:12 +10:00
										 |  |  | typedef struct PAINT_TEXTURE_PrivateData { | 
					
						
							| 
									
										
										
										
											2017-03-18 22:14:53 +01:00
										 |  |  | 	/* This keeps the references of the shading groups for
 | 
					
						
							|  |  |  | 	 * easy access in PAINT_TEXTURE_cache_populate() */ | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	DRWShadingGroup *shgroup_fallback; | 
					
						
							|  |  |  | 	DRWShadingGroup **shgroup_image_array; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* face-mask  */ | 
					
						
							|  |  |  | 	DRWShadingGroup *lwire_shgrp; | 
					
						
							|  |  |  | 	DRWShadingGroup *face_shgrp; | 
					
						
							| 
									
										
										
										
											2017-04-29 16:52:12 +10:00
										 |  |  | } PAINT_TEXTURE_PrivateData; /* Transient data */ | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* *********** FUNCTIONS *********** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Init Textures, Framebuffers, Storage and Shaders.
 | 
					
						
							|  |  |  |  * It is called for every frames. | 
					
						
							| 
									
										
										
										
											2017-03-14 18:40:23 +11:00
										 |  |  |  * (Optional) */ | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | static void PAINT_TEXTURE_engine_init(void *vedata) | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-03-26 20:13:34 +02:00
										 |  |  | 	PAINT_TEXTURE_TextureList *txl = ((PAINT_TEXTURE_Data *)vedata)->txl; | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | 	PAINT_TEXTURE_FramebufferList *fbl = ((PAINT_TEXTURE_Data *)vedata)->fbl; | 
					
						
							| 
									
										
										
										
											2017-03-26 20:13:34 +02:00
										 |  |  | 	PAINT_TEXTURE_StorageList *stl = ((PAINT_TEXTURE_Data *)vedata)->stl; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-14 18:40:23 +11:00
										 |  |  | 	UNUSED_VARS(txl, fbl, stl); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	/* Init Framebuffers like this: order is attachment order (for color texs) */ | 
					
						
							|  |  |  | 	/*
 | 
					
						
							| 
									
										
										
										
											2018-04-30 16:02:24 +02:00
										 |  |  | 	 * DRWFboTexture tex[2] = {{&txl->depth, GPU_DEPTH_COMPONENT24, 0}, | 
					
						
							|  |  |  | 	 *                         {&txl->color, GPU_RGBA8, DRW_TEX_FILTER}}; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* DRW_framebuffer_init takes care of checking if
 | 
					
						
							|  |  |  | 	 * the framebuffer is valid and has the right size*/ | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * float *viewport_size = DRW_viewport_size_get(); | 
					
						
							|  |  |  | 	 * DRW_framebuffer_init(&fbl->occlude_wire_fb, | 
					
						
							|  |  |  | 	 *                     (int)viewport_size[0], (int)viewport_size[1], | 
					
						
							|  |  |  | 	 *                     tex, 2); | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	if (!e_data.fallback_sh) { | 
					
						
							|  |  |  | 		e_data.fallback_sh = GPU_shader_get_builtin_shader(GPU_SHADER_3D_UNIFORM_COLOR); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	if (!e_data.image_sh) { | 
					
						
							|  |  |  | 		e_data.image_sh = GPU_shader_get_builtin_shader(GPU_SHADER_3D_UNIFORM_COLOR); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 		e_data.image_sh = DRW_shader_create_with_lib( | 
					
						
							|  |  |  | 		        datatoc_paint_texture_vert_glsl, NULL, | 
					
						
							| 
									
										
										
										
											2018-01-17 20:35:06 +11:00
										 |  |  | 		        datatoc_paint_texture_frag_glsl, | 
					
						
							|  |  |  | 		        datatoc_common_globals_lib_glsl, NULL); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!e_data.wire_overlay_shader) { | 
					
						
							|  |  |  | 		e_data.wire_overlay_shader = DRW_shader_create_with_lib( | 
					
						
							|  |  |  | 		        datatoc_paint_wire_vert_glsl, NULL, | 
					
						
							|  |  |  | 		        datatoc_paint_wire_frag_glsl, | 
					
						
							| 
									
										
										
										
											2018-01-17 20:35:06 +11:00
										 |  |  | 		        datatoc_common_globals_lib_glsl, | 
					
						
							|  |  |  | 		        "#define VERTEX_MODE\n"); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!e_data.face_overlay_shader) { | 
					
						
							|  |  |  | 		e_data.face_overlay_shader = GPU_shader_get_builtin_shader(GPU_SHADER_3D_UNIFORM_COLOR); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-07-13 00:27:06 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | /* Here init all passes and shading groups
 | 
					
						
							|  |  |  |  * Assume that all Passes are NULL */ | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | static void PAINT_TEXTURE_cache_init(void *vedata) | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | 	PAINT_TEXTURE_PassList *psl = ((PAINT_TEXTURE_Data *)vedata)->psl; | 
					
						
							|  |  |  | 	PAINT_TEXTURE_StorageList *stl = ((PAINT_TEXTURE_Data *)vedata)->stl; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-26 20:13:34 +02:00
										 |  |  | 	if (!stl->g_data) { | 
					
						
							|  |  |  | 		/* Alloc transient pointers */ | 
					
						
							| 
									
										
										
										
											2017-04-29 16:52:12 +10:00
										 |  |  | 		stl->g_data = MEM_mallocN(sizeof(*stl->g_data), __func__); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 		stl->g_data->shgroup_image_array = NULL; | 
					
						
							| 
									
										
										
										
											2017-03-26 20:13:34 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-14 18:40:23 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		/* Create a pass */ | 
					
						
							| 
									
										
										
										
											2018-06-22 10:37:38 +02:00
										 |  |  | 		DRWState state = DRW_STATE_WRITE_COLOR | DRW_STATE_DEPTH_EQUAL | DRW_STATE_BLEND; | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 		psl->image_faces = DRW_pass_create("Image Color Pass", state); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 		stl->g_data->shgroup_fallback = DRW_shgroup_create(e_data.fallback_sh, psl->image_faces); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* Uniforms need a pointer to it's value so be sure it's accessible at
 | 
					
						
							|  |  |  | 		 * any given time (i.e. use static vars) */ | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 		static float color[4] = {1.0f, 0.0f, 1.0f, 1.0}; | 
					
						
							|  |  |  | 		DRW_shgroup_uniform_vec4(stl->g_data->shgroup_fallback, "color", color, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		MEM_SAFE_FREE(stl->g_data->shgroup_image_array); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const DRWContextState *draw_ctx = DRW_context_state_get(); | 
					
						
							|  |  |  | 		Object *ob = draw_ctx->obact; | 
					
						
							|  |  |  | 		if (ob && ob->type == OB_MESH) { | 
					
						
							|  |  |  | 			Scene *scene = draw_ctx->scene; | 
					
						
							| 
									
										
										
										
											2017-07-14 16:45:47 +10:00
										 |  |  | 			const bool use_material_slots = (scene->toolsettings->imapaint.mode == IMAGEPAINT_MODE_MATERIAL); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 			const Mesh *me = ob->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			stl->g_data->shgroup_image_array = MEM_mallocN( | 
					
						
							|  |  |  | 			        sizeof(*stl->g_data->shgroup_image_array) * (use_material_slots ? me->totcol : 1), __func__); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (use_material_slots) { | 
					
						
							|  |  |  | 				for (int i = 0; i < me->totcol; i++) { | 
					
						
							|  |  |  | 					Material *ma = give_current_material(ob, i + 1); | 
					
						
							|  |  |  | 					Image *ima = (ma && ma->texpaintslot) ? ma->texpaintslot[ma->paint_active_slot].ima : NULL; | 
					
						
							|  |  |  | 					GPUTexture *tex = ima ? | 
					
						
							| 
									
										
										
										
											2018-06-14 12:58:30 +02:00
										 |  |  | 					        GPU_texture_from_blender(ima, NULL, GL_TEXTURE_2D, false, 0.0f) : NULL; | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					if (tex) { | 
					
						
							|  |  |  | 						DRWShadingGroup *grp = DRW_shgroup_create(e_data.image_sh, psl->image_faces); | 
					
						
							|  |  |  | 						DRW_shgroup_uniform_texture(grp, "image", tex); | 
					
						
							| 
									
										
										
										
											2018-06-22 10:37:38 +02:00
										 |  |  | 						DRW_shgroup_uniform_float(grp, "alpha", &draw_ctx->v3d->overlay.texture_paint_mode_opacity, 1); | 
					
						
							| 
									
										
										
										
											2018-08-08 15:08:50 +10:00
										 |  |  | 						DRW_shgroup_uniform_block(grp, "globalsBlock", globals_ubo); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 						stl->g_data->shgroup_image_array[i] = grp; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							|  |  |  | 						stl->g_data->shgroup_image_array[i] = NULL; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				Image *ima = scene->toolsettings->imapaint.canvas; | 
					
						
							|  |  |  | 				GPUTexture *tex = ima ? | 
					
						
							| 
									
										
										
										
											2018-06-14 12:58:30 +02:00
										 |  |  | 				        GPU_texture_from_blender(ima, NULL, GL_TEXTURE_2D, false, 0.0f) : NULL; | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				if (tex) { | 
					
						
							|  |  |  | 					DRWShadingGroup *grp = DRW_shgroup_create(e_data.image_sh, psl->image_faces); | 
					
						
							|  |  |  | 					DRW_shgroup_uniform_texture(grp, "image", tex); | 
					
						
							| 
									
										
										
										
											2018-06-22 10:37:38 +02:00
										 |  |  | 					DRW_shgroup_uniform_float(grp, "alpha", &draw_ctx->v3d->overlay.texture_paint_mode_opacity, 1); | 
					
						
							| 
									
										
										
										
											2018-10-02 14:35:33 +02:00
										 |  |  | 					DRW_shgroup_uniform_block(grp, "globalsBlock", globals_ubo); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 					stl->g_data->shgroup_image_array[0] = grp; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					stl->g_data->shgroup_image_array[0] = NULL; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	/* Face Mask */ | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		psl->wire_overlay = DRW_pass_create( | 
					
						
							|  |  |  | 		        "Wire Pass", | 
					
						
							| 
									
										
										
										
											2018-05-20 19:05:13 +02:00
										 |  |  | 		        DRW_STATE_WRITE_COLOR | DRW_STATE_WRITE_DEPTH | DRW_STATE_DEPTH_LESS_EQUAL); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		stl->g_data->lwire_shgrp = DRW_shgroup_create(e_data.wire_overlay_shader, psl->wire_overlay); | 
					
						
							| 
									
										
										
										
											2018-08-08 15:08:50 +10:00
										 |  |  | 		DRW_shgroup_uniform_block(stl->g_data->lwire_shgrp, "globalsBlock", globals_ubo); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		psl->face_overlay = DRW_pass_create( | 
					
						
							|  |  |  | 		        "Face Mask Pass", | 
					
						
							| 
									
										
										
										
											2018-05-20 19:05:13 +02:00
										 |  |  | 		        DRW_STATE_WRITE_COLOR | DRW_STATE_WRITE_DEPTH | DRW_STATE_DEPTH_LESS_EQUAL | DRW_STATE_BLEND); | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		stl->g_data->face_shgrp = DRW_shgroup_create(e_data.face_overlay_shader, psl->face_overlay); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		static float col[4] = {1.0f, 1.0f, 1.0f, 0.2f}; | 
					
						
							|  |  |  | 		DRW_shgroup_uniform_vec4(stl->g_data->face_shgrp, "color", col, 1); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Add geometry to shadingGroups. Execute for each objects */ | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | static void PAINT_TEXTURE_cache_populate(void *vedata, Object *ob) | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | 	PAINT_TEXTURE_PassList *psl = ((PAINT_TEXTURE_Data *)vedata)->psl; | 
					
						
							|  |  |  | 	PAINT_TEXTURE_StorageList *stl = ((PAINT_TEXTURE_Data *)vedata)->stl; | 
					
						
							| 
									
										
										
										
											2017-05-29 20:15:32 +10:00
										 |  |  | 	const DRWContextState *draw_ctx = DRW_context_state_get(); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-14 18:40:23 +11:00
										 |  |  | 	UNUSED_VARS(psl, stl); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-29 20:15:32 +10:00
										 |  |  | 	if ((ob->type == OB_MESH) && (draw_ctx->obact == ob)) { | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 		/* Get geometry cache */ | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 		const Mesh *me = ob->data; | 
					
						
							|  |  |  | 		Scene *scene = draw_ctx->scene; | 
					
						
							| 
									
										
										
										
											2018-06-22 10:37:38 +02:00
										 |  |  | 		const bool use_surface = draw_ctx->v3d->overlay.texture_paint_mode_opacity != 0.0; //DRW_object_is_mode_shade(ob) == true;
 | 
					
						
							| 
									
										
										
										
											2017-07-14 16:45:47 +10:00
										 |  |  | 		const bool use_material_slots = (scene->toolsettings->imapaint.mode == IMAGEPAINT_MODE_MATERIAL); | 
					
						
							| 
									
										
										
										
											2017-05-29 20:15:32 +10:00
										 |  |  | 		bool ok = false; | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 16:45:47 +10:00
										 |  |  | 		if (use_surface) { | 
					
						
							|  |  |  | 			if (me->mloopuv != NULL) { | 
					
						
							|  |  |  | 				if (use_material_slots) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 					struct GPUBatch **geom_array = me->totcol ? DRW_cache_mesh_surface_texpaint_get(ob) : NULL; | 
					
						
							| 
									
										
										
										
											2017-07-14 16:45:47 +10:00
										 |  |  | 					if ((me->totcol == 0) || (geom_array == NULL)) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 						struct GPUBatch *geom = DRW_cache_mesh_surface_get(ob); | 
					
						
							| 
									
										
										
										
											2017-07-14 16:45:47 +10:00
										 |  |  | 						DRW_shgroup_call_add(stl->g_data->shgroup_fallback, geom, ob->obmat); | 
					
						
							|  |  |  | 						ok = true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					else { | 
					
						
							|  |  |  | 						for (int i = 0; i < me->totcol; i++) { | 
					
						
							|  |  |  | 							if (stl->g_data->shgroup_image_array[i]) { | 
					
						
							|  |  |  | 								DRW_shgroup_call_add(stl->g_data->shgroup_image_array[i], geom_array[i], ob->obmat); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							else { | 
					
						
							|  |  |  | 								DRW_shgroup_call_add(stl->g_data->shgroup_fallback, geom_array[i], ob->obmat); | 
					
						
							|  |  |  | 							} | 
					
						
							|  |  |  | 							ok = true; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2017-05-29 20:15:32 +10:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 					struct GPUBatch *geom = DRW_cache_mesh_surface_texpaint_single_get(ob); | 
					
						
							| 
									
										
										
										
											2017-07-14 16:45:47 +10:00
										 |  |  | 					if (geom && stl->g_data->shgroup_image_array[0]) { | 
					
						
							|  |  |  | 						DRW_shgroup_call_add(stl->g_data->shgroup_image_array[0], geom, ob->obmat); | 
					
						
							| 
									
										
										
										
											2017-05-29 20:15:32 +10:00
										 |  |  | 						ok = true; | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-14 16:45:47 +10:00
										 |  |  | 			if (!ok) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 				struct GPUBatch *geom = DRW_cache_mesh_surface_get(ob); | 
					
						
							| 
									
										
										
										
											2017-07-14 16:45:47 +10:00
										 |  |  | 				DRW_shgroup_call_add(stl->g_data->shgroup_fallback, geom, ob->obmat); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-05-29 20:15:32 +10:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 		/* Face Mask */ | 
					
						
							|  |  |  | 		const bool use_face_sel = (me->editflag & ME_EDIT_PAINT_FACE_SEL) != 0; | 
					
						
							|  |  |  | 		if (use_face_sel) { | 
					
						
							| 
									
										
										
										
											2018-07-18 00:12:21 +02:00
										 |  |  | 			struct GPUBatch *geom; | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 			/* Note: ideally selected faces wouldn't show interior wire. */ | 
					
						
							|  |  |  | 			const bool use_wire = true; | 
					
						
							|  |  |  | 			geom = DRW_cache_mesh_edges_paint_overlay_get(ob, use_wire, use_face_sel); | 
					
						
							|  |  |  | 			DRW_shgroup_call_add(stl->g_data->lwire_shgrp, geom, ob->obmat); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			geom = DRW_cache_mesh_faces_weight_overlay_get(ob); | 
					
						
							|  |  |  | 			DRW_shgroup_call_add(stl->g_data->face_shgrp, geom, ob->obmat); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-14 18:40:23 +11:00
										 |  |  | /* Optional: Post-cache_populate callback */ | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | static void PAINT_TEXTURE_cache_finish(void *vedata) | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | 	PAINT_TEXTURE_PassList *psl = ((PAINT_TEXTURE_Data *)vedata)->psl; | 
					
						
							|  |  |  | 	PAINT_TEXTURE_StorageList *stl = ((PAINT_TEXTURE_Data *)vedata)->stl; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-19 18:19:49 +02:00
										 |  |  | 	/* Do something here! dependent on the objects gathered */ | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	UNUSED_VARS(psl); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MEM_SAFE_FREE(stl->g_data->shgroup_image_array); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Draw time ! Control rendering pipeline from here */ | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | static void PAINT_TEXTURE_draw_scene(void *vedata) | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-03-26 19:10:53 +02:00
										 |  |  | 	PAINT_TEXTURE_PassList *psl = ((PAINT_TEXTURE_Data *)vedata)->psl; | 
					
						
							|  |  |  | 	PAINT_TEXTURE_FramebufferList *fbl = ((PAINT_TEXTURE_Data *)vedata)->fbl; | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Default framebuffer and texture */ | 
					
						
							|  |  |  | 	DefaultFramebufferList *dfbl = DRW_viewport_framebuffer_list_get(); | 
					
						
							|  |  |  | 	DefaultTextureList *dtxl = DRW_viewport_texture_list_get(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-14 18:40:23 +11:00
										 |  |  | 	UNUSED_VARS(fbl, dfbl, dtxl); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	DRW_draw_pass(psl->image_faces); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 23:31:46 +10:00
										 |  |  | 	DRW_draw_pass(psl->face_overlay); | 
					
						
							|  |  |  | 	DRW_draw_pass(psl->wire_overlay); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Cleanup when destroying the engine.
 | 
					
						
							|  |  |  |  * This is not per viewport ! only when quitting blender. | 
					
						
							|  |  |  |  * Mostly used for freeing shaders */ | 
					
						
							|  |  |  | static void PAINT_TEXTURE_engine_free(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-05-29 20:48:02 +10:00
										 |  |  | 	DRW_SHADER_FREE_SAFE(e_data.image_sh); | 
					
						
							|  |  |  | 	DRW_SHADER_FREE_SAFE(e_data.wire_overlay_shader); | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 19:49:19 +10:00
										 |  |  | static const DrawEngineDataSize PAINT_TEXTURE_data_size = DRW_VIEWPORT_DATA_SIZE(PAINT_TEXTURE_Data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | DrawEngineType draw_engine_paint_texture_type = { | 
					
						
							|  |  |  | 	NULL, NULL, | 
					
						
							|  |  |  | 	N_("PaintTextureMode"), | 
					
						
							| 
									
										
										
										
											2017-04-12 19:49:19 +10:00
										 |  |  | 	&PAINT_TEXTURE_data_size, | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | 	&PAINT_TEXTURE_engine_init, | 
					
						
							|  |  |  | 	&PAINT_TEXTURE_engine_free, | 
					
						
							|  |  |  | 	&PAINT_TEXTURE_cache_init, | 
					
						
							|  |  |  | 	&PAINT_TEXTURE_cache_populate, | 
					
						
							|  |  |  | 	&PAINT_TEXTURE_cache_finish, | 
					
						
							|  |  |  | 	NULL, /* draw_background but not needed by mode engines */ | 
					
						
							| 
									
										
										
										
											2017-09-25 20:07:02 +02:00
										 |  |  | 	&PAINT_TEXTURE_draw_scene, | 
					
						
							|  |  |  | 	NULL, | 
					
						
							| 
									
										
										
										
											2018-01-29 14:56:16 +01:00
										 |  |  | 	NULL, | 
					
						
							| 
									
										
										
										
											2017-03-12 21:16:03 +01:00
										 |  |  | }; |