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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
20 * The Original Code is Copyright (C) 2004 Blender Foundation
21 * All rights reserved.
23 * The Original Code is: all of this file.
25 * Contributor(s): none yet.
27 * ***** END GPL LICENSE BLOCK *****
35 #include "MEM_guardedalloc.h"
37 #include "DNA_object_types.h"
39 #include "BKE_blender.h"
40 #include "BKE_context.h"
41 #include "BKE_global.h"
42 #include "BKE_screen.h"
44 #include "BLI_blenlib.h"
45 #include "BLI_editVert.h"
46 #include "BLI_dynstr.h"
49 #include "ED_armature.h"
50 #include "ED_particle.h"
54 #include "ED_object.h"
55 #include "ED_screen.h"
56 #include "ED_sculpt.h"
64 #include "util_intern.h"
66 /* ***************** generic undo system ********************* */
68 void ED_undo_push(bContext *C, const char *str)
70 wmWindowManager *wm= CTX_wm_manager(C);
71 Object *obedit= CTX_data_edit_object(C);
72 Object *obact= CTX_data_active_object(C);
75 if (U.undosteps == 0) return;
77 if(obedit->type==OB_MESH)
78 undo_push_mesh(C, str);
79 else if ELEM(obedit->type, OB_CURVE, OB_SURF)
80 undo_push_curve(C, str);
81 else if (obedit->type==OB_FONT)
82 undo_push_font(C, str);
83 else if (obedit->type==OB_MBALL)
84 undo_push_mball(C, str);
85 else if (obedit->type==OB_LATTICE)
86 undo_push_lattice(C, str);
87 else if (obedit->type==OB_ARMATURE)
88 undo_push_armature(C, str);
90 else if(obact && obact->mode & OB_MODE_PARTICLE_EDIT) {
91 if (U.undosteps == 0) return;
93 PE_undo_push(CTX_data_scene(C), str);
96 if(U.uiflag & USER_GLOBALUNDO)
97 BKE_write_undo(C, str);
102 WM_event_add_notifier(C, NC_WM|ND_DATACHANGED, NULL);
106 static int ed_undo_step(bContext *C, int step, const char *undoname)
108 Object *obedit= CTX_data_edit_object(C);
109 Object *obact= CTX_data_active_object(C);
110 ScrArea *sa= CTX_wm_area(C);
112 if(sa && sa->spacetype==SPACE_IMAGE) {
113 SpaceImage *sima= (SpaceImage *)sa->spacedata.first;
115 if((obact && obact->mode & OB_MODE_TEXTURE_PAINT) || sima->flag & SI_DRAWTOOL) {
116 if(!ED_undo_paint_step(C, UNDO_PAINT_IMAGE, step, undoname) && undoname)
117 if(U.uiflag & USER_GLOBALUNDO)
118 BKE_undo_name(C, undoname);
120 WM_event_add_notifier(C, NC_WINDOW, NULL);
121 return OPERATOR_FINISHED;
125 if(sa && sa->spacetype==SPACE_TEXT) {
126 ED_text_undo_step(C, step);
129 if ELEM7(obedit->type, OB_MESH, OB_FONT, OB_CURVE, OB_SURF, OB_MBALL, OB_LATTICE, OB_ARMATURE) {
131 undo_editmode_name(C, undoname);
133 undo_editmode_step(C, step);
135 WM_event_add_notifier(C, NC_GEOM|ND_DATA, NULL);
141 if(obact && obact->mode & OB_MODE_TEXTURE_PAINT) {
142 if(!ED_undo_paint_step(C, UNDO_PAINT_IMAGE, step, undoname) && undoname)
145 else if(obact && obact->mode & OB_MODE_SCULPT) {
146 if(!ED_undo_paint_step(C, UNDO_PAINT_MESH, step, undoname) && undoname)
149 else if(obact && obact->mode & OB_MODE_PARTICLE_EDIT) {
151 PE_undo(CTX_data_scene(C));
153 PE_redo(CTX_data_scene(C));
160 if(U.uiflag & USER_GLOBALUNDO) {
161 // note python defines not valid here anymore.
163 // XXX BPY_scripts_clear_pyobjects();
166 BKE_undo_name(C, undoname);
168 BKE_undo_step(C, step);
170 WM_event_add_notifier(C, NC_SCENE|ND_LAYER_CONTENT, CTX_data_scene(C));
176 WM_event_add_notifier(C, NC_WINDOW, NULL);
178 return OPERATOR_FINISHED;
181 void ED_undo_pop(bContext *C)
183 ed_undo_step(C, 1, NULL);
185 void ED_undo_redo(bContext *C)
187 ed_undo_step(C, -1, NULL);
190 void ED_undo_push_op(bContext *C, wmOperator *op)
192 /* in future, get undo string info? */
193 ED_undo_push(C, op->type->name);
196 void ED_undo_pop_op(bContext *C, wmOperator *op)
198 /* search back a couple of undo's, in case something else added pushes */
199 ed_undo_step(C, 0, op->type->name);
202 static int ed_undo_exec(bContext *C, wmOperator *UNUSED(op))
204 /* "last operator" should disappear, later we can tie ths with undo stack nicer */
205 WM_operator_stack_clear(C);
206 return ed_undo_step(C, 1, NULL);
209 static int ed_redo_exec(bContext *C, wmOperator *UNUSED(op))
211 return ed_undo_step(C, -1, NULL);
214 void ED_undo_menu(bContext *C)
216 Object *obedit= CTX_data_edit_object(C);
217 Object *obact= CTX_data_active_object(C);
220 //if ELEM7(obedit->type, OB_MESH, OB_FONT, OB_CURVE, OB_SURF, OB_MBALL, OB_LATTICE, OB_ARMATURE)
221 // undo_editmode_menu();
224 if(obact && obact->mode & OB_MODE_PARTICLE_EDIT)
225 PE_undo_menu(CTX_data_scene(C), CTX_data_active_object(C));
226 else if(U.uiflag & USER_GLOBALUNDO) {
227 char *menu= BKE_undo_menu_string();
229 short event= 0; // XXX pupmenu_col(menu, 20);
232 BKE_undo_number(C, event);
239 /* ********************** */
241 void ED_OT_undo(wmOperatorType *ot)
245 ot->description= "Undo previous action";
246 ot->idname= "ED_OT_undo";
249 ot->exec= ed_undo_exec;
250 ot->poll= ED_operator_screenactive;
253 void ED_OT_redo(wmOperatorType *ot)
257 ot->description= "Redo previous action";
258 ot->idname= "ED_OT_redo";
261 ot->exec= ed_redo_exec;
262 ot->poll= ED_operator_screenactive;
266 /* ui callbacks should call this rather then calling WM_operator_repeat() themselves */
267 int ED_undo_operator_repeat(bContext *C, struct wmOperator *op)
272 ARegion *ar= CTX_wm_region(C);
273 ARegion *ar1= BKE_area_find_region_type(CTX_wm_area(C), RGN_TYPE_WINDOW);
276 CTX_wm_region_set(C, ar1);
278 if(WM_operator_repeat_check(C, op) && WM_operator_poll(C, op->type)) {
282 printf("redo_cb: operator redo %s\n", op->type->name);
283 ED_undo_pop_op(C, op);
284 retval= WM_operator_repeat(C, op);
285 if((retval & OPERATOR_FINISHED)==0) {
287 printf("redo_cb: operator redo failed: %s, return %d\n", op->type->name, retval);
295 /* set region back */
296 CTX_wm_region_set(C, ar);
300 printf("redo_cb: WM_operator_repeat_check returned false %s\n", op->type->name);
307 void ED_undo_operator_repeat_cb(bContext *C, void *arg_op, void *UNUSED(arg_unused))
309 ED_undo_operator_repeat(C, (wmOperator *)arg_op);
312 void ED_undo_operator_repeat_cb_evt(bContext *C, void *arg_op, int UNUSED(arg_event))
314 ED_undo_operator_repeat(C, (wmOperator *)arg_op);