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,
|
|
|
|
|
};
|