4 * ***** BEGIN GPL LICENSE BLOCK *****
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software Foundation,
18 * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
20 * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
21 * All rights reserved.
23 * Contributor(s): Blender Foundation, 2009
25 * ***** END GPL LICENSE BLOCK *****
32 #include "MEM_guardedalloc.h"
34 #include "DNA_action_types.h"
35 #include "DNA_curve_types.h"
36 #include "DNA_mesh_types.h"
37 #include "DNA_meshdata_types.h"
38 #include "DNA_modifier_types.h"
39 #include "DNA_object_types.h"
40 #include "DNA_object_force.h"
41 #include "DNA_scene_types.h"
43 #include "BLI_arithb.h"
44 #include "BLI_listbase.h"
46 #include "BKE_action.h"
47 #include "BKE_curve.h"
48 #include "BKE_context.h"
49 #include "BKE_depsgraph.h"
50 #include "BKE_displist.h"
51 #include "BKE_DerivedMesh.h"
52 #include "BKE_effect.h"
53 #include "BKE_global.h"
54 #include "BKE_lattice.h"
56 #include "BKE_modifier.h"
57 #include "BKE_multires.h"
58 #include "BKE_report.h"
59 #include "BKE_object.h"
60 #include "BKE_particle.h"
61 #include "BKE_softbody.h"
62 #include "BKE_utildefines.h"
64 #include "RNA_access.h"
65 #include "RNA_define.h"
66 #include "RNA_enum_types.h"
68 #include "ED_screen.h"
73 #include "object_intern.h"
75 /******************************** API ****************************/
77 int ED_object_modifier_add(ReportList *reports, Scene *scene, Object *ob, int type)
80 ModifierTypeInfo *mti = modifierType_getInfo(type);
82 if(mti->flags&eModifierTypeFlag_Single) {
83 if(modifiers_findByType(ob, type)) {
84 BKE_report(reports, RPT_WARNING, "Only one modifier of this type allowed.");
89 if(type == eModifierType_ParticleSystem) {
90 object_add_particle_system(scene, ob);
93 if(mti->flags&eModifierTypeFlag_RequiresOriginalData) {
94 md = ob->modifiers.first;
96 while(md && modifierType_getInfo(md->type)->type==eModifierTypeType_OnlyDeform)
99 BLI_insertlinkbefore(&ob->modifiers, md, modifier_new(type));
102 BLI_addtail(&ob->modifiers, modifier_new(type));
105 if(type == eModifierType_Softbody) {
107 ob->soft= sbNew(scene);
108 ob->softflag |= OB_SB_GOAL|OB_SB_EDGES;
111 else if(type == eModifierType_Collision) {
113 ob->pd= object_add_collision_fields();
116 DAG_scene_sort(scene);
118 else if(type == eModifierType_Surface)
119 DAG_scene_sort(scene);
122 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
127 int ED_object_modifier_remove(ReportList *reports, Scene *scene, Object *ob, ModifierData *md)
131 /* It seems on rapid delete it is possible to
132 * get called twice on same modifier, so make
133 * sure it is in list. */
134 for(obmd=ob->modifiers.first; obmd; obmd=obmd->next)
142 if(md->type == eModifierType_ParticleSystem) {
143 ParticleSystemModifierData *psmd=(ParticleSystemModifierData*)md;
145 BLI_remlink(&ob->particlesystem, psmd->psys);
146 psys_free(ob, psmd->psys);
148 else if(md->type == eModifierType_Softbody) {
155 else if(md->type == eModifierType_Collision) {
159 DAG_scene_sort(scene);
161 else if(md->type == eModifierType_Surface) {
163 ob->pd->flag &= ~PFIELD_SURFACE;
165 DAG_scene_sort(scene);
167 else if(md->type == eModifierType_Smoke) {
168 ModifierData *tmd = modifiers_findByType(ob, eModifierType_SmokeHR);
170 BLI_remlink(&ob->modifiers, tmd);
175 BLI_remlink(&ob->modifiers, md);
178 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
183 int ED_object_modifier_move_up(ReportList *reports, Object *ob, ModifierData *md)
186 ModifierTypeInfo *mti = modifierType_getInfo(md->type);
188 if(mti->type!=eModifierTypeType_OnlyDeform) {
189 ModifierTypeInfo *nmti = modifierType_getInfo(md->prev->type);
191 if(nmti->flags&eModifierTypeFlag_RequiresOriginalData) {
192 BKE_report(reports, RPT_WARNING, "Cannot move above a modifier requiring original data.");
197 BLI_remlink(&ob->modifiers, md);
198 BLI_insertlink(&ob->modifiers, md->prev->prev, md);
204 int ED_object_modifier_move_down(ReportList *reports, Object *ob, ModifierData *md)
207 ModifierTypeInfo *mti = modifierType_getInfo(md->type);
209 if(mti->flags&eModifierTypeFlag_RequiresOriginalData) {
210 ModifierTypeInfo *nmti = modifierType_getInfo(md->next->type);
212 if(nmti->type!=eModifierTypeType_OnlyDeform) {
213 BKE_report(reports, RPT_WARNING, "Cannot move beyond a non-deforming modifier.");
218 BLI_remlink(&ob->modifiers, md);
219 BLI_insertlink(&ob->modifiers, md->next, md);
225 int ED_object_modifier_convert(ReportList *reports, Scene *scene, Object *ob, ModifierData *md)
228 ParticleSystem *psys;
229 ParticleCacheKey *key, **cache;
230 ParticleSettings *part;
235 int totvert=0, totedge=0, cvert=0;
236 int totpart=0, totchild=0;
238 if(md->type != eModifierType_ParticleSystem) return 0;
239 if(ob && ob->mode & OB_MODE_PARTICLE_EDIT) return 0;
241 psys=((ParticleSystemModifierData *)md)->psys;
244 if(part->ren_as == PART_DRAW_GR || part->ren_as == PART_DRAW_OB) {
245 ; // XXX make_object_duplilist_real(NULL);
248 if(part->ren_as != PART_DRAW_PATH || psys->pathcache == 0)
251 totpart= psys->totcached;
252 totchild= psys->totchildcache;
254 if(totchild && (part->draw&PART_DRAW_PARENT)==0)
258 cache= psys->pathcache;
259 for(a=0; a<totpart; a++) {
261 totvert+= key->steps+1;
262 totedge+= key->steps;
265 cache= psys->childcache;
266 for(a=0; a<totchild; a++) {
268 totvert+= key->steps+1;
269 totedge+= key->steps;
272 if(totvert==0) return 0;
275 obn= add_object(scene, OB_MESH);
278 me->totvert= totvert;
279 me->totedge= totedge;
281 me->mvert= CustomData_add_layer(&me->vdata, CD_MVERT, CD_CALLOC, NULL, totvert);
282 me->medge= CustomData_add_layer(&me->edata, CD_MEDGE, CD_CALLOC, NULL, totedge);
283 me->mface= CustomData_add_layer(&me->fdata, CD_MFACE, CD_CALLOC, NULL, 0);
288 /* copy coordinates */
289 cache= psys->pathcache;
290 for(a=0; a<totpart; a++) {
293 for(k=0; k<=kmax; k++,key++,cvert++,mvert++) {
294 VECCOPY(mvert->co,key->co);
298 medge->flag= ME_EDGEDRAW|ME_EDGERENDER|ME_LOOSEEDGE;
304 cache=psys->childcache;
305 for(a=0; a<totchild; a++) {
308 for(k=0; k<=kmax; k++,key++,cvert++,mvert++) {
309 VECCOPY(mvert->co,key->co);
313 medge->flag= ME_EDGEDRAW|ME_EDGERENDER|ME_LOOSEEDGE;
320 DAG_scene_sort(scene);
325 int ED_object_modifier_apply(ReportList *reports, Scene *scene, Object *ob, ModifierData *md)
332 BKE_report(reports, RPT_ERROR, "Modifiers cannot be applied in editmode");
334 } else if (((ID*) ob->data)->us>1) {
335 BKE_report(reports, RPT_ERROR, "Modifiers cannot be applied to multi-user data");
339 if (md!=ob->modifiers.first)
340 BKE_report(reports, RPT_INFO, "Applied modifier was not first, result may not be as expected.");
342 if (ob->type==OB_MESH) {
344 BKE_report(reports, RPT_ERROR, "Modifier cannot be applied to Mesh with Shape Keys");
348 mesh_pmv_off(ob, me);
350 /* Multires: ensure that recent sculpting is applied */
351 if(md->type == eModifierType_Multires)
352 multires_force_update(ob);
354 dm = mesh_create_derived_for_modifier(scene, ob, md);
356 BKE_report(reports, RPT_ERROR, "Modifier is disabled or returned error, skipping apply");
365 else if (ELEM(ob->type, OB_CURVE, OB_SURF)) {
366 ModifierTypeInfo *mti = modifierType_getInfo(md->type);
367 Curve *cu = ob->data;
369 float (*vertexCos)[3];
371 BKE_report(reports, RPT_INFO, "Applied modifier only changed CV points, not tesselated/bevel vertices");
373 if (!(md->mode&eModifierMode_Realtime) || (mti->isDisabled && mti->isDisabled(md))) {
374 BKE_report(reports, RPT_ERROR, "Modifier is disabled, skipping apply");
378 vertexCos = curve_getVertexCos(cu, &cu->nurb, &numVerts);
379 mti->deformVerts(md, ob, NULL, vertexCos, numVerts, 0, 0);
380 curve_applyVertexCos(cu, &cu->nurb, vertexCos);
384 MEM_freeN(vertexCos);
386 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
389 BKE_report(reports, RPT_ERROR, "Cannot apply modifier for this object type");
394 BLI_remlink(&ob->modifiers, md);
403 int ED_object_modifier_copy(ReportList *reports, Object *ob, ModifierData *md)
407 nmd = modifier_new(md->type);
408 modifier_copyData(md, nmd);
409 BLI_insertlink(&ob->modifiers, md, nmd);
414 /***************************** OPERATORS ****************************/
416 static int modifier_poll(bContext *C)
418 return (CTX_data_pointer_get_type(C, "modifier", &RNA_Modifier).data != NULL);
421 /************************ add modifier operator *********************/
423 static int modifier_add_exec(bContext *C, wmOperator *op)
425 Scene *scene= CTX_data_scene(C);
426 Object *ob = CTX_data_active_object(C);
427 int type= RNA_enum_get(op->ptr, "type");
429 if(!ED_object_modifier_add(op->reports, scene, ob, type))
430 return OPERATOR_CANCELLED;
432 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
434 return OPERATOR_FINISHED;
437 static EnumPropertyItem *modifier_add_itemf(bContext *C, PointerRNA *ptr, int *free)
439 EnumPropertyItem *item= NULL, *md_item;
440 ModifierTypeInfo *mti;
444 if(!C) /* needed for docs */
445 return modifier_type_items;
447 ob= CTX_data_active_object(C);
449 for(a=0; modifier_type_items[a].identifier; a++) {
450 md_item= &modifier_type_items[a];
452 if(md_item->identifier[0]) {
453 mti= modifierType_getInfo(md_item->value);
455 if(mti->flags & eModifierTypeFlag_NoUserAdd)
458 if(!((mti->flags & eModifierTypeFlag_AcceptsCVs) ||
459 (ob->type==OB_MESH && (mti->flags & eModifierTypeFlag_AcceptsMesh))))
463 RNA_enum_item_add(&item, &totitem, md_item);
466 RNA_enum_item_end(&item, &totitem);
473 void OBJECT_OT_modifier_add(wmOperatorType *ot)
478 ot->name= "Add Modifier";
479 ot->description = "Add a modifier to the active object.";
480 ot->idname= "OBJECT_OT_modifier_add";
483 ot->invoke= WM_menu_invoke;
484 ot->exec= modifier_add_exec;
485 ot->poll= ED_operator_object_active;
488 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
491 prop= RNA_def_enum(ot->srna, "type", modifier_type_items, eModifierType_Subsurf, "Type", "");
492 RNA_def_enum_funcs(prop, modifier_add_itemf);
495 /************************ remove modifier operator *********************/
497 static int modifier_remove_exec(bContext *C, wmOperator *op)
499 Scene *scene= CTX_data_scene(C);
500 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_Modifier);
501 Object *ob= ptr.id.data;
502 ModifierData *md= ptr.data;
504 if(!ED_object_modifier_remove(op->reports, scene, ob, md))
505 return OPERATOR_CANCELLED;
507 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
509 return OPERATOR_FINISHED;
512 void OBJECT_OT_modifier_remove(wmOperatorType *ot)
514 ot->name= "Remove Modifier";
515 ot->description= "Remove a modifier from the active object.";
516 ot->idname= "OBJECT_OT_modifier_remove";
517 ot->poll= ED_operator_object_active;
519 ot->exec= modifier_remove_exec;
520 ot->poll= modifier_poll;
523 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
526 /************************ move up modifier operator *********************/
528 static int modifier_move_up_exec(bContext *C, wmOperator *op)
530 Scene *scene= CTX_data_scene(C);
531 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_Modifier);
532 Object *ob= ptr.id.data;
533 ModifierData *md= ptr.data;
535 if(!ED_object_modifier_move_up(op->reports, ob, md))
536 return OPERATOR_CANCELLED;
538 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
539 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
541 return OPERATOR_FINISHED;
544 void OBJECT_OT_modifier_move_up(wmOperatorType *ot)
546 ot->name= "Move Up Modifier";
547 ot->description= "Move modifier up in the stack.";
548 ot->idname= "OBJECT_OT_modifier_move_up";
549 ot->poll= ED_operator_object_active;
551 ot->exec= modifier_move_up_exec;
552 ot->poll= modifier_poll;
555 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
558 /************************ move down modifier operator *********************/
560 static int modifier_move_down_exec(bContext *C, wmOperator *op)
562 Scene *scene= CTX_data_scene(C);
563 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_Modifier);
564 Object *ob= ptr.id.data;
565 ModifierData *md= ptr.data;
567 if(!ob || !md || !ED_object_modifier_move_down(op->reports, ob, md))
568 return OPERATOR_CANCELLED;
570 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
571 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
573 return OPERATOR_FINISHED;
576 void OBJECT_OT_modifier_move_down(wmOperatorType *ot)
578 ot->name= "Move Down Modifier";
579 ot->description= "Move modifier down in the stack.";
580 ot->idname= "OBJECT_OT_modifier_move_down";
581 ot->poll= ED_operator_object_active;
583 ot->exec= modifier_move_down_exec;
584 ot->poll= modifier_poll;
587 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
590 /************************ apply modifier operator *********************/
592 static int modifier_apply_exec(bContext *C, wmOperator *op)
594 Scene *scene= CTX_data_scene(C);
595 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_Modifier);
596 Object *ob= ptr.id.data;
597 ModifierData *md= ptr.data;
599 if(!ob || !md || !ED_object_modifier_apply(op->reports, scene, ob, md))
600 return OPERATOR_CANCELLED;
602 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
603 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
605 return OPERATOR_FINISHED;
608 void OBJECT_OT_modifier_apply(wmOperatorType *ot)
610 ot->name= "Apply Modifier";
611 ot->description= "Apply modifier and remove from the stack.";
612 ot->idname= "OBJECT_OT_modifier_apply";
613 ot->poll= ED_operator_object_active;
615 ot->exec= modifier_apply_exec;
616 ot->poll= modifier_poll;
619 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
622 /************************ convert modifier operator *********************/
624 static int modifier_convert_exec(bContext *C, wmOperator *op)
626 Scene *scene= CTX_data_scene(C);
627 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_Modifier);
628 Object *ob= ptr.id.data;
629 ModifierData *md= ptr.data;
631 if(!ob || !md || !ED_object_modifier_convert(op->reports, scene, ob, md))
632 return OPERATOR_CANCELLED;
634 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
635 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
637 return OPERATOR_FINISHED;
640 void OBJECT_OT_modifier_convert(wmOperatorType *ot)
642 ot->name= "Convert Modifier";
643 ot->description= "Convert particles to a mesh object.";
644 ot->idname= "OBJECT_OT_modifier_convert";
645 ot->poll= ED_operator_object_active;
647 ot->exec= modifier_convert_exec;
648 ot->poll= modifier_poll;
651 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
654 /************************ copy modifier operator *********************/
656 static int modifier_copy_exec(bContext *C, wmOperator *op)
658 Scene *scene= CTX_data_scene(C);
659 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_Modifier);
660 Object *ob= ptr.id.data;
661 ModifierData *md= ptr.data;
663 if(!ob || !md || !ED_object_modifier_copy(op->reports, ob, md))
664 return OPERATOR_CANCELLED;
666 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
667 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
669 return OPERATOR_FINISHED;
672 void OBJECT_OT_modifier_copy(wmOperatorType *ot)
674 ot->name= "Copy Modifier";
675 ot->description= "Duplicate modifier at the same position in the stack.";
676 ot->idname= "OBJECT_OT_modifier_copy";
677 ot->poll= ED_operator_object_active;
679 ot->exec= modifier_copy_exec;
680 ot->poll= modifier_poll;
683 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
686 /****************** multires subdivide operator *********************/
688 static int multires_subdivide_exec(bContext *C, wmOperator *op)
690 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_MultiresModifier);
691 Object *ob= ptr.id.data;
692 MultiresModifierData *mmd= ptr.data;
694 multiresModifier_subdivide(mmd, ob, 1, 0, mmd->simple);
695 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
697 return OPERATOR_FINISHED;
700 static int multires_subdivide_poll(bContext *C)
702 return (CTX_data_pointer_get_type(C, "modifier", &RNA_MultiresModifier).data != NULL) &&
703 CTX_data_edit_object(C) == NULL;
706 void OBJECT_OT_multires_subdivide(wmOperatorType *ot)
708 ot->name= "Multires Subdivide";
709 ot->description= "Add a new level of subdivision.";
710 ot->idname= "OBJECT_OT_multires_subdivide";
712 ot->exec= multires_subdivide_exec;
713 ot->poll= multires_subdivide_poll;
716 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
719 /************************ mdef bind operator *********************/
721 static int meshdeform_poll(bContext *C)
723 return CTX_data_pointer_get_type(C, "modifier", &RNA_MeshDeformModifier).data != NULL;
726 static int meshdeform_bind_exec(bContext *C, wmOperator *op)
728 Scene *scene= CTX_data_scene(C);
729 PointerRNA ptr= CTX_data_pointer_get(C, "modifier");
730 Object *ob= ptr.id.data;
731 MeshDeformModifierData *mmd= ptr.data;
734 if(mmd->bindweights) MEM_freeN(mmd->bindweights);
735 if(mmd->bindcos) MEM_freeN(mmd->bindcos);
736 if(mmd->dyngrid) MEM_freeN(mmd->dyngrid);
737 if(mmd->dyninfluences) MEM_freeN(mmd->dyninfluences);
738 if(mmd->dynverts) MEM_freeN(mmd->dynverts);
739 mmd->bindweights= NULL;
742 mmd->dyninfluences= NULL;
746 mmd->totinfluence= 0;
750 int mode= mmd->modifier.mode;
752 /* force modifier to run, it will call binding routine */
754 mmd->modifier.mode |= eModifierMode_Realtime;
756 if(ob->type == OB_MESH) {
757 dm= mesh_create_derived_view(scene, ob, 0);
760 else if(ob->type == OB_LATTICE) {
761 lattice_calc_modifiers(scene, ob);
763 else if(ob->type==OB_MBALL) {
764 makeDispListMBall(scene, ob);
766 else if(ELEM3(ob->type, OB_CURVE, OB_SURF, OB_FONT)) {
767 makeDispListCurveTypes(scene, ob, 0);
771 mmd->modifier.mode= mode;
774 return OPERATOR_FINISHED;
777 void OBJECT_OT_meshdeform_bind(wmOperatorType *ot)
780 ot->name= "Mesh Deform Bind";
781 ot->description = "Bind mesh to cage in mesh deform modifier.";
782 ot->idname= "OBJECT_OT_meshdeform_bind";
785 ot->poll= meshdeform_poll;
786 ot->exec= meshdeform_bind_exec;
789 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
793 typedef struct MenuEntry {
798 static int menuEntry_compare_names(const void *entry1, const void *entry2)
800 return strcmp(((MenuEntry *)entry1)->name, ((MenuEntry *)entry2)->name);
803 static uiBlock *modifiers_add_menu(void *ob_v)
809 MenuEntry entries[NUM_MODIFIER_TYPES];
811 block= uiNewBlock(&curarea->uiblocks, "modifier_add_menu",
812 UI_EMBOSSP, UI_HELV, curarea->win);
813 uiBlockSetButmFunc(block, modifiers_add, ob);
815 for (i=eModifierType_None+1; i<NUM_MODIFIER_TYPES; i++) {
816 ModifierTypeInfo *mti = modifierType_getInfo(i);
818 /* Only allow adding through appropriate other interfaces */
819 if(ELEM(i, eModifierType_ParticleSystem, eModifierType_Surface)) continue;
821 if((mti->flags&eModifierTypeFlag_AcceptsCVs) ||
822 (ob->type==OB_MESH && (mti->flags&eModifierTypeFlag_AcceptsMesh))) {
823 entries[numEntries].name = mti->name;
824 entries[numEntries].ID = i;
830 qsort(entries, numEntries, sizeof(*entries), menuEntry_compare_names);
833 for(i = 0; i < numEntries; ++i)
834 uiDefBut(block, BUTM, B_MODIFIER_RECALC, entries[i].name,
835 0, yco -= 20, 160, 19, NULL, 0, 0, 1, entries[i].ID, "");
837 uiTextBoundsBlock(block, 50);
838 uiBlockSetDirection(block, UI_DOWN);
844 /******************** hook operators ************************/
846 static int hook_poll(bContext *C)
848 return CTX_data_pointer_get_type(C, "modifier", &RNA_HookModifier).data != NULL;
851 static int hook_reset_exec(bContext *C, wmOperator *op)
853 Scene *scene= CTX_data_scene(C);
854 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_HookModifier);
855 Object *ob= ptr.id.data;
856 HookModifierData *hmd= ptr.data;
859 bPoseChannel *pchan= get_pose_channel(hmd->object->pose, hmd->subtarget);
861 if(hmd->subtarget[0] && pchan) {
862 float imat[4][4], mat[4][4];
864 /* calculate the world-space matrix for the pose-channel target first, then carry on as usual */
865 Mat4MulMat4(mat, pchan->pose_mat, hmd->object->obmat);
867 Mat4Invert(imat, mat);
868 Mat4MulSerie(hmd->parentinv, imat, mat, NULL, NULL, NULL, NULL, NULL, NULL);
871 Mat4Invert(hmd->object->imat, hmd->object->obmat);
872 Mat4MulSerie(hmd->parentinv, hmd->object->imat, ob->obmat, NULL, NULL, NULL, NULL, NULL, NULL);
876 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
877 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
879 return OPERATOR_FINISHED;
882 void OBJECT_OT_hook_reset(wmOperatorType *ot)
884 ot->name= "Hook Reset";
885 ot->description= "Recalculate and and clear offset transformation.";
886 ot->idname= "OBJECT_OT_hook_reset";
888 ot->exec= hook_reset_exec;
892 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
895 static int hook_recenter_exec(bContext *C, wmOperator *op)
897 Scene *scene= CTX_data_scene(C);
898 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_HookModifier);
899 Object *ob= ptr.id.data;
900 HookModifierData *hmd= ptr.data;
901 float bmat[3][3], imat[3][3];
903 Mat3CpyMat4(bmat, ob->obmat);
906 VECSUB(hmd->cent, scene->cursor, ob->obmat[3]);
907 Mat3MulVecfl(imat, hmd->cent);
909 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
910 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
912 return OPERATOR_FINISHED;
915 void OBJECT_OT_hook_recenter(wmOperatorType *ot)
917 ot->name= "Hook Recenter";
918 ot->description= "Set hook center to cursor position.";
919 ot->idname= "OBJECT_OT_hook_recenter";
921 ot->exec= hook_recenter_exec;
925 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
928 static int hook_select_exec(bContext *C, wmOperator *op)
930 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_HookModifier);
931 Object *ob= ptr.id.data;
932 HookModifierData *hmd= ptr.data;
934 object_hook_select(ob, hmd);
936 WM_event_add_notifier(C, NC_OBJECT|ND_GEOM_SELECT, ob);
938 return OPERATOR_FINISHED;
941 void OBJECT_OT_hook_select(wmOperatorType *ot)
943 ot->name= "Hook Select";
944 ot->description= "Selects effected vertices on mesh.";
945 ot->idname= "OBJECT_OT_hook_select";
947 ot->exec= hook_select_exec;
951 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
954 static int hook_assign_exec(bContext *C, wmOperator *op)
956 Scene *scene= CTX_data_scene(C);
957 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_HookModifier);
958 Object *ob= ptr.id.data;
959 HookModifierData *hmd= ptr.data;
964 if(!object_hook_index_array(ob, &tot, &indexar, name, cent)) {
965 BKE_report(op->reports, RPT_WARNING, "Requires selected vertices or active vertex group");
966 return OPERATOR_CANCELLED;
970 MEM_freeN(hmd->indexar);
972 VECCOPY(hmd->cent, cent);
973 hmd->indexar= indexar;
976 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
977 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
979 return OPERATOR_FINISHED;
982 void OBJECT_OT_hook_assign(wmOperatorType *ot)
984 ot->name= "Hook Assign";
985 ot->description= "Reassigns selected vertices to hook.";
986 ot->idname= "OBJECT_OT_hook_assign";
988 ot->exec= hook_assign_exec;
992 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;
995 /****************** explode refresh operator *********************/
997 static int explode_refresh_poll(bContext *C)
999 return CTX_data_pointer_get_type(C, "modifier", &RNA_ExplodeModifier).data != NULL;
1002 static int explode_refresh_exec(bContext *C, wmOperator *op)
1004 Scene *scene= CTX_data_scene(C);
1005 PointerRNA ptr= CTX_data_pointer_get_type(C, "modifier", &RNA_ExplodeModifier);
1006 Object *ob= ptr.id.data;
1007 ExplodeModifierData *emd= ptr.data;
1009 emd->flag |= eExplodeFlag_CalcFaces;
1011 DAG_object_flush_update(scene, ob, OB_RECALC_DATA);
1012 WM_event_add_notifier(C, NC_OBJECT|ND_MODIFIER, ob);
1014 return OPERATOR_FINISHED;
1017 void OBJECT_OT_explode_refresh(wmOperatorType *ot)
1019 ot->name= "Explode Refresh";
1020 ot->description= "Refresh data in the Explode modifier.";
1021 ot->idname= "OBJECT_OT_explode_refresh";
1023 ot->exec= explode_refresh_exec;
1024 ot->poll= explode_refresh_poll;
1027 ot->flag= OPTYPE_REGISTER|OPTYPE_UNDO;