2009-08-15 19:48:50 +00:00
|
|
|
/*
|
|
|
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
|
|
|
*
|
|
|
|
* 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,
|
2010-02-12 13:34:04 +00:00
|
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2009-08-15 19:48:50 +00:00
|
|
|
*
|
|
|
|
* The Original Code is Copyright (C) 2009 by Nicholas Bishop
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* The Original Code is: all of this file.
|
|
|
|
*
|
|
|
|
* Contributor(s): none yet.
|
|
|
|
*
|
|
|
|
* ***** END GPL LICENSE BLOCK *****
|
2011-02-23 18:03:40 +00:00
|
|
|
*/
|
2009-08-15 19:48:50 +00:00
|
|
|
|
2011-02-27 20:40:57 +00:00
|
|
|
/** \file blender/blenkernel/intern/paint.c
|
|
|
|
* \ingroup bke
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2009-08-16 19:50:00 +00:00
|
|
|
|
2009-08-15 19:48:50 +00:00
|
|
|
#include "DNA_object_types.h"
|
2009-10-20 19:27:27 +00:00
|
|
|
#include "DNA_mesh_types.h"
|
2012-03-14 06:31:38 +00:00
|
|
|
#include "DNA_meshdata_types.h"
|
2009-08-16 19:50:00 +00:00
|
|
|
#include "DNA_scene_types.h"
|
2010-08-04 04:01:27 +00:00
|
|
|
#include "DNA_brush_types.h"
|
2009-08-15 19:48:50 +00:00
|
|
|
|
2012-03-14 06:31:38 +00:00
|
|
|
#include "BLI_bitmap.h"
|
2011-01-07 18:36:47 +00:00
|
|
|
#include "BLI_utildefines.h"
|
|
|
|
|
2009-08-16 19:50:00 +00:00
|
|
|
#include "BKE_brush.h"
|
2012-06-04 07:29:45 +00:00
|
|
|
#include "BKE_context.h"
|
2009-08-17 15:05:18 +00:00
|
|
|
#include "BKE_library.h"
|
2009-08-15 19:48:50 +00:00
|
|
|
#include "BKE_paint.h"
|
2012-03-14 06:31:38 +00:00
|
|
|
#include "BKE_subsurf.h"
|
2009-08-15 19:48:50 +00:00
|
|
|
|
2009-08-16 19:50:00 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2009-08-20 16:00:17 +00:00
|
|
|
const char PAINT_CURSOR_SCULPT[3] = {255, 100, 100};
|
|
|
|
const char PAINT_CURSOR_VERTEX_PAINT[3] = {255, 255, 255};
|
|
|
|
const char PAINT_CURSOR_WEIGHT_PAINT[3] = {200, 200, 255};
|
|
|
|
const char PAINT_CURSOR_TEXTURE_PAINT[3] = {255, 255, 255};
|
|
|
|
|
2009-08-16 19:50:00 +00:00
|
|
|
Paint *paint_get_active(Scene *sce)
|
|
|
|
{
|
2012-03-24 06:18:31 +00:00
|
|
|
if (sce) {
|
2009-08-17 02:49:31 +00:00
|
|
|
ToolSettings *ts = sce->toolsettings;
|
2009-09-16 19:27:08 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (sce->basact && sce->basact->object) {
|
2012-04-28 06:31:57 +00:00
|
|
|
switch (sce->basact->object->mode) {
|
2012-05-11 08:05:47 +00:00
|
|
|
case OB_MODE_SCULPT:
|
|
|
|
return &ts->sculpt->paint;
|
|
|
|
case OB_MODE_VERTEX_PAINT:
|
|
|
|
return &ts->vpaint->paint;
|
|
|
|
case OB_MODE_WEIGHT_PAINT:
|
|
|
|
return &ts->wpaint->paint;
|
|
|
|
case OB_MODE_TEXTURE_PAINT:
|
2012-01-17 16:31:13 +00:00
|
|
|
return &ts->imapaint.paint;
|
2012-05-11 08:05:47 +00:00
|
|
|
case OB_MODE_EDIT:
|
|
|
|
if (ts->use_uv_sculpt)
|
|
|
|
return &ts->uvsculpt->paint;
|
|
|
|
else
|
|
|
|
return &ts->imapaint.paint;
|
2009-09-16 19:27:08 +00:00
|
|
|
}
|
2009-08-16 19:50:00 +00:00
|
|
|
}
|
2009-09-16 19:27:08 +00:00
|
|
|
|
|
|
|
/* default to image paint */
|
|
|
|
return &ts->imapaint.paint;
|
2009-08-16 19:50:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-06-04 07:29:45 +00:00
|
|
|
Paint *paint_get_active_from_context(const bContext *C)
|
|
|
|
{
|
|
|
|
Scene *sce = CTX_data_scene(C);
|
|
|
|
|
|
|
|
if (sce) {
|
|
|
|
ToolSettings *ts = sce->toolsettings;
|
|
|
|
Object *obact = NULL;
|
|
|
|
|
|
|
|
if (sce->basact && sce->basact->object)
|
|
|
|
obact = sce->basact->object;
|
|
|
|
|
|
|
|
if (CTX_wm_space_image(C) != NULL) {
|
2012-06-18 14:08:54 +00:00
|
|
|
if (obact && obact->mode == OB_MODE_EDIT) {
|
2012-06-04 07:29:45 +00:00
|
|
|
if (ts->use_uv_sculpt)
|
|
|
|
return &ts->uvsculpt->paint;
|
|
|
|
else
|
|
|
|
return &ts->imapaint.paint;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return &ts->imapaint.paint;
|
|
|
|
}
|
|
|
|
}
|
2012-06-04 07:44:57 +00:00
|
|
|
else if (obact) {
|
2012-06-04 07:29:45 +00:00
|
|
|
switch (obact->mode) {
|
|
|
|
case OB_MODE_SCULPT:
|
|
|
|
return &ts->sculpt->paint;
|
|
|
|
case OB_MODE_VERTEX_PAINT:
|
|
|
|
return &ts->vpaint->paint;
|
|
|
|
case OB_MODE_WEIGHT_PAINT:
|
|
|
|
return &ts->wpaint->paint;
|
|
|
|
case OB_MODE_TEXTURE_PAINT:
|
|
|
|
return &ts->imapaint.paint;
|
|
|
|
case OB_MODE_EDIT:
|
|
|
|
if (ts->use_uv_sculpt)
|
|
|
|
return &ts->uvsculpt->paint;
|
|
|
|
else
|
|
|
|
return &ts->imapaint.paint;
|
|
|
|
}
|
2012-06-04 07:44:57 +00:00
|
|
|
}
|
|
|
|
else {
|
2012-06-04 07:29:45 +00:00
|
|
|
/* default to image paint */
|
|
|
|
return &ts->imapaint.paint;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-08-16 19:50:00 +00:00
|
|
|
Brush *paint_brush(Paint *p)
|
|
|
|
{
|
2010-08-04 12:18:07 +00:00
|
|
|
return p ? p->brush : NULL;
|
2009-08-16 19:50:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void paint_brush_set(Paint *p, Brush *br)
|
|
|
|
{
|
2012-03-24 06:18:31 +00:00
|
|
|
if (p) {
|
2011-06-27 04:05:19 +00:00
|
|
|
id_us_min((ID *)p->brush);
|
|
|
|
id_us_plus((ID *)br);
|
2012-05-11 08:05:47 +00:00
|
|
|
p->brush = br;
|
2011-06-27 04:05:19 +00:00
|
|
|
}
|
2009-08-16 19:50:00 +00:00
|
|
|
}
|
|
|
|
|
2011-12-03 20:26:53 +00:00
|
|
|
/* are we in vertex paint or weight pain face select mode? */
|
2009-08-15 19:48:50 +00:00
|
|
|
int paint_facesel_test(Object *ob)
|
|
|
|
{
|
2011-12-03 20:26:53 +00:00
|
|
|
return ( (ob != NULL) &&
|
|
|
|
(ob->type == OB_MESH) &&
|
|
|
|
(ob->data != NULL) &&
|
|
|
|
(((Mesh *)ob->data)->editflag & ME_EDIT_PAINT_MASK) &&
|
2012-05-11 08:05:47 +00:00
|
|
|
(ob->mode & (OB_MODE_VERTEX_PAINT | OB_MODE_WEIGHT_PAINT | OB_MODE_TEXTURE_PAINT))
|
2011-12-03 20:26:53 +00:00
|
|
|
);
|
2009-08-15 19:48:50 +00:00
|
|
|
}
|
2011-09-18 17:10:28 +00:00
|
|
|
|
2011-12-03 20:26:53 +00:00
|
|
|
/* are we in weight paint vertex select mode? */
|
2011-07-12 19:06:06 +00:00
|
|
|
int paint_vertsel_test(Object *ob)
|
|
|
|
{
|
2011-12-03 20:26:53 +00:00
|
|
|
return ( (ob != NULL) &&
|
|
|
|
(ob->type == OB_MESH) &&
|
|
|
|
(ob->data != NULL) &&
|
|
|
|
(((Mesh *)ob->data)->editflag & ME_EDIT_VERT_SEL) &&
|
|
|
|
(ob->mode & OB_MODE_WEIGHT_PAINT)
|
|
|
|
);
|
2011-07-12 19:06:06 +00:00
|
|
|
}
|
2011-12-03 20:26:53 +00:00
|
|
|
|
2012-07-25 13:44:59 +00:00
|
|
|
void BKE_paint_init(Paint *p, const char col[3])
|
2009-08-17 02:49:31 +00:00
|
|
|
{
|
|
|
|
Brush *brush;
|
|
|
|
|
|
|
|
/* If there's no brush, create one */
|
|
|
|
brush = paint_brush(p);
|
2012-03-24 06:18:31 +00:00
|
|
|
if (brush == NULL)
|
2012-05-11 08:05:47 +00:00
|
|
|
brush = BKE_brush_add("Brush");
|
2009-08-17 02:49:31 +00:00
|
|
|
paint_brush_set(p, brush);
|
2009-08-20 05:13:07 +00:00
|
|
|
|
2009-08-20 16:00:17 +00:00
|
|
|
memcpy(p->paint_cursor_col, col, 3);
|
2009-08-20 05:13:07 +00:00
|
|
|
p->paint_cursor_col[3] = 128;
|
2009-11-03 22:50:09 +00:00
|
|
|
|
|
|
|
p->flags |= PAINT_SHOW_BRUSH;
|
2009-08-17 02:49:31 +00:00
|
|
|
}
|
|
|
|
|
2012-07-25 22:37:52 +00:00
|
|
|
void BKE_paint_free(Paint *paint)
|
2009-08-16 19:50:00 +00:00
|
|
|
{
|
2011-06-27 04:05:19 +00:00
|
|
|
id_us_min((ID *)paint->brush);
|
2009-08-16 19:50:00 +00:00
|
|
|
}
|
|
|
|
|
2011-06-27 04:05:19 +00:00
|
|
|
/* called when copying scene settings, so even if 'src' and 'tar' are the same
|
|
|
|
* still do a id_us_plus(), rather then if we were copying betweem 2 existing
|
|
|
|
* scenes where a matching value should decrease the existing user count as
|
|
|
|
* with paint_brush_set() */
|
2012-07-25 22:37:52 +00:00
|
|
|
void BKE_paint_copy(Paint *src, Paint *tar)
|
2009-08-16 19:50:00 +00:00
|
|
|
{
|
2012-05-11 08:05:47 +00:00
|
|
|
tar->brush = src->brush;
|
2011-06-27 04:05:19 +00:00
|
|
|
id_us_plus((ID *)tar->brush);
|
2010-07-26 02:35:43 +00:00
|
|
|
}
|
2012-03-14 06:31:38 +00:00
|
|
|
|
|
|
|
/* returns non-zero if any of the face's vertices
|
2012-04-22 11:54:53 +00:00
|
|
|
* are hidden, zero otherwise */
|
2012-03-14 06:31:38 +00:00
|
|
|
int paint_is_face_hidden(const MFace *f, const MVert *mvert)
|
|
|
|
{
|
|
|
|
return ((mvert[f->v1].flag & ME_HIDE) ||
|
2012-05-11 08:05:47 +00:00
|
|
|
(mvert[f->v2].flag & ME_HIDE) ||
|
|
|
|
(mvert[f->v3].flag & ME_HIDE) ||
|
|
|
|
(f->v4 && (mvert[f->v4].flag & ME_HIDE)));
|
2012-03-14 06:31:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* returns non-zero if any of the corners of the grid
|
2012-04-22 11:54:53 +00:00
|
|
|
* face whose inner corner is at (x,y) are hidden,
|
|
|
|
* zero otherwise */
|
2012-03-14 06:31:38 +00:00
|
|
|
int paint_is_grid_face_hidden(const unsigned int *grid_hidden,
|
2012-05-11 08:05:47 +00:00
|
|
|
int gridsize, int x, int y)
|
2012-03-14 06:31:38 +00:00
|
|
|
{
|
|
|
|
/* skip face if any of its corners are hidden */
|
|
|
|
return (BLI_BITMAP_GET(grid_hidden, y * gridsize + x) ||
|
2012-05-11 08:05:47 +00:00
|
|
|
BLI_BITMAP_GET(grid_hidden, y * gridsize + x + 1) ||
|
|
|
|
BLI_BITMAP_GET(grid_hidden, (y + 1) * gridsize + x + 1) ||
|
|
|
|
BLI_BITMAP_GET(grid_hidden, (y + 1) * gridsize + x));
|
2012-03-14 06:31:38 +00:00
|
|
|
}
|
2012-05-10 20:33:55 +00:00
|
|
|
|
|
|
|
float paint_grid_paint_mask(const GridPaintMask *gpm, unsigned level,
|
2012-05-11 08:05:47 +00:00
|
|
|
unsigned x, unsigned y)
|
2012-05-10 20:33:55 +00:00
|
|
|
{
|
|
|
|
int factor = ccg_factor(level, gpm->level);
|
|
|
|
int gridsize = ccg_gridsize(gpm->level);
|
|
|
|
|
2012-05-11 08:05:47 +00:00
|
|
|
return gpm->data[(y * factor) * gridsize + (x * factor)];
|
2012-05-10 20:33:55 +00:00
|
|
|
}
|