2092 lines
57 KiB
C
2092 lines
57 KiB
C
/*
|
|
* ***** 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,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
* The Original Code is Copyright (C) 2008, Blender Foundation, Joshua Leung
|
|
* This is a new part of Blender
|
|
*
|
|
* Contributor(s): Joshua Leung, Antonio Vazquez
|
|
*
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
*
|
|
* Operators for dealing with GP datablocks and layers
|
|
*/
|
|
|
|
/** \file blender/editors/gpencil/gpencil_data.c
|
|
* \ingroup edgpencil
|
|
*/
|
|
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <stddef.h>
|
|
#include <math.h>
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "BLI_blenlib.h"
|
|
#include "BLI_utildefines.h"
|
|
#include "BLI_ghash.h"
|
|
#include "BLI_math.h"
|
|
#include "BLI_string_utils.h"
|
|
|
|
#include "BLT_translation.h"
|
|
|
|
#include "DNA_scene_types.h"
|
|
#include "DNA_screen_types.h"
|
|
#include "DNA_space_types.h"
|
|
#include "DNA_view3d_types.h"
|
|
#include "DNA_gpencil_types.h"
|
|
|
|
#include "BKE_colortools.h"
|
|
#include "BKE_context.h"
|
|
#include "BKE_gpencil.h"
|
|
#include "BKE_library.h"
|
|
#include "BKE_main.h"
|
|
#include "BKE_object.h"
|
|
#include "BKE_report.h"
|
|
#include "BKE_scene.h"
|
|
#include "BKE_screen.h"
|
|
|
|
#include "UI_interface.h"
|
|
#include "UI_resources.h"
|
|
|
|
#include "WM_api.h"
|
|
#include "WM_types.h"
|
|
|
|
#include "RNA_access.h"
|
|
#include "RNA_define.h"
|
|
#include "RNA_enum_types.h"
|
|
|
|
#include "ED_gpencil.h"
|
|
|
|
#include "gpencil_intern.h"
|
|
|
|
/* ************************************************ */
|
|
/* Datablock Operators */
|
|
|
|
/* ******************* Add New Data ************************ */
|
|
|
|
/* add new datablock - wrapper around API */
|
|
static int gp_data_add_exec(bContext *C, wmOperator *op)
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
bGPdata **gpd_ptr = ED_gpencil_data_get_pointers(C, NULL);
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
|
|
if (gpd_ptr == NULL) {
|
|
BKE_report(op->reports, RPT_ERROR, "Nowhere for grease pencil data to go");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
else {
|
|
/* decrement user count and add new datablock */
|
|
bGPdata *gpd = (*gpd_ptr);
|
|
|
|
id_us_min(&gpd->id);
|
|
*gpd_ptr = BKE_gpencil_data_addnew(bmain, DATA_("GPencil"));
|
|
|
|
/* if not exist brushes, create a new set */
|
|
if (ts) {
|
|
if (BLI_listbase_is_empty(&ts->gp_brushes)) {
|
|
/* create new brushes */
|
|
BKE_gpencil_brush_init_presets(ts);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_data_add(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Grease Pencil Add New";
|
|
ot->idname = "GPENCIL_OT_data_add";
|
|
ot->description = "Add new Grease Pencil data-block";
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_data_add_exec;
|
|
ot->poll = gp_add_poll;
|
|
}
|
|
|
|
/* ******************* Unlink Data ************************ */
|
|
|
|
/* poll callback for adding data/layers - special */
|
|
static bool gp_data_unlink_poll(bContext *C)
|
|
{
|
|
bGPdata **gpd_ptr = ED_gpencil_data_get_pointers(C, NULL);
|
|
|
|
/* if we have access to some active data, make sure there's a datablock before enabling this */
|
|
return (gpd_ptr && *gpd_ptr);
|
|
}
|
|
|
|
|
|
/* unlink datablock - wrapper around API */
|
|
static int gp_data_unlink_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata **gpd_ptr = ED_gpencil_data_get_pointers(C, NULL);
|
|
|
|
if (gpd_ptr == NULL) {
|
|
BKE_report(op->reports, RPT_ERROR, "Nowhere for grease pencil data to go");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
else {
|
|
/* just unlink datablock now, decreasing its user count */
|
|
bGPdata *gpd = (*gpd_ptr);
|
|
|
|
id_us_min(&gpd->id);
|
|
*gpd_ptr = NULL;
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_data_unlink(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Grease Pencil Unlink";
|
|
ot->idname = "GPENCIL_OT_data_unlink";
|
|
ot->description = "Unlink active Grease Pencil data-block";
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_data_unlink_exec;
|
|
ot->poll = gp_data_unlink_poll;
|
|
}
|
|
|
|
|
|
/* ************************************************ */
|
|
/* Layer Operators */
|
|
|
|
/* ******************* Add New Layer ************************ */
|
|
|
|
/* add new layer - wrapper around API */
|
|
static int gp_layer_add_exec(bContext *C, wmOperator *op)
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
bGPdata **gpd_ptr = ED_gpencil_data_get_pointers(C, NULL);
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
|
|
/* if there's no existing Grease-Pencil data there, add some */
|
|
if (gpd_ptr == NULL) {
|
|
BKE_report(op->reports, RPT_ERROR, "Nowhere for grease pencil data to go");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
if (*gpd_ptr == NULL)
|
|
*gpd_ptr = BKE_gpencil_data_addnew(bmain, DATA_("GPencil"));
|
|
|
|
/* if not exist brushes, create a new set */
|
|
if (ts) {
|
|
if (BLI_listbase_is_empty(&ts->gp_brushes)) {
|
|
/* create new brushes */
|
|
BKE_gpencil_brush_init_presets(ts);
|
|
}
|
|
}
|
|
|
|
/* add new layer now */
|
|
BKE_gpencil_layer_addnew(*gpd_ptr, DATA_("GP_Layer"), true);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_layer_add(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Add New Layer";
|
|
ot->idname = "GPENCIL_OT_layer_add";
|
|
ot->description = "Add new Grease Pencil layer for the active Grease Pencil data-block";
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_layer_add_exec;
|
|
ot->poll = gp_add_poll;
|
|
}
|
|
|
|
/* ******************* Remove Active Layer ************************* */
|
|
|
|
static int gp_layer_remove_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *gpl = BKE_gpencil_layer_getactive(gpd);
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, gpl))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
if (gpl->flag & GP_LAYER_LOCKED) {
|
|
BKE_report(op->reports, RPT_ERROR, "Cannot delete locked layers");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
/* make the layer before this the new active layer
|
|
* - use the one after if this is the first
|
|
* - if this is the only layer, this naturally becomes NULL
|
|
*/
|
|
if (gpl->prev)
|
|
BKE_gpencil_layer_setactive(gpd, gpl->prev);
|
|
else
|
|
BKE_gpencil_layer_setactive(gpd, gpl->next);
|
|
|
|
/* delete the layer now... */
|
|
BKE_gpencil_layer_delete(gpd, gpl);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_layer_remove(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Remove Layer";
|
|
ot->idname = "GPENCIL_OT_layer_remove";
|
|
ot->description = "Remove active Grease Pencil layer";
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_layer_remove_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
}
|
|
|
|
/* ******************* Move Layer Up/Down ************************** */
|
|
|
|
enum {
|
|
GP_LAYER_MOVE_UP = -1,
|
|
GP_LAYER_MOVE_DOWN = 1
|
|
};
|
|
|
|
static int gp_layer_move_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *gpl = BKE_gpencil_layer_getactive(gpd);
|
|
|
|
int direction = RNA_enum_get(op->ptr, "type");
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, gpl))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
BLI_assert(ELEM(direction, -1, 0, 1)); /* we use value below */
|
|
if (BLI_listbase_link_move(&gpd->layers, gpl, direction)) {
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
}
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_layer_move(wmOperatorType *ot)
|
|
{
|
|
static const EnumPropertyItem slot_move[] = {
|
|
{GP_LAYER_MOVE_UP, "UP", 0, "Up", ""},
|
|
{GP_LAYER_MOVE_DOWN, "DOWN", 0, "Down", ""},
|
|
{0, NULL, 0, NULL, NULL}
|
|
};
|
|
|
|
/* identifiers */
|
|
ot->name = "Move Grease Pencil Layer";
|
|
ot->idname = "GPENCIL_OT_layer_move";
|
|
ot->description = "Move the active Grease Pencil layer up/down in the list";
|
|
|
|
/* api callbacks */
|
|
ot->exec = gp_layer_move_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
ot->prop = RNA_def_enum(ot->srna, "type", slot_move, 0, "Type", "");
|
|
}
|
|
|
|
/* ********************* Duplicate Layer ************************** */
|
|
|
|
static int gp_layer_copy_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *gpl = BKE_gpencil_layer_getactive(gpd);
|
|
bGPDlayer *new_layer;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, gpl))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* make copy of layer, and add it immediately after the existing layer */
|
|
new_layer = BKE_gpencil_layer_duplicate(gpl);
|
|
BLI_insertlinkafter(&gpd->layers, gpl, new_layer);
|
|
|
|
/* ensure new layer has a unique name, and is now the active layer */
|
|
BLI_uniquename(&gpd->layers, new_layer, DATA_("GP_Layer"), '.', offsetof(bGPDlayer, info), sizeof(new_layer->info));
|
|
BKE_gpencil_layer_setactive(gpd, new_layer);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_layer_duplicate(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Duplicate Layer";
|
|
ot->idname = "GPENCIL_OT_layer_duplicate";
|
|
ot->description = "Make a copy of the active Grease Pencil layer";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_layer_copy_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* *********************** Hide Layers ******************************** */
|
|
|
|
static int gp_hide_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *layer = BKE_gpencil_layer_getactive(gpd);
|
|
bool unselected = RNA_boolean_get(op->ptr, "unselected");
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, layer))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
if (unselected) {
|
|
bGPDlayer *gpl;
|
|
|
|
/* hide unselected */
|
|
for (gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
if (gpl != layer) {
|
|
gpl->flag |= GP_LAYER_HIDE;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
/* hide selected/active */
|
|
layer->flag |= GP_LAYER_HIDE;
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_hide(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Hide Layer(s)";
|
|
ot->idname = "GPENCIL_OT_hide";
|
|
ot->description = "Hide selected/unselected Grease Pencil layers";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_hide_exec;
|
|
ot->poll = gp_active_layer_poll; /* NOTE: we need an active layer to play with */
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* props */
|
|
RNA_def_boolean(ot->srna, "unselected", 0, "Unselected", "Hide unselected rather than selected layers");
|
|
}
|
|
|
|
/* ********************** Show All Layers ***************************** */
|
|
|
|
/* poll callback for showing layers */
|
|
static bool gp_reveal_poll(bContext *C)
|
|
{
|
|
return ED_gpencil_data_get_active(C) != NULL;
|
|
}
|
|
|
|
static void gp_reveal_select_frame(bContext *C, bGPDframe *frame, bool select)
|
|
{
|
|
bGPDstroke *gps;
|
|
for (gps = frame->strokes.first; gps; gps = gps->next) {
|
|
|
|
/* only deselect strokes that are valid in this view */
|
|
if (ED_gpencil_stroke_can_use(C, gps)) {
|
|
|
|
/* (de)select points */
|
|
int i;
|
|
bGPDspoint *pt;
|
|
for (i = 0, pt = gps->points; i < gps->totpoints; i++, pt++) {
|
|
SET_FLAG_FROM_TEST(pt->flag, select, GP_SPOINT_SELECT);
|
|
}
|
|
|
|
/* (de)select stroke */
|
|
SET_FLAG_FROM_TEST(gps->flag, select, GP_STROKE_SELECT);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int gp_reveal_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *gpl;
|
|
const bool select = RNA_boolean_get(op->ptr, "select");
|
|
|
|
/* sanity checks */
|
|
if (gpd == NULL)
|
|
return OPERATOR_CANCELLED;
|
|
|
|
for (gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
|
|
if (gpl->flag & GP_LAYER_HIDE) {
|
|
gpl->flag &= ~GP_LAYER_HIDE;
|
|
|
|
/* select or deselect if requested, only on hidden layers */
|
|
if (gpd->flag & GP_DATA_STROKE_EDITMODE) {
|
|
if (select) {
|
|
/* select all strokes on active frame only (same as select all operator) */
|
|
if (gpl->actframe) {
|
|
gp_reveal_select_frame(C, gpl->actframe, true);
|
|
}
|
|
}
|
|
else {
|
|
/* deselect strokes on all frames (same as deselect all operator) */
|
|
bGPDframe *gpf;
|
|
for (gpf = gpl->frames.first; gpf; gpf = gpf->next) {
|
|
gp_reveal_select_frame(C, gpf, false);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_reveal(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Show All Layers";
|
|
ot->idname = "GPENCIL_OT_reveal";
|
|
ot->description = "Show all Grease Pencil layers";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_reveal_exec;
|
|
ot->poll = gp_reveal_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* props */
|
|
RNA_def_boolean(ot->srna, "select", true, "Select", "");
|
|
}
|
|
|
|
/* ***************** Lock/Unlock All Layers ************************ */
|
|
|
|
static int gp_lock_all_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *gpl;
|
|
|
|
/* sanity checks */
|
|
if (gpd == NULL)
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* make all layers non-editable */
|
|
for (gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
gpl->flag |= GP_LAYER_LOCKED;
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_lock_all(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Lock All Layers";
|
|
ot->idname = "GPENCIL_OT_lock_all";
|
|
ot->description = "Lock all Grease Pencil layers to prevent them from being accidentally modified";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_lock_all_exec;
|
|
ot->poll = gp_reveal_poll; /* XXX: could use dedicated poll later */
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* -------------------------- */
|
|
|
|
static int gp_unlock_all_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *gpl;
|
|
|
|
/* sanity checks */
|
|
if (gpd == NULL)
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* make all layers editable again */
|
|
for (gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
gpl->flag &= ~GP_LAYER_LOCKED;
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_unlock_all(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Unlock All Layers";
|
|
ot->idname = "GPENCIL_OT_unlock_all";
|
|
ot->description = "Unlock all Grease Pencil layers so that they can be edited";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_unlock_all_exec;
|
|
ot->poll = gp_reveal_poll; /* XXX: could use dedicated poll later */
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* ********************** Isolate Layer **************************** */
|
|
|
|
static int gp_isolate_layer_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *layer = BKE_gpencil_layer_getactive(gpd);
|
|
bGPDlayer *gpl;
|
|
int flags = GP_LAYER_LOCKED;
|
|
bool isolate = false;
|
|
|
|
if (RNA_boolean_get(op->ptr, "affect_visibility"))
|
|
flags |= GP_LAYER_HIDE;
|
|
|
|
if (ELEM(NULL, gpd, layer)) {
|
|
BKE_report(op->reports, RPT_ERROR, "No active layer to isolate");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
/* Test whether to isolate or clear all flags */
|
|
for (gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
/* Skip if this is the active layer */
|
|
if (gpl == layer)
|
|
continue;
|
|
|
|
/* If the flags aren't set, that means that the layer is
|
|
* not alone, so we have some layers to isolate still
|
|
*/
|
|
if ((gpl->flag & flags) == 0) {
|
|
isolate = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Set/Clear flags as appropriate */
|
|
/* TODO: Include onionskinning on this list? */
|
|
if (isolate) {
|
|
/* Set flags on all "other" layers */
|
|
for (gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
if (gpl == layer)
|
|
continue;
|
|
else
|
|
gpl->flag |= flags;
|
|
}
|
|
}
|
|
else {
|
|
/* Clear flags - Restore everything else */
|
|
for (gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
gpl->flag &= ~flags;
|
|
}
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_layer_isolate(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Isolate Layer";
|
|
ot->idname = "GPENCIL_OT_layer_isolate";
|
|
ot->description = "Toggle whether the active layer is the only one that can be edited and/or visible";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_isolate_layer_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* properties */
|
|
RNA_def_boolean(ot->srna, "affect_visibility", false, "Affect Visibility",
|
|
"In addition to toggling the editability, also affect the visibility");
|
|
}
|
|
|
|
/* ********************** Merge Layer with the next layer **************************** */
|
|
|
|
static int gp_merge_layer_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *gpl_current = BKE_gpencil_layer_getactive(gpd);
|
|
bGPDlayer *gpl_next = gpl_current->next;
|
|
|
|
if (ELEM(NULL, gpd, gpl_current, gpl_next)) {
|
|
BKE_report(op->reports, RPT_ERROR, "No layers to merge");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
/* Collect frames of gpl_current in hash table to avoid O(n^2) lookups */
|
|
GHash *gh_frames_cur = BLI_ghash_int_new_ex(__func__, 64);
|
|
for (bGPDframe *gpf = gpl_current->frames.first; gpf; gpf = gpf->next) {
|
|
BLI_ghash_insert(gh_frames_cur, SET_INT_IN_POINTER(gpf->framenum), gpf);
|
|
}
|
|
|
|
/* read all frames from next layer */
|
|
for (bGPDframe *gpf = gpl_next->frames.first; gpf; gpf = gpf->next) {
|
|
/* try to find frame in active layer */
|
|
bGPDframe *frame = BLI_ghash_lookup(gh_frames_cur, SET_INT_IN_POINTER(gpf->framenum));
|
|
if (!frame) {
|
|
/* nothing found, create new */
|
|
frame = BKE_gpencil_frame_addnew(gpl_current, gpf->framenum);
|
|
}
|
|
/* add to tail all strokes */
|
|
BLI_movelisttolist(&frame->strokes, &gpf->strokes);
|
|
}
|
|
/* Now delete next layer */
|
|
BKE_gpencil_layer_delete(gpd, gpl_next);
|
|
BLI_ghash_free(gh_frames_cur, NULL, NULL);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_layer_merge(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Merge Down";
|
|
ot->idname = "GPENCIL_OT_layer_merge";
|
|
ot->description = "Merge the current layer with the layer below";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_merge_layer_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* ********************** Change Layer ***************************** */
|
|
|
|
static int gp_layer_change_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(evt))
|
|
{
|
|
uiPopupMenu *pup;
|
|
uiLayout *layout;
|
|
|
|
/* call the menu, which will call this operator again, hence the canceled */
|
|
pup = UI_popup_menu_begin(C, op->type->name, ICON_NONE);
|
|
layout = UI_popup_menu_layout(pup);
|
|
uiItemsEnumO(layout, "GPENCIL_OT_layer_change", "layer");
|
|
UI_popup_menu_end(C, pup);
|
|
|
|
return OPERATOR_INTERFACE;
|
|
}
|
|
|
|
static int gp_layer_change_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = CTX_data_gpencil_data(C);
|
|
bGPDlayer *gpl = NULL;
|
|
int layer_num = RNA_enum_get(op->ptr, "layer");
|
|
|
|
/* Get layer or create new one */
|
|
if (layer_num == -1) {
|
|
/* Create layer */
|
|
gpl = BKE_gpencil_layer_addnew(gpd, DATA_("GP_Layer"), true);
|
|
}
|
|
else {
|
|
/* Try to get layer */
|
|
gpl = BLI_findlink(&gpd->layers, layer_num);
|
|
|
|
if (gpl == NULL) {
|
|
BKE_reportf(op->reports, RPT_ERROR, "Cannot change to non-existent layer (index = %d)", layer_num);
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
}
|
|
|
|
/* Set active layer */
|
|
BKE_gpencil_layer_setactive(gpd, gpl);
|
|
|
|
/* updates */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_layer_change(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Change Layer";
|
|
ot->idname = "GPENCIL_OT_layer_change";
|
|
ot->description = "Change active Grease Pencil layer";
|
|
|
|
/* callbacks */
|
|
ot->invoke = gp_layer_change_invoke;
|
|
ot->exec = gp_layer_change_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* gp layer to use (dynamic enum) */
|
|
ot->prop = RNA_def_enum(ot->srna, "layer", DummyRNA_DEFAULT_items, 0, "Grease Pencil Layer", "");
|
|
RNA_def_enum_funcs(ot->prop, ED_gpencil_layers_with_new_enum_itemf);
|
|
}
|
|
|
|
/* ************************************************ */
|
|
|
|
/* ******************* Arrange Stroke Up/Down in drawing order ************************** */
|
|
|
|
enum {
|
|
GP_STROKE_MOVE_UP = -1,
|
|
GP_STROKE_MOVE_DOWN = 1,
|
|
GP_STROKE_MOVE_TOP = 2,
|
|
GP_STROKE_MOVE_BOTTOM = 3
|
|
};
|
|
|
|
static int gp_stroke_arrange_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDlayer *gpl = BKE_gpencil_layer_getactive(gpd);
|
|
bGPDstroke *gps;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, gpl, gpl->actframe)) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
bGPDframe *gpf = gpl->actframe;
|
|
/* temp listbase to store selected strokes */
|
|
ListBase selected = {NULL};
|
|
const int direction = RNA_enum_get(op->ptr, "direction");
|
|
|
|
/* verify if any selected stroke is in the extreme of the stack and select to move */
|
|
for (gps = gpf->strokes.first; gps; gps = gps->next) {
|
|
/* only if selected */
|
|
if (gps->flag & GP_STROKE_SELECT) {
|
|
/* skip strokes that are invalid for current view */
|
|
if (ED_gpencil_stroke_can_use(C, gps) == false) {
|
|
continue;
|
|
}
|
|
/* check if the color is editable */
|
|
if (ED_gpencil_stroke_color_use(gpl, gps) == false) {
|
|
continue;
|
|
}
|
|
/* some stroke is already at front*/
|
|
if ((direction == GP_STROKE_MOVE_TOP) || (direction == GP_STROKE_MOVE_UP)) {
|
|
if (gps == gpf->strokes.last) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
}
|
|
/* some stroke is already at botom */
|
|
if ((direction == GP_STROKE_MOVE_BOTTOM) || (direction == GP_STROKE_MOVE_DOWN)) {
|
|
if (gps == gpf->strokes.first) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
}
|
|
/* add to list */
|
|
BLI_addtail(&selected, BLI_genericNodeN(gps));
|
|
}
|
|
}
|
|
|
|
/* Now do the movement of the stroke */
|
|
switch (direction) {
|
|
/* Bring to Front */
|
|
case GP_STROKE_MOVE_TOP:
|
|
for (LinkData *link = selected.first; link; link = link->next) {
|
|
gps = link->data;
|
|
BLI_remlink(&gpf->strokes, gps);
|
|
BLI_addtail(&gpf->strokes, gps);
|
|
}
|
|
break;
|
|
/* Bring Forward */
|
|
case GP_STROKE_MOVE_UP:
|
|
for (LinkData *link = selected.last; link; link = link->prev) {
|
|
gps = link->data;
|
|
BLI_listbase_link_move(&gpf->strokes, gps, 1);
|
|
}
|
|
break;
|
|
/* Send Backward */
|
|
case GP_STROKE_MOVE_DOWN:
|
|
for (LinkData *link = selected.first; link; link = link->next) {
|
|
gps = link->data;
|
|
BLI_listbase_link_move(&gpf->strokes, gps, -1);
|
|
}
|
|
break;
|
|
/* Send to Back */
|
|
case GP_STROKE_MOVE_BOTTOM:
|
|
for (LinkData *link = selected.last; link; link = link->prev) {
|
|
gps = link->data;
|
|
BLI_remlink(&gpf->strokes, gps);
|
|
BLI_addhead(&gpf->strokes, gps);
|
|
}
|
|
break;
|
|
default:
|
|
BLI_assert(0);
|
|
break;
|
|
}
|
|
BLI_freelistN(&selected);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_stroke_arrange(wmOperatorType *ot)
|
|
{
|
|
static const EnumPropertyItem slot_move[] = {
|
|
{GP_STROKE_MOVE_UP, "UP", 0, "Bring Forward", ""},
|
|
{GP_STROKE_MOVE_DOWN, "DOWN", 0, "Send Backward", ""},
|
|
{GP_STROKE_MOVE_TOP, "TOP", 0, "Bring to Front", ""},
|
|
{GP_STROKE_MOVE_BOTTOM, "BOTTOM", 0, "Send to Back", ""},
|
|
{0, NULL, 0, NULL, NULL }
|
|
};
|
|
|
|
/* identifiers */
|
|
ot->name = "Arrange Stroke";
|
|
ot->idname = "GPENCIL_OT_stroke_arrange";
|
|
ot->description = "Arrange selected strokes up/down in the drawing order of the active layer";
|
|
|
|
/* api callbacks */
|
|
ot->exec = gp_stroke_arrange_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
ot->prop = RNA_def_enum(ot->srna, "direction", slot_move, GP_STROKE_MOVE_UP, "Direction", "");
|
|
}
|
|
/* ******************* Move Stroke to new color ************************** */
|
|
|
|
static int gp_stroke_change_color_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette;
|
|
bGPDpalettecolor *color;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd)) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
palette = BKE_gpencil_palette_getactive(gpd);
|
|
color = BKE_gpencil_palettecolor_getactive(palette);
|
|
if (ELEM(NULL, palette, color)) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
/* loop all strokes */
|
|
for (bGPDlayer *gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
/* only editable and visible layers are considered */
|
|
if (gpencil_layer_is_editable(gpl) && (gpl->actframe != NULL)) {
|
|
for (bGPDstroke *gps = gpl->actframe->strokes.last; gps; gps = gps->prev) {
|
|
/* only if selected */
|
|
if (gps->flag & GP_STROKE_SELECT) {
|
|
/* skip strokes that are invalid for current view */
|
|
if (ED_gpencil_stroke_can_use(C, gps) == false)
|
|
continue;
|
|
/* check if the color is editable */
|
|
if (ED_gpencil_stroke_color_use(gpl, gps) == false)
|
|
continue;
|
|
|
|
/* asign new color (only if different) */
|
|
if ((STREQ(gps->colorname, color->info) == false) || (gps->palcolor != color)) {
|
|
BLI_strncpy(gps->colorname, color->info, sizeof(gps->colorname));
|
|
gps->palcolor = color;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_stroke_change_color(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Change Stroke Color";
|
|
ot->idname = "GPENCIL_OT_stroke_change_color";
|
|
ot->description = "Move selected strokes to active color";
|
|
|
|
/* api callbacks */
|
|
ot->exec = gp_stroke_change_color_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
}
|
|
|
|
/* ******************* Lock color of non selected Strokes colors ************************** */
|
|
|
|
static int gp_stroke_lock_color_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
palette = BKE_gpencil_palette_getactive(gpd);
|
|
if (ELEM(NULL, palette))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* first lock all colors */
|
|
for (bGPDpalettecolor *palcolor = palette->colors.first; palcolor; palcolor = palcolor->next) {
|
|
palcolor->flag |= PC_COLOR_LOCKED;
|
|
}
|
|
|
|
/* loop all selected strokes and unlock any color */
|
|
for (bGPDlayer *gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
/* only editable and visible layers are considered */
|
|
if (gpencil_layer_is_editable(gpl) && (gpl->actframe != NULL)) {
|
|
for (bGPDstroke *gps = gpl->actframe->strokes.last; gps; gps = gps->prev) {
|
|
/* only if selected */
|
|
if (gps->flag & GP_STROKE_SELECT) {
|
|
/* skip strokes that are invalid for current view */
|
|
if (ED_gpencil_stroke_can_use(C, gps) == false) {
|
|
continue;
|
|
}
|
|
/* unlock color */
|
|
if (gps->palcolor != NULL) {
|
|
gps->palcolor->flag &= ~PC_COLOR_LOCKED;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_stroke_lock_color(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Lock Unused Colors";
|
|
ot->idname = "GPENCIL_OT_stroke_lock_color";
|
|
ot->description = "Lock any color not used in any selected stroke";
|
|
|
|
/* api callbacks */
|
|
ot->exec = gp_stroke_lock_color_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
}
|
|
|
|
/* ************************************************ */
|
|
/* Drawing Brushes Operators */
|
|
|
|
/* ******************* Add New Brush ************************ */
|
|
|
|
/* add new brush - wrapper around API */
|
|
static int gp_brush_add_exec(bContext *C, wmOperator *op)
|
|
{
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
|
|
/* if there's no existing container */
|
|
if (ts == NULL) {
|
|
BKE_report(op->reports, RPT_ERROR, "Nowhere for brush data to go");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
/* add new brush now */
|
|
BKE_gpencil_brush_addnew(ts, DATA_("GP_Brush"), true);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_brush_add(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Add Brush";
|
|
ot->idname = "GPENCIL_OT_brush_add";
|
|
ot->description = "Add new Grease Pencil drawing brush for the active Grease Pencil data-block";
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_brush_add_exec;
|
|
ot->poll = gp_add_poll;
|
|
}
|
|
|
|
/* ******************* Remove Active Brush ************************* */
|
|
|
|
static int gp_brush_remove_exec(bContext *C, wmOperator *op)
|
|
{
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
bGPDbrush *brush = BKE_gpencil_brush_getactive(ts);
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, ts, brush))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
if (BLI_listbase_count_at_most(&ts->gp_brushes, 2) < 2) {
|
|
BKE_report(op->reports, RPT_ERROR, "Grease Pencil needs a brush, unable to delete the last one");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
|
|
/* make the brush before this the new active brush
|
|
* - use the one after if this is the first
|
|
* - if this is the only brush, this naturally becomes NULL
|
|
*/
|
|
if (brush->prev)
|
|
BKE_gpencil_brush_setactive(ts, brush->prev);
|
|
else
|
|
BKE_gpencil_brush_setactive(ts, brush->next);
|
|
|
|
/* delete the brush now... */
|
|
BKE_gpencil_brush_delete(ts, brush);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_brush_remove(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Remove Brush";
|
|
ot->idname = "GPENCIL_OT_brush_remove";
|
|
ot->description = "Remove active Grease Pencil drawing brush";
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_brush_remove_exec;
|
|
ot->poll = gp_active_brush_poll;
|
|
}
|
|
|
|
/* ********************** Change Brush ***************************** */
|
|
|
|
static int gp_brush_change_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(evt))
|
|
{
|
|
uiPopupMenu *pup;
|
|
uiLayout *layout;
|
|
|
|
/* call the menu, which will call this operator again, hence the canceled */
|
|
pup = UI_popup_menu_begin(C, op->type->name, ICON_NONE);
|
|
layout = UI_popup_menu_layout(pup);
|
|
uiItemsEnumO(layout, "GPENCIL_OT_brush_change", "brush");
|
|
UI_popup_menu_end(C, pup);
|
|
|
|
return OPERATOR_INTERFACE;
|
|
}
|
|
|
|
static int gp_brush_change_exec(bContext *C, wmOperator *op)
|
|
{
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
bGPDbrush *brush = NULL;
|
|
int brush_num = RNA_enum_get(op->ptr, "brush");
|
|
|
|
/* Get brush or create new one */
|
|
if (brush_num == -1) {
|
|
/* Create brush */
|
|
brush = BKE_gpencil_brush_addnew(ts, DATA_("GP_Brush"), true);
|
|
}
|
|
else {
|
|
/* Try to get brush */
|
|
brush = BLI_findlink(&ts->gp_brushes, brush_num);
|
|
|
|
if (brush == NULL) {
|
|
BKE_reportf(op->reports, RPT_ERROR, "Cannot change to non-existent brush (index = %d)", brush_num);
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
}
|
|
|
|
/* Set active brush */
|
|
BKE_gpencil_brush_setactive(ts, brush);
|
|
|
|
/* updates */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_brush_change(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Change Brush";
|
|
ot->idname = "GPENCIL_OT_brush_change";
|
|
ot->description = "Change active Grease Pencil drawing brush";
|
|
|
|
/* callbacks */
|
|
ot->invoke = gp_brush_change_invoke;
|
|
ot->exec = gp_brush_change_exec;
|
|
ot->poll = gp_active_brush_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* gp brush to use (dynamic enum) */
|
|
ot->prop = RNA_def_enum(ot->srna, "brush", DummyRNA_DEFAULT_items, 0, "Grease Pencil Brush", "");
|
|
RNA_def_enum_funcs(ot->prop, ED_gpencil_brushes_enum_itemf);
|
|
}
|
|
|
|
/* ******************* Move Brush Up/Down ************************** */
|
|
|
|
enum {
|
|
GP_BRUSH_MOVE_UP = -1,
|
|
GP_BRUSH_MOVE_DOWN = 1
|
|
};
|
|
|
|
static int gp_brush_move_exec(bContext *C, wmOperator *op)
|
|
{
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
bGPDbrush *brush = BKE_gpencil_brush_getactive(ts);
|
|
|
|
int direction = RNA_enum_get(op->ptr, "type");
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, ts, brush)) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
/* up or down? */
|
|
if (direction == GP_BRUSH_MOVE_UP) {
|
|
/* up */
|
|
BLI_remlink(&ts->gp_brushes, brush);
|
|
BLI_insertlinkbefore(&ts->gp_brushes, brush->prev, brush);
|
|
}
|
|
else if (direction == GP_BRUSH_MOVE_DOWN) {
|
|
/* down */
|
|
BLI_remlink(&ts->gp_brushes, brush);
|
|
BLI_insertlinkafter(&ts->gp_brushes, brush->next, brush);
|
|
}
|
|
else {
|
|
BLI_assert(0);
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_brush_move(wmOperatorType *ot)
|
|
{
|
|
static const EnumPropertyItem slot_move[] = {
|
|
{GP_BRUSH_MOVE_UP, "UP", 0, "Up", ""},
|
|
{GP_BRUSH_MOVE_DOWN, "DOWN", 0, "Down", ""},
|
|
{0, NULL, 0, NULL, NULL }
|
|
};
|
|
|
|
/* identifiers */
|
|
ot->name = "Move Brush";
|
|
ot->idname = "GPENCIL_OT_brush_move";
|
|
ot->description = "Move the active Grease Pencil drawing brush up/down in the list";
|
|
|
|
/* api callbacks */
|
|
ot->exec = gp_brush_move_exec;
|
|
ot->poll = gp_active_brush_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
ot->prop = RNA_def_enum(ot->srna, "type", slot_move, GP_BRUSH_MOVE_UP, "Type", "");
|
|
}
|
|
|
|
/* ******************* Brush create presets ************************** */
|
|
|
|
static int gp_brush_presets_create_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
BKE_gpencil_brush_init_presets(ts);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_brush_presets_create(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Create Preset Brushes";
|
|
ot->idname = "GPENCIL_OT_brush_presets_create";
|
|
ot->description = "Create a set of predefined Grease Pencil drawing brushes";
|
|
|
|
/* api callbacks */
|
|
ot->exec = gp_brush_presets_create_exec;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
}
|
|
|
|
/* ***************** Copy Brush ************************ */
|
|
|
|
static int gp_brush_copy_exec(bContext *C, wmOperator *op)
|
|
{
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
|
|
/* if there's no existing container */
|
|
if (ts == NULL) {
|
|
BKE_report(op->reports, RPT_ERROR, "Nowhere for brush data to go");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
bGPDbrush *brush = BKE_gpencil_brush_getactive(ts);
|
|
bGPDbrush *newbrush;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, brush))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* create a brush and duplicate data */
|
|
newbrush = BKE_gpencil_brush_addnew(ts, brush->info, true);
|
|
newbrush->thickness = brush->thickness;
|
|
newbrush->draw_smoothfac = brush->draw_smoothfac;
|
|
newbrush->draw_smoothlvl = brush->draw_smoothlvl;
|
|
newbrush->sublevel = brush->sublevel;
|
|
newbrush->flag = brush->flag;
|
|
newbrush->draw_sensitivity = brush->draw_sensitivity;
|
|
newbrush->draw_strength = brush->draw_strength;
|
|
newbrush->draw_jitter = brush->draw_jitter;
|
|
newbrush->draw_angle = brush->draw_angle;
|
|
newbrush->draw_angle_factor = brush->draw_angle_factor;
|
|
newbrush->draw_random_press = brush->draw_random_press;
|
|
newbrush->draw_random_sub = brush->draw_random_sub;
|
|
|
|
/* free automatic curves created by default (replaced by copy) */
|
|
curvemapping_free(newbrush->cur_sensitivity);
|
|
curvemapping_free(newbrush->cur_strength);
|
|
curvemapping_free(newbrush->cur_jitter);
|
|
|
|
/* make a copy of curves */
|
|
newbrush->cur_sensitivity = curvemapping_copy(brush->cur_sensitivity);
|
|
newbrush->cur_strength = curvemapping_copy(brush->cur_strength);
|
|
newbrush->cur_jitter = curvemapping_copy(brush->cur_jitter);
|
|
|
|
BKE_gpencil_brush_setactive(ts, newbrush);
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_brush_copy(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Copy Brush";
|
|
ot->idname = "GPENCIL_OT_brush_copy";
|
|
ot->description = "Copy current Grease Pencil drawing brush";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_brush_copy_exec;
|
|
ot->poll = gp_active_brush_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* ***************** Select Brush ************************ */
|
|
|
|
static int gp_brush_select_exec(bContext *C, wmOperator *op)
|
|
{
|
|
ToolSettings *ts = CTX_data_tool_settings(C);
|
|
|
|
/* if there's no existing container */
|
|
if (ts == NULL) {
|
|
BKE_report(op->reports, RPT_ERROR, "Nowhere to go");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
const int index = RNA_int_get(op->ptr, "index");
|
|
bGPDbrush *brush = BLI_findlink(&ts->gp_brushes, index);
|
|
/* sanity checks */
|
|
if (ELEM(NULL, brush)) {
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
BKE_gpencil_brush_setactive(ts, brush);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_brush_select(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Select Brush";
|
|
ot->idname = "GPENCIL_OT_brush_select";
|
|
ot->description = "Select a Grease Pencil drawing brush";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_brush_select_exec;
|
|
ot->poll = gp_active_brush_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* properties */
|
|
RNA_def_int(ot->srna, "index", 0, 0, INT_MAX, "Index", "Index of Drawing Brush", 0, INT_MAX);
|
|
}
|
|
|
|
/* ************************************************ */
|
|
/* Palette Operators */
|
|
|
|
/* ******************* Add New Palette ************************ */
|
|
|
|
/* add new palette - wrapper around API */
|
|
static int gp_palette_add_exec(bContext *C, wmOperator *op)
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
bGPdata **gpd_ptr = ED_gpencil_data_get_pointers(C, NULL);
|
|
|
|
/* if there's no existing Grease-Pencil data there, add some */
|
|
if (gpd_ptr == NULL) {
|
|
BKE_report(op->reports, RPT_ERROR, "Nowhere for grease pencil data to go");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
if (*gpd_ptr == NULL)
|
|
*gpd_ptr = BKE_gpencil_data_addnew(bmain, DATA_("GPencil"));
|
|
|
|
/* add new palette now */
|
|
BKE_gpencil_palette_addnew(*gpd_ptr, DATA_("GP_Palette"), true);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palette_add(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Add Palette";
|
|
ot->idname = "GPENCIL_OT_palette_add";
|
|
ot->description = "Add new Grease Pencil palette for the active Grease Pencil data-block";
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palette_add_exec;
|
|
ot->poll = gp_add_poll;
|
|
}
|
|
|
|
/* ******************* Remove Active Palette ************************* */
|
|
|
|
static int gp_palette_remove_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
if (BLI_listbase_count_at_most(&gpd->palettes, 2) < 2) {
|
|
BKE_report(op->reports, RPT_ERROR, "Grease Pencil needs a palette, unable to delete the last one");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
|
|
/* make the palette before this the new active palette
|
|
* - use the one after if this is the first
|
|
* - if this is the only palette, this naturally becomes NULL
|
|
*/
|
|
if (palette->prev)
|
|
BKE_gpencil_palette_setactive(gpd, palette->prev);
|
|
else
|
|
BKE_gpencil_palette_setactive(gpd, palette->next);
|
|
|
|
/* delete the palette now... */
|
|
BKE_gpencil_palette_delete(gpd, palette);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palette_remove(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Remove palette";
|
|
ot->idname = "GPENCIL_OT_palette_remove";
|
|
ot->description = "Remove active Grease Pencil palette";
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palette_remove_exec;
|
|
ot->poll = gp_active_palette_poll;
|
|
}
|
|
|
|
/* ********************** Change Palette ***************************** */
|
|
|
|
static int gp_palette_change_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(evt))
|
|
{
|
|
uiPopupMenu *pup;
|
|
uiLayout *layout;
|
|
|
|
/* call the menu, which will call this operator again, hence the canceled */
|
|
pup = UI_popup_menu_begin(C, op->type->name, ICON_NONE);
|
|
layout = UI_popup_menu_layout(pup);
|
|
uiItemsEnumO(layout, "GPENCIL_OT_palette_change", "palette");
|
|
UI_popup_menu_end(C, pup);
|
|
|
|
return OPERATOR_INTERFACE;
|
|
}
|
|
|
|
static int gp_palette_change_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = CTX_data_gpencil_data(C);
|
|
bGPDpalette *palette = NULL;
|
|
int palette_num = RNA_enum_get(op->ptr, "palette");
|
|
|
|
/* Get palette or create new one */
|
|
if (palette_num == -1) {
|
|
/* Create palette */
|
|
palette = BKE_gpencil_palette_addnew(gpd, DATA_("GP_Palette"), true);
|
|
}
|
|
else {
|
|
/* Try to get palette */
|
|
palette = BLI_findlink(&gpd->palettes, palette_num);
|
|
|
|
if (palette == NULL) {
|
|
BKE_reportf(op->reports, RPT_ERROR, "Cannot change to non-existent palette (index = %d)", palette_num);
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
}
|
|
|
|
/* Set active palette */
|
|
BKE_gpencil_palette_setactive(gpd, palette);
|
|
|
|
/* updates */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palette_change(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Change Palette";
|
|
ot->idname = "GPENCIL_OT_palette_change";
|
|
ot->description = "Change active Grease Pencil palette";
|
|
|
|
/* callbacks */
|
|
ot->invoke = gp_palette_change_invoke;
|
|
ot->exec = gp_palette_change_exec;
|
|
ot->poll = gp_active_palette_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* gp palette to use (dynamic enum) */
|
|
ot->prop = RNA_def_enum(ot->srna, "palette", DummyRNA_DEFAULT_items, 0, "Grease Pencil Palette", "");
|
|
RNA_def_enum_funcs(ot->prop, ED_gpencil_palettes_enum_itemf);
|
|
}
|
|
|
|
/* ******************* Lock and hide any color non used in current layer ************************** */
|
|
|
|
static int gp_palette_lock_layer_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
palette = BKE_gpencil_palette_getactive(gpd);
|
|
if (ELEM(NULL, palette))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* first lock and hide all colors */
|
|
for (bGPDpalettecolor *palcolor = palette->colors.first; palcolor; palcolor = palcolor->next) {
|
|
palcolor->flag |= PC_COLOR_LOCKED;
|
|
palcolor->flag |= PC_COLOR_HIDE;
|
|
}
|
|
|
|
/* loop all selected strokes and unlock any color used in active layer */
|
|
for (bGPDlayer *gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
/* only editable and visible layers are considered */
|
|
if (gpencil_layer_is_editable(gpl) && (gpl->actframe != NULL) && (gpl->flag & GP_LAYER_ACTIVE)) {
|
|
for (bGPDstroke *gps = gpl->actframe->strokes.last; gps; gps = gps->prev) {
|
|
/* skip strokes that are invalid for current view */
|
|
if (ED_gpencil_stroke_can_use(C, gps) == false)
|
|
continue;
|
|
|
|
/* unlock/unhide color if not unlocked before */
|
|
if (gps->palcolor != NULL) {
|
|
gps->palcolor->flag &= ~PC_COLOR_LOCKED;
|
|
gps->palcolor->flag &= ~PC_COLOR_HIDE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palette_lock_layer(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Disable Unused Layer Colors";
|
|
ot->idname = "GPENCIL_OT_palette_lock_layer";
|
|
ot->description = "Lock and hide any color not used in any layer";
|
|
|
|
/* api callbacks */
|
|
ot->exec = gp_palette_lock_layer_exec;
|
|
ot->poll = gp_active_layer_poll;
|
|
}
|
|
|
|
/* ************************************************ */
|
|
/* Palette Colors Operators */
|
|
|
|
/* ******************* Add New Palette ************************ */
|
|
|
|
/* add new palette - wrapper around API */
|
|
static int gp_palettecolor_add_exec(bContext *C, wmOperator *op)
|
|
{
|
|
Main *bmain = CTX_data_main(C);
|
|
bGPdata **gpd_ptr = ED_gpencil_data_get_pointers(C, NULL);
|
|
|
|
/* if there's no existing Grease-Pencil data there, add some */
|
|
if (gpd_ptr == NULL) {
|
|
BKE_report(op->reports, RPT_ERROR, "Nowhere for grease pencil data to go");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
if (*gpd_ptr == NULL)
|
|
*gpd_ptr = BKE_gpencil_data_addnew(bmain, DATA_("GPencil"));
|
|
|
|
/* verify palette */
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(*gpd_ptr);
|
|
if (palette == NULL)
|
|
palette = BKE_gpencil_palette_addnew(*gpd_ptr, DATA_("GP_Palette"), true);
|
|
|
|
/* add new palette color now */
|
|
BKE_gpencil_palettecolor_addnew(palette, DATA_("Color"), true);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_add(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Add Palette Color";
|
|
ot->idname = "GPENCIL_OT_palettecolor_add";
|
|
ot->description = "Add new Grease Pencil palette color for the active Grease Pencil data-block";
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palettecolor_add_exec;
|
|
ot->poll = gp_add_poll;
|
|
}
|
|
|
|
/* ******************* Remove Active Palette color ************************* */
|
|
|
|
static int gp_palettecolor_remove_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *color = BKE_gpencil_palettecolor_getactive(palette);
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette, color))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* make the palette color before this the new active color
|
|
* - use the one after if this is the first
|
|
* - if this is the only color, this naturally becomes NULL
|
|
*/
|
|
if (color->prev)
|
|
BKE_gpencil_palettecolor_setactive(palette, color->prev);
|
|
else
|
|
BKE_gpencil_palettecolor_setactive(palette, color->next);
|
|
|
|
/* delete the strokes */
|
|
BKE_gpencil_palettecolor_delete_strokes(gpd, color->info);
|
|
|
|
/* delete the palette color now... */
|
|
BKE_gpencil_palettecolor_delete(palette, color);
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_remove(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Remove palette color";
|
|
ot->idname = "GPENCIL_OT_palettecolor_remove";
|
|
ot->description = "Remove active Grease Pencil palette color";
|
|
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palettecolor_remove_exec;
|
|
ot->poll = gp_active_palettecolor_poll;
|
|
}
|
|
|
|
/* ********************** Isolate palette color **************************** */
|
|
|
|
static int gp_isolate_palettecolor_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *active_color = BKE_gpencil_palettecolor_getactive(palette);
|
|
bGPDpalettecolor *palcolor;
|
|
|
|
int flags = PC_COLOR_LOCKED;
|
|
bool isolate = false;
|
|
|
|
if (RNA_boolean_get(op->ptr, "affect_visibility"))
|
|
flags |= PC_COLOR_HIDE;
|
|
|
|
if (ELEM(NULL, gpd, active_color)) {
|
|
BKE_report(op->reports, RPT_ERROR, "No active color to isolate");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
/* Test whether to isolate or clear all flags */
|
|
for (palcolor = palette->colors.first; palcolor; palcolor = palcolor->next) {
|
|
/* Skip if this is the active one */
|
|
if (palcolor == active_color)
|
|
continue;
|
|
|
|
/* If the flags aren't set, that means that the color is
|
|
* not alone, so we have some colors to isolate still
|
|
*/
|
|
if ((palcolor->flag & flags) == 0) {
|
|
isolate = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Set/Clear flags as appropriate */
|
|
if (isolate) {
|
|
/* Set flags on all "other" colors */
|
|
for (palcolor = palette->colors.first; palcolor; palcolor = palcolor->next) {
|
|
if (palcolor == active_color)
|
|
continue;
|
|
else
|
|
palcolor->flag |= flags;
|
|
}
|
|
}
|
|
else {
|
|
/* Clear flags - Restore everything else */
|
|
for (palcolor = palette->colors.first; palcolor; palcolor = palcolor->next) {
|
|
palcolor->flag &= ~flags;
|
|
}
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_isolate(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Isolate Palette Color";
|
|
ot->idname = "GPENCIL_OT_palettecolor_isolate";
|
|
ot->description = "Toggle whether the active color is the only one that is editable and/or visible";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_isolate_palettecolor_exec;
|
|
ot->poll = gp_active_palettecolor_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* properties */
|
|
RNA_def_boolean(ot->srna, "affect_visibility", false, "Affect Visibility", "In addition to toggling "
|
|
"the editability, also affect the visibility");
|
|
}
|
|
|
|
/* *********************** Hide Palette colors ******************************** */
|
|
|
|
static int gp_palettecolor_hide_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *palcolor = BKE_gpencil_palettecolor_getactive(palette);
|
|
|
|
bool unselected = RNA_boolean_get(op->ptr, "unselected");
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette, palcolor))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
if (unselected) {
|
|
bGPDpalettecolor *color;
|
|
|
|
/* hide unselected */
|
|
for (color = palette->colors.first; color; color = color->next) {
|
|
if (color != palcolor) {
|
|
color->flag |= PC_COLOR_HIDE;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
/* hide selected/active */
|
|
palcolor->flag |= PC_COLOR_HIDE;
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_hide(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Hide Color(s)";
|
|
ot->idname = "GPENCIL_OT_palettecolor_hide";
|
|
ot->description = "Hide selected/unselected Grease Pencil colors";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palettecolor_hide_exec;
|
|
ot->poll = gp_active_palettecolor_poll; /* NOTE: we need an active color to play with */
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
/* props */
|
|
RNA_def_boolean(ot->srna, "unselected", 0, "Unselected", "Hide unselected rather than selected colors");
|
|
}
|
|
|
|
/* ********************** Show All Colors ***************************** */
|
|
|
|
/* poll callback for showing colors */
|
|
static bool gp_palettecolor_reveal_poll(bContext *C)
|
|
{
|
|
return ED_gpencil_data_get_active(C) != NULL;
|
|
}
|
|
|
|
static int gp_palettecolor_reveal_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *palcolor;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* make all colors visible */
|
|
for (palcolor = palette->colors.first; palcolor; palcolor = palcolor->next) {
|
|
palcolor->flag &= ~PC_COLOR_HIDE;
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_reveal(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Show All Colors";
|
|
ot->idname = "GPENCIL_OT_palettecolor_reveal";
|
|
ot->description = "Unhide all hidden Grease Pencil palette colors";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palettecolor_reveal_exec;
|
|
ot->poll = gp_palettecolor_reveal_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* ***************** Lock/Unlock All Palette colors ************************ */
|
|
|
|
static int gp_palettecolor_lock_all_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *palcolor;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* make all layers non-editable */
|
|
for (palcolor = palette->colors.first; palcolor; palcolor = palcolor->next) {
|
|
palcolor->flag |= PC_COLOR_LOCKED;
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_lock_all(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Lock All Colors";
|
|
ot->idname = "GPENCIL_OT_palettecolor_lock_all";
|
|
ot->description = "Lock all Grease Pencil colors to prevent them from being accidentally modified";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palettecolor_lock_all_exec;
|
|
ot->poll = gp_palettecolor_reveal_poll; /* XXX: could use dedicated poll later */
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* -------------------------- */
|
|
|
|
static int gp_palettecolor_unlock_all_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *palcolor;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* make all layers editable again*/
|
|
for (palcolor = palette->colors.first; palcolor; palcolor = palcolor->next) {
|
|
palcolor->flag &= ~PC_COLOR_LOCKED;
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_unlock_all(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Unlock All Colors";
|
|
ot->idname = "GPENCIL_OT_palettecolor_unlock_all";
|
|
ot->description = "Unlock all Grease Pencil colors so that they can be edited";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palettecolor_unlock_all_exec;
|
|
ot->poll = gp_palettecolor_reveal_poll; /* XXX: could use dedicated poll later */
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* ******************* Move Color Up/Down ************************** */
|
|
|
|
enum {
|
|
GP_COLOR_MOVE_UP = -1,
|
|
GP_COLOR_MOVE_DOWN = 1
|
|
};
|
|
|
|
static int gp_palettecolor_move_exec(bContext *C, wmOperator *op)
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *palcolor = BKE_gpencil_palettecolor_getactive(palette);
|
|
|
|
int direction = RNA_enum_get(op->ptr, "direction");
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette, palcolor))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* up or down? */
|
|
if (direction == GP_COLOR_MOVE_UP) {
|
|
/* up */
|
|
BLI_remlink(&palette->colors, palcolor);
|
|
BLI_insertlinkbefore(&palette->colors, palcolor->prev, palcolor);
|
|
}
|
|
else if (direction == GP_COLOR_MOVE_DOWN) {
|
|
/* down */
|
|
BLI_remlink(&palette->colors, palcolor);
|
|
BLI_insertlinkafter(&palette->colors, palcolor->next, palcolor);
|
|
}
|
|
else {
|
|
BLI_assert(0);
|
|
}
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_move(wmOperatorType *ot)
|
|
{
|
|
static const EnumPropertyItem slot_move[] = {
|
|
{GP_COLOR_MOVE_UP, "UP", 0, "Up", ""},
|
|
{GP_COLOR_MOVE_DOWN, "DOWN", 0, "Down", ""},
|
|
{0, NULL, 0, NULL, NULL}
|
|
};
|
|
|
|
/* identifiers */
|
|
ot->name = "Move Palette color";
|
|
ot->idname = "GPENCIL_OT_palettecolor_move";
|
|
ot->description = "Move the active Grease Pencil palette color up/down in the list";
|
|
|
|
/* api callbacks */
|
|
ot->exec = gp_palettecolor_move_exec;
|
|
ot->poll = gp_palettecolor_reveal_poll; /* XXX: could use dedicated poll later */
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
ot->prop = RNA_def_enum(ot->srna, "direction", slot_move, GP_COLOR_MOVE_UP, "Direction", "");
|
|
}
|
|
|
|
/* ***************** Select all strokes using Palette color ************************ */
|
|
|
|
static int gp_palettecolor_select_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *palcolor = BKE_gpencil_palettecolor_getactive(palette);
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette, palcolor))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* read all strokes and select*/
|
|
for (bGPDlayer *gpl = gpd->layers.first; gpl; gpl = gpl->next) {
|
|
/* only editable and visible layers are considered */
|
|
if (gpencil_layer_is_editable(gpl) && (gpl->actframe != NULL)) {
|
|
/* verify something to do */
|
|
for (bGPDstroke *gps = gpl->actframe->strokes.first; gps; gps = gps->next) {
|
|
/* skip strokes that are invalid for current view */
|
|
if (ED_gpencil_stroke_can_use(C, gps) == false)
|
|
continue;
|
|
/* check if the color is editable */
|
|
if (ED_gpencil_stroke_color_use(gpl, gps) == false)
|
|
continue;
|
|
|
|
/* select */
|
|
if (strcmp(palcolor->info, gps->colorname) == 0) {
|
|
bGPDspoint *pt;
|
|
int i;
|
|
|
|
gps->flag |= GP_STROKE_SELECT;
|
|
for (i = 0, pt = gps->points; i < gps->totpoints; i++, pt++) {
|
|
pt->flag |= GP_SPOINT_SELECT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_select(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Select Color";
|
|
ot->idname = "GPENCIL_OT_palettecolor_select";
|
|
ot->description = "Select all Grease Pencil strokes using current color";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palettecolor_select_exec;
|
|
ot->poll = gp_palettecolor_reveal_poll; /* XXX: could use dedicated poll later */
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|
|
|
|
/* ***************** Copy Palette color ************************ */
|
|
|
|
static int gp_palettecolor_copy_exec(bContext *C, wmOperator *UNUSED(op))
|
|
{
|
|
bGPdata *gpd = ED_gpencil_data_get_active(C);
|
|
bGPDpalette *palette = BKE_gpencil_palette_getactive(gpd);
|
|
bGPDpalettecolor *palcolor = BKE_gpencil_palettecolor_getactive(palette);
|
|
bGPDpalettecolor *newcolor;
|
|
|
|
/* sanity checks */
|
|
if (ELEM(NULL, gpd, palette, palcolor))
|
|
return OPERATOR_CANCELLED;
|
|
|
|
/* create a new color and duplicate data */
|
|
newcolor = BKE_gpencil_palettecolor_addnew(palette, palcolor->info, true);
|
|
copy_v4_v4(newcolor->color, palcolor->color);
|
|
copy_v4_v4(newcolor->fill, palcolor->fill);
|
|
newcolor->flag = palcolor->flag;
|
|
|
|
/* notifiers */
|
|
WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
void GPENCIL_OT_palettecolor_copy(wmOperatorType *ot)
|
|
{
|
|
/* identifiers */
|
|
ot->name = "Copy Color";
|
|
ot->idname = "GPENCIL_OT_palettecolor_copy";
|
|
ot->description = "Copy current Grease Pencil palette color";
|
|
|
|
/* callbacks */
|
|
ot->exec = gp_palettecolor_copy_exec;
|
|
ot->poll = gp_active_palettecolor_poll;
|
|
|
|
/* flags */
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
}
|