/* * 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 * This is a new part of Blender */ /** \file * \ingroup edgpencil * * Operators for dealing with GP data-blocks and layers. */ #include #include #include #include #include #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_anim_types.h" #include "DNA_brush_types.h" #include "DNA_gpencil_types.h" #include "DNA_meshdata_types.h" #include "DNA_modifier_types.h" #include "DNA_object_types.h" #include "DNA_scene_types.h" #include "DNA_screen_types.h" #include "DNA_space_types.h" #include "DNA_view3d_types.h" #include "BKE_animsys.h" #include "BKE_brush.h" #include "BKE_context.h" #include "BKE_deform.h" #include "BKE_fcurve.h" #include "BKE_gpencil.h" #include "BKE_gpencil_modifier.h" #include "BKE_library.h" #include "BKE_main.h" #include "BKE_material.h" #include "BKE_modifier.h" #include "BKE_object.h" #include "BKE_paint.h" #include "BKE_report.h" #include "BKE_scene.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_object.h" #include "ED_gpencil.h" #include "DEG_depsgraph.h" #include "DEG_depsgraph_build.h" #include "DEG_depsgraph_query.h" #include "gpencil_intern.h" /* ************************************************ */ /* Datablock Operators */ /* ******************* Add New Data ************************ */ static bool gp_data_add_poll(bContext *C) { /* the base line we have is that we have somewhere to add Grease Pencil data */ return ED_annotation_data_get_pointers(C, NULL) != NULL; } /* add new datablock - wrapper around API */ static int gp_data_add_exec(bContext *C, wmOperator *op) { PointerRNA gpd_owner = {NULL}; bGPdata **gpd_ptr = ED_annotation_data_get_pointers(C, &gpd_owner); if (gpd_ptr == NULL) { BKE_report(op->reports, RPT_ERROR, "Nowhere for grease pencil data to go"); return OPERATOR_CANCELLED; } /* decrement user count and add new datablock */ /* TODO: if a datablock exists, * we should make a copy of it instead of starting fresh (as in other areas) */ Main *bmain = CTX_data_main(C); /* decrement user count of old GP datablock */ if (*gpd_ptr) { bGPdata *gpd = (*gpd_ptr); id_us_min(&gpd->id); } /* Add new datablock, with a single layer ready to use * (so users don't have to perform an extra step). */ bGPdata *gpd = BKE_gpencil_data_addnew(bmain, DATA_("Annotations")); *gpd_ptr = gpd; /* tag for annotations */ gpd->flag |= GP_DATA_ANNOTATIONS; /* add new layer (i.e. a "note") */ BKE_gpencil_layer_addnew(*gpd_ptr, DATA_("Note"), true); /* notifiers */ WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_annotation_add(wmOperatorType *ot) { /* identifiers */ ot->name = "Annotation Add New"; ot->idname = "GPENCIL_OT_annotation_add"; ot->description = "Add new Annotation data-block"; ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; /* callbacks */ ot->exec = gp_data_add_exec; ot->poll = gp_data_add_poll; } /* ******************* Unlink Data ************************ */ /* poll callback for adding data/layers - special */ static bool gp_data_unlink_poll(bContext *C) { bGPdata **gpd_ptr = ED_annotation_data_get_pointers(C, NULL); /* only unlink annotation datablocks */ if ((gpd_ptr != NULL) && (*gpd_ptr != NULL)) { bGPdata *gpd = (*gpd_ptr); if ((gpd->flag & GP_DATA_ANNOTATIONS) == 0) { return false; } } /* 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_annotation_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 = "Annotation Unlink"; ot->idname = "GPENCIL_OT_data_unlink"; ot->description = "Unlink active Annotation 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) { const bool is_annotation = STREQ(op->idname, "GPENCIL_OT_layer_annotation_add"); PointerRNA gpd_owner = {NULL}; Main *bmain = CTX_data_main(C); bGPdata *gpd = NULL; if (is_annotation) { bGPdata **gpd_ptr = ED_annotation_data_get_pointers(C, &gpd_owner); /* 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; } /* Annotations */ if (*gpd_ptr == NULL) { *gpd_ptr = BKE_gpencil_data_addnew(bmain, DATA_("Annotations")); } /* mark as annotation */ (*gpd_ptr)->flag |= GP_DATA_ANNOTATIONS; BKE_gpencil_layer_addnew(*gpd_ptr, DATA_("Note"), true); gpd = *gpd_ptr; } else { /* GP Object */ Object *ob = CTX_data_active_object(C); if ((ob != NULL) && (ob->type == OB_GPENCIL)) { gpd = (bGPdata *)ob->data; BKE_gpencil_layer_addnew(gpd, DATA_("GP_Layer"), true); } } /* notifiers */ if (gpd) { DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY | ID_RECALC_COPY_ON_WRITE); } 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 layer or note for the active data-block"; ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; /* callbacks */ ot->exec = gp_layer_add_exec; ot->poll = gp_add_poll; } static bool gp_add_annotation_poll(bContext *C) { return ED_annotation_data_get_pointers(C, NULL) != NULL; } void GPENCIL_OT_layer_annotation_add(wmOperatorType *ot) { /* identifiers */ ot->name = "Add New Annotation Layer"; ot->idname = "GPENCIL_OT_layer_annotation_add"; ot->description = "Add new Annotation layer or note for the active data-block"; ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; /* callbacks */ ot->exec = gp_layer_add_exec; ot->poll = gp_add_annotation_poll; } /* ******************* Remove Active Layer ************************* */ static int gp_layer_remove_exec(bContext *C, wmOperator *op) { const bool is_annotation = STREQ(op->idname, "GPENCIL_OT_layer_annotation_remove"); bGPdata *gpd = (!is_annotation) ? ED_gpencil_data_get_active(C) : ED_annotation_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 */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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; } static bool gp_active_layer_annotation_poll(bContext *C) { bGPdata *gpd = ED_annotation_data_get_active(C); bGPDlayer *gpl = BKE_gpencil_layer_getactive(gpd); return (gpl != NULL); } void GPENCIL_OT_layer_annotation_remove(wmOperatorType *ot) { /* identifiers */ ot->name = "Remove Annotation Layer"; ot->idname = "GPENCIL_OT_layer_annotation_remove"; ot->description = "Remove active Annotation layer"; ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; /* callbacks */ ot->exec = gp_layer_remove_exec; ot->poll = gp_active_layer_annotation_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) { const bool is_annotation = STREQ(op->idname, "GPENCIL_OT_layer_annotation_move"); bGPdata *gpd = (!is_annotation) ? ED_gpencil_data_get_active(C) : ED_annotation_data_get_active(C); bGPDlayer *gpl = BKE_gpencil_layer_getactive(gpd); const int direction = RNA_enum_get(op->ptr, "type") * -1; /* 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)) { DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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", ""); } void GPENCIL_OT_layer_annotation_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 Annotation Layer"; ot->idname = "GPENCIL_OT_layer_annotation_move"; ot->description = "Move the active Annotation layer up/down in the list"; /* api callbacks */ ot->exec = gp_layer_move_exec; ot->poll = gp_active_layer_annotation_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 */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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; } /* ********************* Duplicate Layer in a new object ************************** */ enum { GP_LAYER_COPY_OBJECT_ALL_FRAME = 0, GP_LAYER_COPY_OBJECT_ACT_FRAME = 1, }; static bool gp_layer_duplicate_object_poll(bContext *C) { ViewLayer *view_layer = CTX_data_view_layer(C); Object *ob = CTX_data_active_object(C); if ((ob == NULL) || (ob->type != OB_GPENCIL)) { return false; } bGPdata *gpd = (bGPdata *)ob->data; bGPDlayer *gpl = BKE_gpencil_layer_getactive(gpd); if (gpl == NULL) { return false; } /* check there are more grease pencil objects */ for (Base *base = view_layer->object_bases.first; base; base = base->next) { if ((base->object != ob) && (base->object->type == OB_GPENCIL)) { return true; } } return false; } static int gp_layer_duplicate_object_exec(bContext *C, wmOperator *op) { Main *bmain = CTX_data_main(C); Scene *scene = CTX_data_scene(C); char name[MAX_ID_NAME - 2]; RNA_string_get(op->ptr, "object", name); if (name[0] == '\0') { return OPERATOR_CANCELLED; } Object *ob_dst = (Object *)BKE_scene_object_find_by_name(scene, name); int mode = RNA_enum_get(op->ptr, "mode"); Object *ob_src = CTX_data_active_object(C); bGPdata *gpd_src = (bGPdata *)ob_src->data; bGPDlayer *gpl_src = BKE_gpencil_layer_getactive(gpd_src); /* Sanity checks. */ if (ELEM(NULL, gpd_src, gpl_src, ob_dst)) { return OPERATOR_CANCELLED; } /* Cannot copy itself and check destination type. */ if ((ob_src == ob_dst) || (ob_dst->type != OB_GPENCIL)) { return OPERATOR_CANCELLED; } bGPdata *gpd_dst = (bGPdata *)ob_dst->data; /* Create new layer. */ bGPDlayer *gpl_dst = BKE_gpencil_layer_addnew(gpd_dst, gpl_src->info, true); /* Need to copy some variables (not all). */ gpl_dst->onion_flag = gpl_src->onion_flag; gpl_dst->thickness = gpl_src->thickness; gpl_dst->line_change = gpl_src->line_change; copy_v4_v4(gpl_dst->tintcolor, gpl_src->tintcolor); gpl_dst->opacity = gpl_src->opacity; /* Create all frames. */ for (bGPDframe *gpf_src = gpl_src->frames.first; gpf_src; gpf_src = gpf_src->next) { if ((mode == GP_LAYER_COPY_OBJECT_ACT_FRAME) && (gpf_src != gpl_src->actframe)) { continue; } /* Create new frame. */ bGPDframe *gpf_dst = BKE_gpencil_frame_addnew(gpl_dst, gpf_src->framenum); /* Copy strokes. */ for (bGPDstroke *gps_src = gpf_src->strokes.first; gps_src; gps_src = gps_src->next) { /* Make copy of source stroke. */ bGPDstroke *gps_dst = BKE_gpencil_stroke_duplicate(gps_src); /* Check if material is in destination object, * otherwise add the slot with the material. */ Material *ma_src = BKE_object_material_get(ob_src, gps_src->mat_nr + 1); if (ma_src != NULL) { int idx = BKE_gpencil_object_material_ensure(bmain, ob_dst, ma_src); /* Reassign the stroke material to the right slot in destination object. */ gps_dst->mat_nr = idx; } /* Add new stroke to frame. */ BLI_addtail(&gpf_dst->strokes, gps_dst); } } /* notifiers */ DEG_id_tag_update(&gpd_dst->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY | ID_RECALC_COPY_ON_WRITE); DEG_id_tag_update(&ob_dst->id, ID_RECALC_COPY_ON_WRITE); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_layer_duplicate_object(wmOperatorType *ot) { static const EnumPropertyItem copy_mode[] = { {GP_LAYER_COPY_OBJECT_ALL_FRAME, "ALL", 0, "All Frames", ""}, {GP_LAYER_COPY_OBJECT_ACT_FRAME, "ACTIVE", 0, "Active Frame", ""}, {0, NULL, 0, NULL, NULL}, }; /* identifiers */ ot->name = "Duplicate Layer to New Object"; ot->idname = "GPENCIL_OT_layer_duplicate_object"; ot->description = "Make a copy of the active Grease Pencil layer to new object"; /* callbacks */ ot->exec = gp_layer_duplicate_object_exec; ot->poll = gp_layer_duplicate_object_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; ot->prop = RNA_def_string( ot->srna, "object", NULL, MAX_ID_NAME - 2, "Object", "Name of the destination object"); RNA_def_property_flag(ot->prop, PROP_HIDDEN | PROP_SKIP_SAVE); RNA_def_enum(ot->srna, "mode", copy_mode, GP_LAYER_COPY_OBJECT_ALL_FRAME, "Mode", ""); } /* ********************* Duplicate Frame ************************** */ enum { GP_FRAME_DUP_ACTIVE = 0, GP_FRAME_DUP_ALL = 1, }; static int gp_frame_duplicate_exec(bContext *C, wmOperator *op) { bGPdata *gpd = ED_gpencil_data_get_active(C); bGPDlayer *gpl = BKE_gpencil_layer_getactive(gpd); Scene *scene = CTX_data_scene(C); int mode = RNA_enum_get(op->ptr, "mode"); /* sanity checks */ if (ELEM(NULL, gpd, gpl)) { return OPERATOR_CANCELLED; } if (mode == 0) { BKE_gpencil_frame_addcopy(gpl, CFRA); } else { for (gpl = gpd->layers.first; gpl; gpl = gpl->next) { if ((gpl->flag & GP_LAYER_LOCKED) == 0) { BKE_gpencil_frame_addcopy(gpl, CFRA); } } } /* notifiers */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_frame_duplicate(wmOperatorType *ot) { static const EnumPropertyItem duplicate_mode[] = { {GP_FRAME_DUP_ACTIVE, "ACTIVE", 0, "Active", "Duplicate frame in active layer only"}, {GP_FRAME_DUP_ALL, "ALL", 0, "All", "Duplicate active frames in all layers"}, {0, NULL, 0, NULL, NULL}, }; /* identifiers */ ot->name = "Duplicate Frame"; ot->idname = "GPENCIL_OT_frame_duplicate"; ot->description = "Make a copy of the active Grease Pencil Frame"; /* callbacks */ ot->exec = gp_frame_duplicate_exec; ot->poll = gp_active_layer_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; ot->prop = RNA_def_enum(ot->srna, "mode", duplicate_mode, GP_FRAME_DUP_ACTIVE, "Mode", ""); } /* ********************* Clean Fill Boundaries on Frame ************************** */ enum { GP_FRAME_CLEAN_FILL_ACTIVE = 0, GP_FRAME_CLEAN_FILL_ALL = 1, }; static int gp_frame_clean_fill_exec(bContext *C, wmOperator *op) { bool changed = false; bGPdata *gpd = ED_gpencil_data_get_active(C); int mode = RNA_enum_get(op->ptr, "mode"); CTX_DATA_BEGIN (C, bGPDlayer *, gpl, editable_gpencil_layers) { bGPDframe *init_gpf = gpl->actframe; if (mode == GP_FRAME_CLEAN_FILL_ALL) { init_gpf = gpl->frames.first; } for (bGPDframe *gpf = init_gpf; gpf; gpf = gpf->next) { if ((gpf == gpl->actframe) || (mode == GP_FRAME_CLEAN_FILL_ALL)) { bGPDstroke *gps, *gpsn; if (gpf == NULL) { continue; } /* simply delete strokes which are no fill */ for (gps = gpf->strokes.first; gps; gps = gpsn) { gpsn = gps->next; /* skip strokes that are invalid for current view */ if (ED_gpencil_stroke_can_use(C, gps) == false) { continue; } /* free stroke */ if (gps->flag & GP_STROKE_NOFILL) { /* free stroke memory arrays, then stroke itself */ if (gps->points) { MEM_freeN(gps->points); } if (gps->dvert) { BKE_gpencil_free_stroke_weights(gps); MEM_freeN(gps->dvert); } MEM_SAFE_FREE(gps->triangles); BLI_freelinkN(&gpf->strokes, gps); changed = true; } } } } } CTX_DATA_END; /* notifiers */ if (changed) { DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); } return OPERATOR_FINISHED; } void GPENCIL_OT_frame_clean_fill(wmOperatorType *ot) { static const EnumPropertyItem duplicate_mode[] = { {GP_FRAME_CLEAN_FILL_ACTIVE, "ACTIVE", 0, "Active Frame Only", "Clean active frame only"}, {GP_FRAME_CLEAN_FILL_ALL, "ALL", 0, "All Frames", "Clean all frames in all layers"}, {0, NULL, 0, NULL, NULL}, }; /* identifiers */ ot->name = "Clean Fill Boundaries"; ot->idname = "GPENCIL_OT_frame_clean_fill"; ot->description = "Remove 'no fill' boundary strokes"; /* callbacks */ ot->exec = gp_frame_clean_fill_exec; ot->poll = gp_active_layer_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; ot->prop = RNA_def_enum(ot->srna, "mode", duplicate_mode, GP_FRAME_DUP_ACTIVE, "Mode", ""); } /* ********************* Clean Loose Boundaries on Frame ************************** */ static int gp_frame_clean_loose_exec(bContext *C, wmOperator *op) { bool changed = false; bGPdata *gpd = ED_gpencil_data_get_active(C); int limit = RNA_int_get(op->ptr, "limit"); const bool is_multiedit = (bool)GPENCIL_MULTIEDIT_SESSIONS_ON(gpd); CTX_DATA_BEGIN (C, bGPDlayer *, gpl, editable_gpencil_layers) { bGPDframe *init_gpf = (is_multiedit) ? gpl->frames.first : gpl->actframe; bGPDstroke *gps = NULL; bGPDstroke *gpsn = NULL; for (bGPDframe *gpf = init_gpf; gpf; gpf = gpf->next) { if ((gpf == gpl->actframe) || ((gpf->flag & GP_FRAME_SELECT) && (is_multiedit))) { if (gpf == NULL) { continue; } /* simply delete strokes which are no loose */ for (gps = gpf->strokes.first; gps; gps = gpsn) { gpsn = gps->next; /* skip strokes that are invalid for current view */ if (ED_gpencil_stroke_can_use(C, gps) == false) { continue; } /* free stroke */ if (gps->totpoints <= limit) { /* free stroke memory arrays, then stroke itself */ if (gps->points) { MEM_freeN(gps->points); } if (gps->dvert) { BKE_gpencil_free_stroke_weights(gps); MEM_freeN(gps->dvert); } MEM_SAFE_FREE(gps->triangles); BLI_freelinkN(&gpf->strokes, gps); changed = true; } } } /* if not multiedit, exit loop*/ if (!is_multiedit) { break; } } } CTX_DATA_END; /* notifiers */ if (changed) { DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); } return OPERATOR_FINISHED; } void GPENCIL_OT_frame_clean_loose(wmOperatorType *ot) { /* identifiers */ ot->name = "Clean Loose Points"; ot->idname = "GPENCIL_OT_frame_clean_loose"; ot->description = "Remove loose points"; /* callbacks */ ot->exec = gp_frame_clean_loose_exec; ot->poll = gp_active_layer_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; RNA_def_int(ot->srna, "limit", 1, 1, INT_MAX, "Limit", "Number of points to consider stroke as loose", 1, INT_MAX); } /* *********************** 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 { /* Be sure the active layer is unhidden. */ gpl->flag &= ~GP_LAYER_HIDE; } } } else { /* hide selected/active */ layer->flag |= GP_LAYER_HIDE; } /* notifiers */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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 */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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 */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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; /* 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 */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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; /* 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 onion-skinning 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 */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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_next = BKE_gpencil_layer_getactive(gpd); bGPDlayer *gpl_current = gpl_next->prev; 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, POINTER_FROM_INT(gpf->framenum), gpf); } /* read all frames from next layer and add any missing in current layer */ for (bGPDframe *gpf = gpl_next->frames.first; gpf; gpf = gpf->next) { /* try to find frame in current layer */ bGPDframe *frame = BLI_ghash_lookup(gh_frames_cur, POINTER_FROM_INT(gpf->framenum)); if (!frame) { bGPDframe *actframe = BKE_gpencil_layer_getframe( gpl_current, gpf->framenum, GP_GETFRAME_USE_PREV); frame = BKE_gpencil_frame_addnew(gpl_current, gpf->framenum); /* duplicate strokes of current active frame */ if (actframe) { BKE_gpencil_frame_copy_strokes(actframe, frame); } } /* 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 */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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 */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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) { Object *ob = CTX_data_active_object(C); bGPdata *gpd = ED_gpencil_data_get_active(C); bGPDlayer *gpl_act = BKE_gpencil_layer_getactive(gpd); /* sanity checks */ if (ELEM(NULL, gpd, gpl_act, gpl_act->actframe)) { return OPERATOR_CANCELLED; } const int direction = RNA_enum_get(op->ptr, "direction"); const bool is_multiedit = (bool)GPENCIL_MULTIEDIT_SESSIONS_ON(gpd); CTX_DATA_BEGIN (C, bGPDlayer *, gpl, editable_gpencil_layers) { /* temp listbase to store selected strokes */ ListBase selected = {NULL}; bGPDframe *init_gpf = (is_multiedit) ? gpl->frames.first : gpl->actframe; bGPDstroke *gps = NULL; for (bGPDframe *gpf = init_gpf; gpf; gpf = gpf->next) { if ((gpf == gpl->actframe) || ((gpf->flag & GP_FRAME_SELECT) && (is_multiedit))) { if (gpf == NULL) { continue; } bool gpf_lock = false; /* 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(ob, 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) { gpf_lock = true; continue; } } /* some stroke is already at botom */ if ((direction == GP_STROKE_MOVE_BOTTOM) || (direction == GP_STROKE_MOVE_DOWN)) { if (gps == gpf->strokes.first) { gpf_lock = true; continue; } } /* add to list (if not locked) */ if (!gpf_lock) { BLI_addtail(&selected, BLI_genericNodeN(gps)); } } } /* Now do the movement of the stroke */ if (!gpf_lock) { 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); } /* if not multiedit, exit loop*/ if (!is_multiedit) { break; } } } CTX_DATA_END; /* notifiers */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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_TOP, "TOP", 0, "Bring to Front", ""}, {GP_STROKE_MOVE_UP, "UP", 0, "Bring Forward", ""}, {GP_STROKE_MOVE_DOWN, "DOWN", 0, "Send Backward", ""}, {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"; /* callbacks */ ot->exec = gp_stroke_arrange_exec; ot->poll = gp_active_layer_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; /* properties */ 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 *op) { Main *bmain = CTX_data_main(C); Material *ma = NULL; char name[MAX_ID_NAME - 2]; RNA_string_get(op->ptr, "material", name); bGPdata *gpd = ED_gpencil_data_get_active(C); Object *ob = CTX_data_active_object(C); if (name[0] == '\0') { ma = BKE_gpencil_material(ob, ob->actcol); } else { ma = (Material *)BKE_libblock_find_name(bmain, ID_MA, name); if (ma == NULL) { return OPERATOR_CANCELLED; } } /* try to find slot */ int idx = BKE_gpencil_object_material_get_index(ob, ma); if (idx < 0) { return OPERATOR_CANCELLED; } /* sanity checks */ if (ELEM(NULL, gpd)) { return OPERATOR_CANCELLED; } const bool is_multiedit = (bool)GPENCIL_MULTIEDIT_SESSIONS_ON(gpd); if (ELEM(NULL, ma)) { return OPERATOR_CANCELLED; } /* loop all strokes */ CTX_DATA_BEGIN (C, bGPDlayer *, gpl, editable_gpencil_layers) { bGPDframe *init_gpf = (is_multiedit) ? gpl->frames.first : gpl->actframe; for (bGPDframe *gpf = init_gpf; gpf; gpf = gpf->next) { if ((gpf == gpl->actframe) || ((gpf->flag & GP_FRAME_SELECT) && (is_multiedit))) { if (gpf == NULL) { continue; } for (bGPDstroke *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(ob, gpl, gps) == false) { continue; } /* assign new color */ gps->mat_nr = idx; } } } /* if not multiedit, exit loop*/ if (!is_multiedit) { break; } } } CTX_DATA_END; /* notifiers */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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 material"; /* callbacks */ ot->exec = gp_stroke_change_color_exec; ot->poll = gp_active_layer_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; RNA_def_string(ot->srna, "material", NULL, MAX_ID_NAME - 2, "Material", "Name of the material"); } /* ******************* 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); Object *ob = CTX_data_active_object(C); short *totcol = BKE_object_material_num(ob); /* sanity checks */ if (ELEM(NULL, gpd)) { return OPERATOR_CANCELLED; } /* first lock all colors */ for (short i = 0; i < *totcol; i++) { Material *tmp_ma = BKE_object_material_get(ob, i + 1); if (tmp_ma) { tmp_ma->gp_style->flag |= GP_STYLE_COLOR_LOCKED; DEG_id_tag_update(&tmp_ma->id, ID_RECALC_COPY_ON_WRITE); } } /* 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 */ Material *tmp_ma = BKE_object_material_get(ob, gps->mat_nr + 1); if (tmp_ma) { tmp_ma->gp_style->flag &= ~GP_STYLE_COLOR_LOCKED; DEG_id_tag_update(&tmp_ma->id, ID_RECALC_COPY_ON_WRITE); } } } } } /* updates */ DEG_id_tag_update(&gpd->id, ID_RECALC_GEOMETRY); /* copy on write tag is needed, or else no refresh happens */ DEG_id_tag_update(&gpd->id, ID_RECALC_COPY_ON_WRITE); /* notifiers */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); 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; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* ************************************************ */ /* Drawing Brushes Operators */ /* ******************* Brush create presets ************************** */ static int gp_brush_presets_create_exec(bContext *C, wmOperator *UNUSED(op)) { Main *bmain = CTX_data_main(C); ToolSettings *ts = CTX_data_tool_settings(C); BKE_brush_gpencil_presets(bmain, 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; } /*********************** Vertex Groups ***********************************/ static bool gpencil_vertex_group_poll(bContext *C) { Object *ob = CTX_data_active_object(C); if ((ob) && (ob->type == OB_GPENCIL)) { if (!ID_IS_LINKED(ob) && !ID_IS_LINKED(ob->data) && ob->defbase.first) { if (ELEM(ob->mode, OB_MODE_EDIT_GPENCIL, OB_MODE_SCULPT_GPENCIL)) { return true; } } } return false; } static bool gpencil_vertex_group_weight_poll(bContext *C) { Object *ob = CTX_data_active_object(C); if ((ob) && (ob->type == OB_GPENCIL)) { if (!ID_IS_LINKED(ob) && !ID_IS_LINKED(ob->data) && ob->defbase.first) { if (ob->mode == OB_MODE_WEIGHT_GPENCIL) { return true; } } } return false; } static int gpencil_vertex_group_assign_exec(bContext *C, wmOperator *UNUSED(op)) { ToolSettings *ts = CTX_data_tool_settings(C); Object *ob = CTX_data_active_object(C); /* sanity checks */ if (ELEM(NULL, ts, ob, ob->data)) { return OPERATOR_CANCELLED; } ED_gpencil_vgroup_assign(C, ob, ts->vgroup_weight); /* notifiers */ bGPdata *gpd = ob->data; DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED | ND_SPACE_PROPERTIES, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_vertex_group_assign(wmOperatorType *ot) { /* identifiers */ ot->name = "Assign to Vertex Group"; ot->idname = "GPENCIL_OT_vertex_group_assign"; ot->description = "Assign the selected vertices to the active vertex group"; /* api callbacks */ ot->poll = gpencil_vertex_group_poll; ot->exec = gpencil_vertex_group_assign_exec; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* remove point from vertex group */ static int gpencil_vertex_group_remove_from_exec(bContext *C, wmOperator *UNUSED(op)) { Object *ob = CTX_data_active_object(C); /* sanity checks */ if (ELEM(NULL, ob, ob->data)) { return OPERATOR_CANCELLED; } ED_gpencil_vgroup_remove(C, ob); /* notifiers */ bGPdata *gpd = ob->data; DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED | ND_SPACE_PROPERTIES, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_vertex_group_remove_from(wmOperatorType *ot) { /* identifiers */ ot->name = "Remove from Vertex Group"; ot->idname = "GPENCIL_OT_vertex_group_remove_from"; ot->description = "Remove the selected vertices from active or all vertex group(s)"; /* api callbacks */ ot->poll = gpencil_vertex_group_poll; ot->exec = gpencil_vertex_group_remove_from_exec; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } static int gpencil_vertex_group_select_exec(bContext *C, wmOperator *UNUSED(op)) { Object *ob = CTX_data_active_object(C); /* sanity checks */ if (ELEM(NULL, ob, ob->data)) { return OPERATOR_CANCELLED; } ED_gpencil_vgroup_select(C, ob); /* notifiers */ bGPdata *gpd = ob->data; DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED | ND_SPACE_PROPERTIES, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_vertex_group_select(wmOperatorType *ot) { /* identifiers */ ot->name = "Select Vertex Group"; ot->idname = "GPENCIL_OT_vertex_group_select"; ot->description = "Select all the vertices assigned to the active vertex group"; /* api callbacks */ ot->poll = gpencil_vertex_group_poll; ot->exec = gpencil_vertex_group_select_exec; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } static int gpencil_vertex_group_deselect_exec(bContext *C, wmOperator *UNUSED(op)) { Object *ob = CTX_data_active_object(C); /* sanity checks */ if (ELEM(NULL, ob, ob->data)) { return OPERATOR_CANCELLED; } ED_gpencil_vgroup_deselect(C, ob); /* notifiers */ bGPdata *gpd = ob->data; DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED | ND_SPACE_PROPERTIES, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_vertex_group_deselect(wmOperatorType *ot) { /* identifiers */ ot->name = "Deselect Vertex Group"; ot->idname = "GPENCIL_OT_vertex_group_deselect"; ot->description = "Deselect all selected vertices assigned to the active vertex group"; /* api callbacks */ ot->poll = gpencil_vertex_group_poll; ot->exec = gpencil_vertex_group_deselect_exec; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* invert */ static int gpencil_vertex_group_invert_exec(bContext *C, wmOperator *op) { ToolSettings *ts = CTX_data_tool_settings(C); Object *ob = CTX_data_active_object(C); /* sanity checks */ if (ELEM(NULL, ts, ob, ob->data)) { return OPERATOR_CANCELLED; } MDeformVert *dvert; const int def_nr = ob->actdef - 1; bDeformGroup *defgroup = BLI_findlink(&ob->defbase, def_nr); if (defgroup == NULL) { return OPERATOR_CANCELLED; } if (defgroup->flag & DG_LOCK_WEIGHT) { BKE_report(op->reports, RPT_ERROR, "Current Vertex Group is locked"); return OPERATOR_CANCELLED; } CTX_DATA_BEGIN (C, bGPDstroke *, gps, editable_gpencil_strokes) { /* Verify the strokes has something to change. */ if ((gps->totpoints == 0) || (gps->dvert == NULL)) { continue; } for (int i = 0; i < gps->totpoints; i++) { dvert = &gps->dvert[i]; MDeformWeight *dw = defvert_find_index(dvert, def_nr); if (dw == NULL) { defvert_add_index_notest(dvert, def_nr, 1.0f); } else if (dw->weight == 1.0f) { defvert_remove_group(dvert, dw); } else { dw->weight = 1.0f - dw->weight; } } } CTX_DATA_END; /* notifiers */ bGPdata *gpd = ob->data; DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED | ND_SPACE_PROPERTIES, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_vertex_group_invert(wmOperatorType *ot) { /* identifiers */ ot->name = "Invert Vertex Group"; ot->idname = "GPENCIL_OT_vertex_group_invert"; ot->description = "Invert weights to the active vertex group"; /* api callbacks */ ot->poll = gpencil_vertex_group_weight_poll; ot->exec = gpencil_vertex_group_invert_exec; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* smooth */ static int gpencil_vertex_group_smooth_exec(bContext *C, wmOperator *op) { const float fac = RNA_float_get(op->ptr, "factor"); const int repeat = RNA_int_get(op->ptr, "repeat"); ToolSettings *ts = CTX_data_tool_settings(C); Object *ob = CTX_data_active_object(C); /* sanity checks */ if (ELEM(NULL, ts, ob, ob->data)) { return OPERATOR_CANCELLED; } const int def_nr = ob->actdef - 1; bDeformGroup *defgroup = BLI_findlink(&ob->defbase, def_nr); if (defgroup == NULL) { return OPERATOR_CANCELLED; } if (defgroup->flag & DG_LOCK_WEIGHT) { BKE_report(op->reports, RPT_ERROR, "Current Vertex Group is locked"); return OPERATOR_CANCELLED; } bGPDspoint *pta, *ptb, *ptc; MDeformVert *dverta, *dvertb; CTX_DATA_BEGIN (C, bGPDstroke *, gps, editable_gpencil_strokes) { /* Verify the strokes has something to change. */ if ((gps->totpoints == 0) || (gps->dvert == NULL)) { continue; } for (int s = 0; s < repeat; s++) { for (int i = 0; i < gps->totpoints; i++) { /* previous point */ if (i > 0) { pta = &gps->points[i - 1]; dverta = &gps->dvert[i - 1]; } else { pta = &gps->points[i]; dverta = &gps->dvert[i]; } /* current */ ptb = &gps->points[i]; dvertb = &gps->dvert[i]; /* next point */ if (i + 1 < gps->totpoints) { ptc = &gps->points[i + 1]; } else { ptc = &gps->points[i]; } float wa = defvert_find_weight(dverta, def_nr); float wb = defvert_find_weight(dvertb, def_nr); /* the optimal value is the corresponding to the interpolation of the weight * at the distance of point b */ const float opfac = line_point_factor_v3(&ptb->x, &pta->x, &ptc->x); const float optimal = interpf(wa, wb, opfac); /* Based on influence factor, blend between original and optimal */ MDeformWeight *dw = defvert_verify_index(dvertb, def_nr); if (dw) { dw->weight = interpf(wb, optimal, fac); CLAMP(dw->weight, 0.0, 1.0f); } } } } CTX_DATA_END; /* notifiers */ bGPdata *gpd = ob->data; DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED | ND_SPACE_PROPERTIES, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_vertex_group_smooth(wmOperatorType *ot) { /* identifiers */ ot->name = "Smooth Vertex Group"; ot->idname = "GPENCIL_OT_vertex_group_smooth"; ot->description = "Smooth weights to the active vertex group"; /* api callbacks */ ot->poll = gpencil_vertex_group_weight_poll; ot->exec = gpencil_vertex_group_smooth_exec; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; RNA_def_float(ot->srna, "factor", 0.5f, 0.0f, 1.0, "Factor", "", 0.0f, 1.0f); RNA_def_int(ot->srna, "repeat", 1, 1, 10000, "Iterations", "", 1, 200); } /* normalize */ static int gpencil_vertex_group_normalize_exec(bContext *C, wmOperator *op) { ToolSettings *ts = CTX_data_tool_settings(C); Object *ob = CTX_data_active_object(C); /* sanity checks */ if (ELEM(NULL, ts, ob, ob->data)) { return OPERATOR_CANCELLED; } MDeformVert *dvert = NULL; MDeformWeight *dw = NULL; const int def_nr = ob->actdef - 1; bDeformGroup *defgroup = BLI_findlink(&ob->defbase, def_nr); if (defgroup == NULL) { return OPERATOR_CANCELLED; } if (defgroup->flag & DG_LOCK_WEIGHT) { BKE_report(op->reports, RPT_ERROR, "Current Vertex Group is locked"); return OPERATOR_CANCELLED; } CTX_DATA_BEGIN (C, bGPDstroke *, gps, editable_gpencil_strokes) { /* Verify the strokes has something to change. */ if ((gps->totpoints == 0) || (gps->dvert == NULL)) { continue; } /* look for max value */ float maxvalue = 0.0f; for (int i = 0; i < gps->totpoints; i++) { dvert = &gps->dvert[i]; dw = defvert_find_index(dvert, def_nr); if ((dw != NULL) && (dw->weight > maxvalue)) { maxvalue = dw->weight; } } /* normalize weights */ if (maxvalue > 0.0f) { for (int i = 0; i < gps->totpoints; i++) { dvert = &gps->dvert[i]; dw = defvert_find_index(dvert, def_nr); if (dw != NULL) { dw->weight = dw->weight / maxvalue; } } } } CTX_DATA_END; /* notifiers */ bGPdata *gpd = ob->data; DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED | ND_SPACE_PROPERTIES, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_vertex_group_normalize(wmOperatorType *ot) { /* identifiers */ ot->name = "Normalize Vertex Group"; ot->idname = "GPENCIL_OT_vertex_group_normalize"; ot->description = "Normalize weights to the active vertex group"; /* api callbacks */ ot->poll = gpencil_vertex_group_weight_poll; ot->exec = gpencil_vertex_group_normalize_exec; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* normalize all */ static int gpencil_vertex_group_normalize_all_exec(bContext *C, wmOperator *op) { ToolSettings *ts = CTX_data_tool_settings(C); Object *ob = CTX_data_active_object(C); bool lock_active = RNA_boolean_get(op->ptr, "lock_active"); /* sanity checks */ if (ELEM(NULL, ts, ob, ob->data)) { return OPERATOR_CANCELLED; } bDeformGroup *defgroup = NULL; MDeformVert *dvert = NULL; MDeformWeight *dw = NULL; const int def_nr = ob->actdef - 1; const int defbase_tot = BLI_listbase_count(&ob->defbase); if (defbase_tot == 0) { return OPERATOR_CANCELLED; } CTX_DATA_BEGIN (C, bGPDstroke *, gps, editable_gpencil_strokes) { /* Verify the strokes has something to change. */ if ((gps->totpoints == 0) || (gps->dvert == NULL)) { continue; } /* look for tot value */ float *tot_values = MEM_callocN(gps->totpoints * sizeof(float), __func__); for (int i = 0; i < gps->totpoints; i++) { dvert = &gps->dvert[i]; for (int v = 0; v < defbase_tot; v++) { defgroup = BLI_findlink(&ob->defbase, v); /* skip NULL or locked groups */ if ((defgroup == NULL) || (defgroup->flag & DG_LOCK_WEIGHT)) { continue; } /* skip current */ if ((lock_active) && (v == def_nr)) { continue; } dw = defvert_find_index(dvert, v); if (dw != NULL) { tot_values[i] += dw->weight; } } } /* normalize weights */ for (int i = 0; i < gps->totpoints; i++) { if (tot_values[i] == 0.0f) { continue; } dvert = &gps->dvert[i]; for (int v = 0; v < defbase_tot; v++) { defgroup = BLI_findlink(&ob->defbase, v); /* skip NULL or locked groups */ if ((defgroup == NULL) || (defgroup->flag & DG_LOCK_WEIGHT)) { continue; } /* skip current */ if ((lock_active) && (v == def_nr)) { continue; } dw = defvert_find_index(dvert, v); if (dw != NULL) { dw->weight = dw->weight / tot_values[i]; } } } /* free temp array */ MEM_SAFE_FREE(tot_values); } CTX_DATA_END; /* notifiers */ bGPdata *gpd = ob->data; DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED | ND_SPACE_PROPERTIES, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_vertex_group_normalize_all(wmOperatorType *ot) { /* identifiers */ ot->name = "Normalize All Vertex Group"; ot->idname = "GPENCIL_OT_vertex_group_normalize_all"; ot->description = "Normalize all weights of all vertex groups, " "so that for each vertex, the sum of all weights is 1.0"; /* api callbacks */ ot->poll = gpencil_vertex_group_weight_poll; ot->exec = gpencil_vertex_group_normalize_all_exec; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; /* props */ RNA_def_boolean(ot->srna, "lock_active", true, "Lock Active", "Keep the values of the active group while normalizing others"); } /****************************** Join ***********************************/ /* userdata for joined_gpencil_fix_animdata_cb() */ typedef struct tJoinGPencil_AdtFixData { bGPdata *src_gpd; bGPdata *tar_gpd; GHash *names_map; } tJoinGPencil_AdtFixData; /** * Callback to pass to #BKE_fcurves_main_cb() * for RNA Paths attached to each F-Curve used in the #AnimData. */ static void joined_gpencil_fix_animdata_cb(ID *id, FCurve *fcu, void *user_data) { tJoinGPencil_AdtFixData *afd = (tJoinGPencil_AdtFixData *)user_data; ID *src_id = &afd->src_gpd->id; ID *dst_id = &afd->tar_gpd->id; GHashIterator gh_iter; /* Fix paths - If this is the target datablock, it will have some "dirty" paths */ if ((id == src_id) && fcu->rna_path && strstr(fcu->rna_path, "layers[")) { GHASH_ITER (gh_iter, afd->names_map) { const char *old_name = BLI_ghashIterator_getKey(&gh_iter); const char *new_name = BLI_ghashIterator_getValue(&gh_iter); /* only remap if changed; * this still means there will be some waste if there aren't many drivers/keys */ if (!STREQ(old_name, new_name) && strstr(fcu->rna_path, old_name)) { fcu->rna_path = BKE_animsys_fix_rna_path_rename( id, fcu->rna_path, "layers", old_name, new_name, 0, 0, false); /* we don't want to apply a second remapping on this F-Curve now, * so stop trying to fix names names */ break; } } } /* Fix driver targets */ if (fcu->driver) { /* Fix driver references to invalid ID's */ for (DriverVar *dvar = fcu->driver->variables.first; dvar; dvar = dvar->next) { /* Only change the used targets, since the others will need fixing manually anyway. */ DRIVER_TARGETS_USED_LOOPER_BEGIN (dvar) { /* Change the ID's used. */ if (dtar->id == src_id) { dtar->id = dst_id; /* Also check on the subtarget... * We duplicate the logic from drivers_path_rename_fix() here, with our own * little twists so that we know that it isn't going to clobber the wrong data */ if (dtar->rna_path && strstr(dtar->rna_path, "layers[")) { GHASH_ITER (gh_iter, afd->names_map) { const char *old_name = BLI_ghashIterator_getKey(&gh_iter); const char *new_name = BLI_ghashIterator_getValue(&gh_iter); /* Only remap if changed. */ if (!STREQ(old_name, new_name)) { if ((dtar->rna_path) && strstr(dtar->rna_path, old_name)) { /* Fix up path */ dtar->rna_path = BKE_animsys_fix_rna_path_rename( id, dtar->rna_path, "layers", old_name, new_name, 0, 0, false); break; /* no need to try any more names for layer path */ } } } } } } DRIVER_TARGETS_LOOPER_END; } } } /* join objects called from OBJECT_OT_join */ int ED_gpencil_join_objects_exec(bContext *C, wmOperator *op) { Main *bmain = CTX_data_main(C); Scene *scene = CTX_data_scene(C); Depsgraph *depsgraph = CTX_data_ensure_evaluated_depsgraph(C); Object *ob_active = CTX_data_active_object(C); bGPdata *gpd_dst = NULL; bool ok = false; /* Ensure we're in right mode and that the active object is correct */ if (!ob_active || ob_active->type != OB_GPENCIL) { return OPERATOR_CANCELLED; } bGPdata *gpd = (bGPdata *)ob_active->data; if ((!gpd) || GPENCIL_ANY_MODE(gpd)) { return OPERATOR_CANCELLED; } /* Ensure all rotations are applied before */ CTX_DATA_BEGIN (C, Object *, ob_iter, selected_editable_objects) { if (ob_iter->type == OB_GPENCIL) { if ((ob_iter->rot[0] != 0) || (ob_iter->rot[1] != 0) || (ob_iter->rot[2] != 0)) { BKE_report(op->reports, RPT_ERROR, "Apply all rotations before join objects"); return OPERATOR_CANCELLED; } } } CTX_DATA_END; CTX_DATA_BEGIN (C, Object *, ob_iter, selected_editable_objects) { if (ob_iter == ob_active) { ok = true; break; } } CTX_DATA_END; /* that way the active object is always selected */ if (ok == false) { BKE_report(op->reports, RPT_WARNING, "Active object is not a selected grease pencil"); return OPERATOR_CANCELLED; } gpd_dst = ob_active->data; Object *ob_dst = ob_active; /* loop and join all data */ CTX_DATA_BEGIN (C, Object *, ob_iter, selected_editable_objects) { if ((ob_iter->type == OB_GPENCIL) && (ob_iter != ob_active)) { /* we assume that each datablock is not already used in active object */ if (ob_active->data != ob_iter->data) { Object *ob_src = ob_iter; bGPdata *gpd_src = ob_iter->data; /* Apply all GP modifiers before */ for (GpencilModifierData *md = ob_iter->greasepencil_modifiers.first; md; md = md->next) { const GpencilModifierTypeInfo *mti = BKE_gpencil_modifierType_getInfo(md->type); if (mti->bakeModifier) { mti->bakeModifier(bmain, depsgraph, md, ob_iter); } } /* copy vertex groups to the base one's */ int old_idx = 0; for (bDeformGroup *dg = ob_iter->defbase.first; dg; dg = dg->next) { bDeformGroup *vgroup = MEM_dupallocN(dg); int idx = BLI_listbase_count(&ob_active->defbase); defgroup_unique_name(vgroup, ob_active); BLI_addtail(&ob_active->defbase, vgroup); /* update vertex groups in strokes in original data */ for (bGPDlayer *gpl_src = gpd->layers.first; gpl_src; gpl_src = gpl_src->next) { for (bGPDframe *gpf = gpl_src->frames.first; gpf; gpf = gpf->next) { for (bGPDstroke *gps = gpf->strokes.first; gps; gps = gps->next) { MDeformVert *dvert; int i; for (i = 0, dvert = gps->dvert; i < gps->totpoints; i++, dvert++) { if ((dvert->dw) && (dvert->dw->def_nr == old_idx)) { dvert->dw->def_nr = idx; } } } } } old_idx++; } if (ob_active->defbase.first && ob_active->actdef == 0) { ob_active->actdef = 1; } /* add missing materials reading source materials and checking in destination object */ short *totcol = BKE_object_material_num(ob_src); for (short i = 0; i < *totcol; i++) { Material *tmp_ma = BKE_gpencil_material(ob_src, i + 1); BKE_gpencil_object_material_ensure(bmain, ob_dst, tmp_ma); } /* duplicate bGPDlayers */ tJoinGPencil_AdtFixData afd = {0}; afd.src_gpd = gpd_src; afd.tar_gpd = gpd_dst; afd.names_map = BLI_ghash_str_new("joined_gp_layers_map"); float imat[3][3], bmat[3][3]; float offset_global[3]; float offset_local[3]; sub_v3_v3v3(offset_global, ob_active->loc, ob_iter->obmat[3]); copy_m3_m4(bmat, ob_active->obmat); invert_m3_m3(imat, bmat); mul_m3_v3(imat, offset_global); mul_v3_m3v3(offset_local, imat, offset_global); for (bGPDlayer *gpl_src = gpd_src->layers.first; gpl_src; gpl_src = gpl_src->next) { bGPDlayer *gpl_new = BKE_gpencil_layer_duplicate(gpl_src); float diff_mat[4][4]; float inverse_diff_mat[4][4]; /* recalculate all stroke points */ ED_gpencil_parent_location(depsgraph, ob_iter, gpd_src, gpl_src, diff_mat); invert_m4_m4(inverse_diff_mat, diff_mat); Material *ma_src = NULL; for (bGPDframe *gpf = gpl_new->frames.first; gpf; gpf = gpf->next) { for (bGPDstroke *gps = gpf->strokes.first; gps; gps = gps->next) { /* Reassign material. Look old material and try to find in destination. */ ma_src = BKE_gpencil_material(ob_src, gps->mat_nr + 1); gps->mat_nr = BKE_gpencil_object_material_ensure(bmain, ob_dst, ma_src); bGPDspoint *pt; int i; for (i = 0, pt = gps->points; i < gps->totpoints; i++, pt++) { float mpt[3]; mul_v3_m4v3(mpt, inverse_diff_mat, &pt->x); sub_v3_v3(mpt, offset_local); mul_v3_m4v3(&pt->x, diff_mat, mpt); } } } /* be sure name is unique in new object */ BLI_uniquename(&gpd_dst->layers, gpl_new, DATA_("GP_Layer"), '.', offsetof(bGPDlayer, info), sizeof(gpl_new->info)); BLI_ghash_insert(afd.names_map, BLI_strdup(gpl_src->info), gpl_new->info); /* add to destination datablock */ BLI_addtail(&gpd_dst->layers, gpl_new); } /* Fix all the animation data */ BKE_fcurves_main_cb(bmain, joined_gpencil_fix_animdata_cb, &afd); BLI_ghash_free(afd.names_map, MEM_freeN, NULL); /* Only copy over animdata now, after all the remapping has been done, * so that we don't have to worry about ambiguities re which datablock * a layer came from! */ if (ob_iter->adt) { if (ob_active->adt == NULL) { /* no animdata, so just use a copy of the whole thing */ ob_active->adt = BKE_animdata_copy(bmain, ob_iter->adt, 0); } else { /* merge in data - we'll fix the drivers manually */ BKE_animdata_merge_copy( bmain, &ob_active->id, &ob_iter->id, ADT_MERGECOPY_KEEP_DST, false); } } if (gpd_src->adt) { if (gpd_dst->adt == NULL) { /* no animdata, so just use a copy of the whole thing */ gpd_dst->adt = BKE_animdata_copy(bmain, gpd_src->adt, 0); } else { /* merge in data - we'll fix the drivers manually */ BKE_animdata_merge_copy( bmain, &gpd_dst->id, &gpd_src->id, ADT_MERGECOPY_KEEP_DST, false); } } DEG_id_tag_update(&gpd_src->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY | ID_RECALC_COPY_ON_WRITE); } /* Free the old object */ ED_object_base_free_and_unlink(bmain, scene, ob_iter); } } CTX_DATA_END; DEG_id_tag_update(&gpd_dst->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY | ID_RECALC_COPY_ON_WRITE); DEG_relations_tag_update(bmain); /* because we removed object(s) */ WM_event_add_notifier(C, NC_SCENE | ND_OB_ACTIVE, scene); return OPERATOR_FINISHED; } /* Color Handle operator */ static bool gpencil_active_color_poll(bContext *C) { Object *ob = CTX_data_active_object(C); if (ob && ob->data && (ob->type == OB_GPENCIL)) { short *totcolp = BKE_object_material_num(ob); return *totcolp > 0; } return false; } /* **************** Lock and hide any color non used in current layer ************************** */ static int gpencil_lock_layer_exec(bContext *C, wmOperator *UNUSED(op)) { bGPdata *gpd = ED_gpencil_data_get_active(C); Object *ob = CTX_data_active_object(C); MaterialGPencilStyle *gp_style = NULL; /* sanity checks */ if (ELEM(NULL, gpd)) { return OPERATOR_CANCELLED; } /* first lock and hide all colors */ Material *ma = NULL; short *totcol = BKE_object_material_num(ob); if (totcol == 0) { return OPERATOR_CANCELLED; } for (short i = 0; i < *totcol; i++) { ma = BKE_gpencil_material(ob, i + 1); if (ma) { gp_style = ma->gp_style; gp_style->flag |= GP_STYLE_COLOR_LOCKED; gp_style->flag |= GP_STYLE_COLOR_HIDE; DEG_id_tag_update(&ma->id, ID_RECALC_COPY_ON_WRITE); } } /* 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; } ma = BKE_gpencil_material(ob, gps->mat_nr + 1); DEG_id_tag_update(&ma->id, ID_RECALC_COPY_ON_WRITE); gp_style = ma->gp_style; /* unlock/unhide color if not unlocked before */ if (gp_style != NULL) { gp_style->flag &= ~GP_STYLE_COLOR_LOCKED; gp_style->flag &= ~GP_STYLE_COLOR_HIDE; } } } } /* updates */ DEG_id_tag_update(&gpd->id, ID_RECALC_GEOMETRY); /* copy on write tag is needed, or else no refresh happens */ DEG_id_tag_update(&gpd->id, ID_RECALC_COPY_ON_WRITE); /* notifiers */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_lock_layer(wmOperatorType *ot) { /* identifiers */ ot->name = "Disable Unused Layer Colors"; ot->idname = "GPENCIL_OT_lock_layer"; ot->description = "Lock and hide any color not used in any layer"; /* api callbacks */ ot->exec = gpencil_lock_layer_exec; ot->poll = gp_active_layer_poll; } /* ********************** Isolate gpencil_ color **************************** */ static int gpencil_color_isolate_exec(bContext *C, wmOperator *op) { bGPdata *gpd = ED_gpencil_data_get_active(C); Object *ob = CTX_data_active_object(C); Material *active_ma = BKE_gpencil_material(ob, ob->actcol); MaterialGPencilStyle *active_color = BKE_gpencil_material_settings(ob, ob->actcol); MaterialGPencilStyle *gp_style; int flags = GP_STYLE_COLOR_LOCKED; bool isolate = false; if (RNA_boolean_get(op->ptr, "affect_visibility")) { flags |= GP_STYLE_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 */ Material *ma = NULL; short *totcol = BKE_object_material_num(ob); for (short i = 0; i < *totcol; i++) { ma = BKE_gpencil_material(ob, i + 1); /* Skip if this is the active one */ if ((ma == NULL) || (ma == active_ma)) { continue; } /* If the flags aren't set, that means that the color is * not alone, so we have some colors to isolate still */ gp_style = ma->gp_style; if ((gp_style->flag & flags) == 0) { isolate = true; break; } } /* Set/Clear flags as appropriate */ if (isolate) { /* Set flags on all "other" colors */ for (short i = 0; i < *totcol; i++) { ma = BKE_gpencil_material(ob, i + 1); if (ma == NULL) { continue; } gp_style = ma->gp_style; if (gp_style == active_color) { continue; } else { gp_style->flag |= flags; } DEG_id_tag_update(&ma->id, ID_RECALC_COPY_ON_WRITE); } } else { /* Clear flags - Restore everything else */ for (short i = 0; i < *totcol; i++) { ma = BKE_gpencil_material(ob, i + 1); if (ma == NULL) { continue; } gp_style = ma->gp_style; gp_style->flag &= ~flags; DEG_id_tag_update(&ma->id, ID_RECALC_COPY_ON_WRITE); } } /* notifiers */ DEG_id_tag_update(&gpd->id, ID_RECALC_GEOMETRY); /* copy on write tag is needed, or else no refresh happens */ DEG_id_tag_update(&gpd->id, ID_RECALC_COPY_ON_WRITE); WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_color_isolate(wmOperatorType *ot) { /* identifiers */ ot->name = "Isolate Color"; ot->idname = "GPENCIL_OT_color_isolate"; ot->description = "Toggle whether the active color is the only one that is editable and/or visible"; /* callbacks */ ot->exec = gpencil_color_isolate_exec; ot->poll = gpencil_active_color_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 colors ******************************** */ static int gpencil_color_hide_exec(bContext *C, wmOperator *op) { Object *ob = CTX_data_active_object(C); bGPdata *gpd = (bGPdata *)ob->data; MaterialGPencilStyle *active_color = BKE_gpencil_material_settings(ob, ob->actcol); bool unselected = RNA_boolean_get(op->ptr, "unselected"); Material *ma = NULL; short *totcol = BKE_object_material_num(ob); if (totcol == 0) { return OPERATOR_CANCELLED; } if (unselected) { /* hide unselected */ MaterialGPencilStyle *color = NULL; for (short i = 0; i < *totcol; i++) { ma = BKE_gpencil_material(ob, i + 1); if (ma) { color = ma->gp_style; if (active_color != color) { color->flag |= GP_STYLE_COLOR_HIDE; DEG_id_tag_update(&ma->id, ID_RECALC_COPY_ON_WRITE); } } } } else { /* hide selected/active */ active_color->flag |= GP_STYLE_COLOR_HIDE; } /* updates */ DEG_id_tag_update(&gpd->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); /* copy on write tag is needed, or else no refresh happens */ DEG_id_tag_update(&gpd->id, ID_RECALC_COPY_ON_WRITE); /* notifiers */ WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_color_hide(wmOperatorType *ot) { /* identifiers */ ot->name = "Hide Color(s)"; ot->idname = "GPENCIL_OT_color_hide"; ot->description = "Hide selected/unselected Grease Pencil colors"; /* callbacks */ ot->exec = gpencil_color_hide_exec; ot->poll = gpencil_active_color_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 ***************************** */ static int gpencil_color_reveal_exec(bContext *C, wmOperator *UNUSED(op)) { Object *ob = CTX_data_active_object(C); bGPdata *gpd = (bGPdata *)ob->data; Material *ma = NULL; short *totcol = BKE_object_material_num(ob); if (totcol == 0) { return OPERATOR_CANCELLED; } /* make all colors visible */ MaterialGPencilStyle *gp_style = NULL; for (short i = 0; i < *totcol; i++) { ma = BKE_gpencil_material(ob, i + 1); if (ma) { gp_style = ma->gp_style; gp_style->flag &= ~GP_STYLE_COLOR_HIDE; DEG_id_tag_update(&ma->id, ID_RECALC_COPY_ON_WRITE); } } /* updates */ DEG_id_tag_update(&gpd->id, ID_RECALC_GEOMETRY); /* copy on write tag is needed, or else no refresh happens */ DEG_id_tag_update(&gpd->id, ID_RECALC_COPY_ON_WRITE); /* notifiers */ WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_color_reveal(wmOperatorType *ot) { /* identifiers */ ot->name = "Show All Colors"; ot->idname = "GPENCIL_OT_color_reveal"; ot->description = "Unhide all hidden Grease Pencil colors"; /* callbacks */ ot->exec = gpencil_color_reveal_exec; ot->poll = gpencil_active_color_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* ***************** Lock/Unlock All colors ************************ */ static int gpencil_color_lock_all_exec(bContext *C, wmOperator *UNUSED(op)) { Object *ob = CTX_data_active_object(C); bGPdata *gpd = (bGPdata *)ob->data; Material *ma = NULL; short *totcol = BKE_object_material_num(ob); if (totcol == 0) { return OPERATOR_CANCELLED; } /* make all layers non-editable */ MaterialGPencilStyle *gp_style = NULL; for (short i = 0; i < *totcol; i++) { ma = BKE_gpencil_material(ob, i + 1); if (ma) { gp_style = ma->gp_style; gp_style->flag |= GP_STYLE_COLOR_LOCKED; DEG_id_tag_update(&ma->id, ID_RECALC_COPY_ON_WRITE); } } /* updates */ DEG_id_tag_update(&gpd->id, ID_RECALC_GEOMETRY); /* copy on write tag is needed, or else no refresh happens */ DEG_id_tag_update(&gpd->id, ID_RECALC_COPY_ON_WRITE); /* notifiers */ WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_color_lock_all(wmOperatorType *ot) { /* identifiers */ ot->name = "Lock All Colors"; ot->idname = "GPENCIL_OT_color_lock_all"; ot->description = "Lock all Grease Pencil colors to prevent them from being accidentally modified"; /* callbacks */ ot->exec = gpencil_color_lock_all_exec; ot->poll = gpencil_active_color_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* -------------------------- */ static int gpencil_color_unlock_all_exec(bContext *C, wmOperator *UNUSED(op)) { Object *ob = CTX_data_active_object(C); bGPdata *gpd = (bGPdata *)ob->data; Material *ma = NULL; short *totcol = BKE_object_material_num(ob); if (totcol == 0) { return OPERATOR_CANCELLED; } /* make all layers editable again*/ MaterialGPencilStyle *gp_style = NULL; for (short i = 0; i < *totcol; i++) { ma = BKE_gpencil_material(ob, i + 1); if (ma) { gp_style = ma->gp_style; gp_style->flag &= ~GP_STYLE_COLOR_LOCKED; DEG_id_tag_update(&ma->id, ID_RECALC_COPY_ON_WRITE); } } /* updates */ DEG_id_tag_update(&gpd->id, ID_RECALC_GEOMETRY); /* copy on write tag is needed, or else no refresh happens */ DEG_id_tag_update(&gpd->id, ID_RECALC_COPY_ON_WRITE); /* notifiers */ WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_color_unlock_all(wmOperatorType *ot) { /* identifiers */ ot->name = "Unlock All Colors"; ot->idname = "GPENCIL_OT_color_unlock_all"; ot->description = "Unlock all Grease Pencil colors so that they can be edited"; /* callbacks */ ot->exec = gpencil_color_unlock_all_exec; ot->poll = gpencil_active_color_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* ***************** Select all strokes using color ************************ */ static int gpencil_color_select_exec(bContext *C, wmOperator *op) { bGPdata *gpd = ED_gpencil_data_get_active(C); Object *ob = CTX_data_active_object(C); MaterialGPencilStyle *gp_style = BKE_gpencil_material_settings(ob, ob->actcol); const bool is_multiedit = (bool)GPENCIL_MULTIEDIT_SESSIONS_ON(gpd); const bool deselected = RNA_boolean_get(op->ptr, "deselect"); /* sanity checks */ if (ELEM(NULL, gpd, gp_style)) { return OPERATOR_CANCELLED; } /* read all strokes and select*/ CTX_DATA_BEGIN (C, bGPDlayer *, gpl, editable_gpencil_layers) { bGPDframe *init_gpf = (is_multiedit) ? gpl->frames.first : gpl->actframe; for (bGPDframe *gpf = init_gpf; gpf; gpf = gpf->next) { if ((gpf == gpl->actframe) || ((gpf->flag & GP_FRAME_SELECT) && (is_multiedit))) { /* verify something to do */ for (bGPDstroke *gps = gpf->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(ob, gpl, gps) == false) { continue; } /* select */ if (ob->actcol == gps->mat_nr + 1) { bGPDspoint *pt; int i; if (!deselected) { gps->flag |= GP_STROKE_SELECT; } else { gps->flag &= ~GP_STROKE_SELECT; } for (i = 0, pt = gps->points; i < gps->totpoints; i++, pt++) { if (!deselected) { pt->flag |= GP_SPOINT_SELECT; } else { pt->flag &= ~GP_SPOINT_SELECT; } } } } } /* if not multiedit, exit loop*/ if (!is_multiedit) { break; } } } CTX_DATA_END; /* copy on write tag is needed, or else no refresh happens */ DEG_id_tag_update(&gpd->id, ID_RECALC_GEOMETRY | ID_RECALC_COPY_ON_WRITE); /* notifiers */ WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); return OPERATOR_FINISHED; } void GPENCIL_OT_color_select(wmOperatorType *ot) { /* identifiers */ ot->name = "Select Color"; ot->idname = "GPENCIL_OT_color_select"; ot->description = "Select all Grease Pencil strokes using current color"; /* callbacks */ ot->exec = gpencil_color_select_exec; ot->poll = gpencil_active_color_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; /* props */ ot->prop = RNA_def_boolean(ot->srna, "deselect", 0, "Deselect", "Unselect strokes"); RNA_def_property_flag(ot->prop, PROP_HIDDEN | PROP_SKIP_SAVE); } /* ***************** Set selected stroke material the active material ************************ */ static int gpencil_set_active_material_exec(bContext *C, wmOperator *op) { Object *ob = CTX_data_active_object(C); bGPdata *gpd = ED_gpencil_data_get_active(C); bool changed = false; /* Sanity checks. */ if (gpd == NULL) { BKE_report(op->reports, RPT_ERROR, "No Grease Pencil data"); return OPERATOR_CANCELLED; } /* Loop all selected strokes. */ GP_EDITABLE_STROKES_BEGIN (gpstroke_iter, C, gpl, gps) { if (gps->flag & GP_STROKE_SELECT) { /* Change Active material. */ ob->actcol = gps->mat_nr + 1; changed = true; break; } } GP_EDITABLE_STROKES_END(gpstroke_iter); /* notifiers */ if (changed) { WM_event_add_notifier(C, NC_GPENCIL | ND_DATA | NA_EDITED, NULL); } return OPERATOR_FINISHED; } void GPENCIL_OT_set_active_material(wmOperatorType *ot) { /* identifiers */ ot->name = "Set active material"; ot->idname = "GPENCIL_OT_set_active_material"; ot->description = "Set the selected stroke material as the active material"; /* callbacks */ ot->exec = gpencil_set_active_material_exec; ot->poll = gpencil_active_color_poll; /* flags */ ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO; } /* Parent GPencil object to Lattice */ bool ED_gpencil_add_lattice_modifier(const bContext *C, ReportList *reports, Object *ob, Object *ob_latt) { Main *bmain = CTX_data_main(C); Scene *scene = CTX_data_scene(C); if (ob == NULL) { return false; } /* if no lattice modifier, add a new one */ GpencilModifierData *md = BKE_gpencil_modifiers_findByType(ob, eGpencilModifierType_Lattice); if (md == NULL) { md = ED_object_gpencil_modifier_add( reports, bmain, scene, ob, "Lattice", eGpencilModifierType_Lattice); if (md == NULL) { BKE_report(reports, RPT_ERROR, "Unable to add a new Lattice modifier to object"); return false; } DEG_id_tag_update(&ob->id, ID_RECALC_TRANSFORM | ID_RECALC_GEOMETRY); } /* verify lattice */ LatticeGpencilModifierData *mmd = (LatticeGpencilModifierData *)md; if (mmd->object == NULL) { mmd->object = ob_latt; } else { if (ob_latt != mmd->object) { BKE_report(reports, RPT_ERROR, "The existing Lattice modifier is already using a different Lattice object"); return false; } } return true; }