Cleanup: unused region init functions
[blender.git] / source / blender / editors / space_action / space_action.c
1 /*
2  * This program is free software; you can redistribute it and/or
3  * modify it under the terms of the GNU General Public License
4  * as published by the Free Software Foundation; either version 2
5  * of the License, or (at your option) any later version.
6  *
7  * This program is distributed in the hope that it will be useful,
8  * but WITHOUT ANY WARRANTY; without even the implied warranty of
9  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10  * GNU General Public License for more details.
11  *
12  * You should have received a copy of the GNU General Public License
13  * along with this program; if not, write to the Free Software Foundation,
14  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
15  *
16  * The Original Code is Copyright (C) 2008 Blender Foundation.
17  * All rights reserved.
18  */
19
20 /** \file
21  * \ingroup spaction
22  */
23
24 #include <string.h>
25 #include <stdio.h>
26
27 #include "DNA_action_types.h"
28 #include "DNA_collection_types.h"
29 #include "DNA_object_types.h"
30 #include "DNA_scene_types.h"
31
32 #include "MEM_guardedalloc.h"
33
34 #include "BLI_blenlib.h"
35 #include "BLI_utildefines.h"
36
37 #include "BKE_context.h"
38 #include "BKE_screen.h"
39
40 #include "RNA_access.h"
41 #include "RNA_define.h"
42 #include "RNA_enum_types.h"
43
44 #include "WM_api.h"
45 #include "WM_types.h"
46 #include "WM_message.h"
47
48 #include "UI_resources.h"
49 #include "UI_view2d.h"
50
51 #include "ED_space_api.h"
52 #include "ED_screen.h"
53 #include "ED_anim_api.h"
54 #include "ED_markers.h"
55
56 #include "action_intern.h" /* own include */
57 #include "GPU_framebuffer.h"
58
59 /* ******************** default callbacks for action space ***************** */
60
61 static SpaceLink *action_new(const ScrArea *sa, const Scene *scene)
62 {
63   SpaceAction *saction;
64   ARegion *ar;
65
66   saction = MEM_callocN(sizeof(SpaceAction), "initaction");
67   saction->spacetype = SPACE_ACTION;
68
69   saction->autosnap = SACTSNAP_FRAME;
70   saction->mode = SACTCONT_DOPESHEET;
71   saction->mode_prev = SACTCONT_DOPESHEET;
72   saction->flag = SACTION_SHOW_INTERPOLATION;
73
74   saction->ads.filterflag |= ADS_FILTER_SUMMARY;
75
76   /* enable all cache display */
77   saction->cache_display |= TIME_CACHE_DISPLAY;
78   saction->cache_display |= (TIME_CACHE_SOFTBODY | TIME_CACHE_PARTICLES);
79   saction->cache_display |= (TIME_CACHE_CLOTH | TIME_CACHE_SMOKE | TIME_CACHE_DYNAMICPAINT);
80   saction->cache_display |= TIME_CACHE_RIGIDBODY;
81
82   /* header */
83   ar = MEM_callocN(sizeof(ARegion), "header for action");
84
85   BLI_addtail(&saction->regionbase, ar);
86   ar->regiontype = RGN_TYPE_HEADER;
87   ar->alignment = (U.uiflag & USER_HEADER_BOTTOM) ? RGN_ALIGN_BOTTOM : RGN_ALIGN_TOP;
88
89   /* channel list region */
90   ar = MEM_callocN(sizeof(ARegion), "channel region for action");
91   BLI_addtail(&saction->regionbase, ar);
92   ar->regiontype = RGN_TYPE_CHANNELS;
93   ar->alignment = RGN_ALIGN_LEFT;
94
95   /* only need to set scroll settings, as this will use 'listview' v2d configuration */
96   ar->v2d.scroll = V2D_SCROLL_BOTTOM;
97   ar->v2d.flag = V2D_VIEWSYNC_AREA_VERTICAL;
98
99   /* ui buttons */
100   ar = MEM_callocN(sizeof(ARegion), "buttons region for action");
101
102   BLI_addtail(&saction->regionbase, ar);
103   ar->regiontype = RGN_TYPE_UI;
104   ar->alignment = RGN_ALIGN_RIGHT;
105   ar->flag = RGN_FLAG_HIDDEN;
106
107   /* main region */
108   ar = MEM_callocN(sizeof(ARegion), "main region for action");
109
110   BLI_addtail(&saction->regionbase, ar);
111   ar->regiontype = RGN_TYPE_WINDOW;
112
113   ar->v2d.tot.xmin = (float)(SFRA - 10);
114   ar->v2d.tot.ymin = (float)(-sa->winy) / 3.0f;
115   ar->v2d.tot.xmax = (float)(EFRA + 10);
116   ar->v2d.tot.ymax = 0.0f;
117
118   ar->v2d.cur = ar->v2d.tot;
119
120   ar->v2d.min[0] = 0.0f;
121   ar->v2d.min[1] = 0.0f;
122
123   ar->v2d.max[0] = MAXFRAMEF;
124   ar->v2d.max[1] = FLT_MAX;
125
126   ar->v2d.minzoom = 0.01f;
127   ar->v2d.maxzoom = 50;
128   ar->v2d.scroll = (V2D_SCROLL_BOTTOM | V2D_SCROLL_SCALE_HORIZONTAL);
129   ar->v2d.scroll |= (V2D_SCROLL_RIGHT);
130   ar->v2d.keepzoom = V2D_LOCKZOOM_Y;
131   ar->v2d.keepofs = V2D_KEEPOFS_Y;
132   ar->v2d.align = V2D_ALIGN_NO_POS_Y;
133   ar->v2d.flag = V2D_VIEWSYNC_AREA_VERTICAL;
134
135   return (SpaceLink *)saction;
136 }
137
138 /* not spacelink itself */
139 static void action_free(SpaceLink *UNUSED(sl))
140 {
141   //  SpaceAction *saction = (SpaceAction *) sl;
142 }
143
144 /* spacetype; init callback */
145 static void action_init(struct wmWindowManager *UNUSED(wm), ScrArea *sa)
146 {
147   SpaceAction *saction = sa->spacedata.first;
148   saction->runtime.flag |= SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC;
149 }
150
151 static SpaceLink *action_duplicate(SpaceLink *sl)
152 {
153   SpaceAction *sactionn = MEM_dupallocN(sl);
154
155   /* clear or remove stuff from old */
156
157   return (SpaceLink *)sactionn;
158 }
159
160 /* add handlers, stuff you only do once or on area/region changes */
161 static void action_main_region_init(wmWindowManager *wm, ARegion *ar)
162 {
163   wmKeyMap *keymap;
164
165   UI_view2d_region_reinit(&ar->v2d, V2D_COMMONVIEW_CUSTOM, ar->winx, ar->winy);
166
167   /* own keymap */
168   keymap = WM_keymap_ensure(wm->defaultconf, "Dopesheet", SPACE_ACTION, 0);
169   WM_event_add_keymap_handler_bb(&ar->handlers, keymap, &ar->v2d.mask, &ar->winrct);
170   keymap = WM_keymap_ensure(wm->defaultconf, "Dopesheet Generic", SPACE_ACTION, 0);
171   WM_event_add_keymap_handler(&ar->handlers, keymap);
172 }
173
174 static void action_main_region_draw(const bContext *C, ARegion *ar)
175 {
176   /* draw entirely, view changes should be handled here */
177   SpaceAction *saction = CTX_wm_space_action(C);
178   Scene *scene = CTX_data_scene(C);
179   Object *obact = CTX_data_active_object(C);
180   bAnimContext ac;
181   View2D *v2d = &ar->v2d;
182   View2DGrid *grid;
183   View2DScrollers *scrollers;
184   short marker_flag = 0;
185   short cfra_flag = 0;
186   short unit = 0;
187
188   /* clear and setup matrix */
189   UI_ThemeClearColor(TH_BACK);
190   GPU_clear(GPU_COLOR_BIT);
191
192   UI_view2d_view_ortho(v2d);
193
194   /* time grid */
195   unit = (saction->flag & SACTION_DRAWTIME) ? V2D_UNIT_SECONDS : V2D_UNIT_FRAMES;
196   grid = UI_view2d_grid_calc(CTX_data_scene(C),
197                              v2d,
198                              unit,
199                              V2D_GRID_CLAMP,
200                              V2D_ARG_DUMMY,
201                              V2D_ARG_DUMMY,
202                              ar->winx,
203                              ar->winy);
204   UI_view2d_grid_draw(v2d, grid, V2D_GRIDLINES_ALL);
205   UI_view2d_grid_free(grid);
206
207   ED_region_draw_cb_draw(C, ar, REGION_DRAW_PRE_VIEW);
208
209   /* start and end frame */
210   ANIM_draw_framerange(scene, v2d);
211
212   /* data */
213   if (ANIM_animdata_get_context(C, &ac)) {
214     draw_channel_strips(&ac, saction, ar);
215   }
216
217   /* current frame */
218   if (saction->flag & SACTION_DRAWTIME)
219     cfra_flag |= DRAWCFRA_UNIT_SECONDS;
220   ANIM_draw_cfra(C, v2d, cfra_flag);
221
222   /* markers */
223   UI_view2d_view_orthoSpecial(ar, v2d, 1);
224
225   marker_flag = ((ac.markers && (ac.markers != &ac.scene->markers)) ? DRAW_MARKERS_LOCAL : 0) |
226                 DRAW_MARKERS_MARGIN;
227   if (saction->flag & SACTION_SHOW_MARKER_LINES)
228     marker_flag |= DRAW_MARKERS_LINES;
229   ED_markers_draw(C, marker_flag);
230
231   /* caches */
232   if (saction->mode == SACTCONT_TIMELINE) {
233     timeline_draw_cache(saction, obact, scene);
234   }
235
236   /* preview range */
237   UI_view2d_view_ortho(v2d);
238   ANIM_draw_previewrange(C, v2d, 0);
239
240   /* callback */
241   UI_view2d_view_ortho(v2d);
242   ED_region_draw_cb_draw(C, ar, REGION_DRAW_POST_VIEW);
243
244   /* reset view matrix */
245   UI_view2d_view_restore(C);
246
247   /* scrollers */
248   scrollers = UI_view2d_scrollers_calc(
249       C, v2d, NULL, unit, V2D_GRID_CLAMP, V2D_ARG_DUMMY, V2D_ARG_DUMMY);
250   UI_view2d_scrollers_draw(C, v2d, scrollers);
251   UI_view2d_scrollers_free(scrollers);
252
253   /* draw current frame number-indicator on top of scrollers */
254   if ((saction->flag & SACTION_NODRAWCFRANUM) == 0) {
255     UI_view2d_view_orthoSpecial(ar, v2d, 1);
256     ANIM_draw_cfra_number(C, v2d, cfra_flag);
257   }
258 }
259
260 /* add handlers, stuff you only do once or on area/region changes */
261 static void action_channel_region_init(wmWindowManager *wm, ARegion *ar)
262 {
263   wmKeyMap *keymap;
264
265   /* ensure the 2d view sync works - main region has bottom scroller */
266   ar->v2d.scroll = V2D_SCROLL_BOTTOM;
267
268   UI_view2d_region_reinit(&ar->v2d, V2D_COMMONVIEW_LIST, ar->winx, ar->winy);
269
270   /* own keymap */
271   keymap = WM_keymap_ensure(wm->defaultconf, "Animation Channels", 0, 0);
272   WM_event_add_keymap_handler_bb(&ar->handlers, keymap, &ar->v2d.mask, &ar->winrct);
273
274   keymap = WM_keymap_ensure(wm->defaultconf, "Dopesheet Generic", SPACE_ACTION, 0);
275   WM_event_add_keymap_handler(&ar->handlers, keymap);
276 }
277
278 static void action_channel_region_draw(const bContext *C, ARegion *ar)
279 {
280   /* draw entirely, view changes should be handled here */
281   bAnimContext ac;
282   View2D *v2d = &ar->v2d;
283
284   /* clear and setup matrix */
285   UI_ThemeClearColor(TH_BACK);
286   GPU_clear(GPU_COLOR_BIT);
287
288   UI_view2d_view_ortho(v2d);
289
290   /* data */
291   if (ANIM_animdata_get_context(C, &ac)) {
292     draw_channel_names((bContext *)C, &ac, ar);
293   }
294
295   /* reset view matrix */
296   UI_view2d_view_restore(C);
297
298   /* no scrollers here */
299 }
300
301 /* add handlers, stuff you only do once or on area/region changes */
302 static void action_header_region_init(wmWindowManager *UNUSED(wm), ARegion *ar)
303 {
304   ED_region_header_init(ar);
305 }
306
307 static void action_header_region_draw(const bContext *C, ARegion *ar)
308 {
309   ED_region_header(C, ar);
310 }
311
312 static void action_channel_region_listener(wmWindow *UNUSED(win),
313                                            ScrArea *UNUSED(sa),
314                                            ARegion *ar,
315                                            wmNotifier *wmn,
316                                            const Scene *UNUSED(scene))
317 {
318   /* context changes */
319   switch (wmn->category) {
320     case NC_ANIMATION:
321       ED_region_tag_redraw(ar);
322       break;
323     case NC_SCENE:
324       switch (wmn->data) {
325         case ND_OB_ACTIVE:
326         case ND_FRAME:
327           ED_region_tag_redraw(ar);
328           break;
329       }
330       break;
331     case NC_OBJECT:
332       switch (wmn->data) {
333         case ND_BONE_ACTIVE:
334         case ND_BONE_SELECT:
335         case ND_KEYS:
336           ED_region_tag_redraw(ar);
337           break;
338         case ND_MODIFIER:
339           if (wmn->action == NA_RENAME)
340             ED_region_tag_redraw(ar);
341           break;
342       }
343       break;
344     case NC_GPENCIL:
345       if (ELEM(wmn->action, NA_RENAME, NA_SELECTED))
346         ED_region_tag_redraw(ar);
347       break;
348     case NC_ID:
349       if (wmn->action == NA_RENAME)
350         ED_region_tag_redraw(ar);
351       break;
352     default:
353       if (wmn->data == ND_KEYS)
354         ED_region_tag_redraw(ar);
355       break;
356   }
357 }
358
359 static void saction_channel_region_message_subscribe(const struct bContext *UNUSED(C),
360                                                      struct WorkSpace *UNUSED(workspace),
361                                                      struct Scene *UNUSED(scene),
362                                                      struct bScreen *screen,
363                                                      struct ScrArea *sa,
364                                                      struct ARegion *ar,
365                                                      struct wmMsgBus *mbus)
366 {
367   PointerRNA ptr;
368   RNA_pointer_create(&screen->id, &RNA_SpaceDopeSheetEditor, sa->spacedata.first, &ptr);
369
370   wmMsgSubscribeValue msg_sub_value_region_tag_redraw = {
371       .owner = ar,
372       .user_data = ar,
373       .notify = ED_region_do_msg_notify_tag_redraw,
374   };
375
376   /* All dopesheet filter settings, etc. affect the drawing of this editor,
377    * also same applies for all animation-related datatypes that may appear here,
378    * so just whitelist the entire structs for updates
379    */
380   {
381     wmMsgParams_RNA msg_key_params = {{{0}}};
382     StructRNA *type_array[] = {
383         &RNA_DopeSheet, /* dopesheet filters */
384
385         &RNA_ActionGroup, /* channel groups */
386
387         &RNA_FCurve, /* F-Curve */
388         &RNA_Keyframe,
389         &RNA_FCurveSample,
390
391         &RNA_GreasePencil, /* Grease Pencil */
392         &RNA_GPencilLayer,
393         &RNA_GPencilFrame,
394     };
395
396     for (int i = 0; i < ARRAY_SIZE(type_array); i++) {
397       msg_key_params.ptr.type = type_array[i];
398       WM_msg_subscribe_rna_params(
399           mbus, &msg_key_params, &msg_sub_value_region_tag_redraw, __func__);
400     }
401   }
402 }
403
404 static void action_main_region_listener(wmWindow *UNUSED(win),
405                                         ScrArea *UNUSED(sa),
406                                         ARegion *ar,
407                                         wmNotifier *wmn,
408                                         const Scene *UNUSED(scene))
409 {
410   /* context changes */
411   switch (wmn->category) {
412     case NC_ANIMATION:
413       ED_region_tag_redraw(ar);
414       break;
415     case NC_SCENE:
416       switch (wmn->data) {
417         case ND_RENDER_OPTIONS:
418         case ND_OB_ACTIVE:
419         case ND_FRAME:
420         case ND_FRAME_RANGE:
421         case ND_MARKERS:
422           ED_region_tag_redraw(ar);
423           break;
424       }
425       break;
426     case NC_OBJECT:
427       switch (wmn->data) {
428         case ND_TRANSFORM:
429           /* moving object shouldn't need to redraw action */
430           break;
431         case ND_BONE_ACTIVE:
432         case ND_BONE_SELECT:
433         case ND_KEYS:
434           ED_region_tag_redraw(ar);
435           break;
436       }
437       break;
438     case NC_NODE:
439       switch (wmn->action) {
440         case NA_EDITED:
441           ED_region_tag_redraw(ar);
442           break;
443       }
444       break;
445     case NC_ID:
446       if (wmn->action == NA_RENAME)
447         ED_region_tag_redraw(ar);
448       break;
449     case NC_SCREEN:
450       if (ELEM(wmn->data, ND_LAYER)) {
451         ED_region_tag_redraw(ar);
452       }
453       break;
454     default:
455       if (wmn->data == ND_KEYS)
456         ED_region_tag_redraw(ar);
457       break;
458   }
459 }
460
461 static void saction_main_region_message_subscribe(const struct bContext *C,
462                                                   struct WorkSpace *workspace,
463                                                   struct Scene *scene,
464                                                   struct bScreen *screen,
465                                                   struct ScrArea *sa,
466                                                   struct ARegion *ar,
467                                                   struct wmMsgBus *mbus)
468 {
469   PointerRNA ptr;
470   RNA_pointer_create(&screen->id, &RNA_SpaceDopeSheetEditor, sa->spacedata.first, &ptr);
471
472   wmMsgSubscribeValue msg_sub_value_region_tag_redraw = {
473       .owner = ar,
474       .user_data = ar,
475       .notify = ED_region_do_msg_notify_tag_redraw,
476   };
477
478   /* Timeline depends on scene properties. */
479   {
480     bool use_preview = (scene->r.flag & SCER_PRV_RANGE);
481     extern PropertyRNA rna_Scene_frame_start;
482     extern PropertyRNA rna_Scene_frame_end;
483     extern PropertyRNA rna_Scene_frame_preview_start;
484     extern PropertyRNA rna_Scene_frame_preview_end;
485     extern PropertyRNA rna_Scene_use_preview_range;
486     extern PropertyRNA rna_Scene_frame_current;
487     const PropertyRNA *props[] = {
488         use_preview ? &rna_Scene_frame_preview_start : &rna_Scene_frame_start,
489         use_preview ? &rna_Scene_frame_preview_end : &rna_Scene_frame_end,
490         &rna_Scene_use_preview_range,
491         &rna_Scene_frame_current,
492     };
493
494     PointerRNA idptr;
495     RNA_id_pointer_create(&scene->id, &idptr);
496
497     for (int i = 0; i < ARRAY_SIZE(props); i++) {
498       WM_msg_subscribe_rna(mbus, &idptr, props[i], &msg_sub_value_region_tag_redraw, __func__);
499     }
500   }
501
502   /* Now run the general "channels region" one - since channels and main should be in sync */
503   saction_channel_region_message_subscribe(C, workspace, scene, screen, sa, ar, mbus);
504 }
505
506 /* editor level listener */
507 static void action_listener(wmWindow *UNUSED(win),
508                             ScrArea *sa,
509                             wmNotifier *wmn,
510                             Scene *UNUSED(scene))
511 {
512   SpaceAction *saction = (SpaceAction *)sa->spacedata.first;
513
514   /* context changes */
515   switch (wmn->category) {
516     case NC_GPENCIL:
517       /* only handle these events in GPencil mode for performance considerations */
518       if (saction->mode == SACTCONT_GPENCIL) {
519         if (wmn->action == NA_EDITED) {
520           ED_area_tag_redraw(sa);
521         }
522         else if (wmn->action == NA_SELECTED) {
523           saction->runtime.flag |= SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC;
524           ED_area_tag_refresh(sa);
525         }
526       }
527       break;
528     case NC_ANIMATION:
529       /* for NLA tweakmode enter/exit, need complete refresh */
530       if (wmn->data == ND_NLA_ACTCHANGE) {
531         saction->runtime.flag |= SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC;
532         ED_area_tag_refresh(sa);
533       }
534       /* autocolor only really needs to change when channels are added/removed,
535        * or previously hidden stuff appears
536        * (assume for now that if just adding these works, that will be fine)
537        */
538       else if (((wmn->data == ND_KEYFRAME) && ELEM(wmn->action, NA_ADDED, NA_REMOVED)) ||
539                ((wmn->data == ND_ANIMCHAN) && (wmn->action != NA_SELECTED))) {
540         ED_area_tag_refresh(sa);
541       }
542       /* for simple edits to the curve data though (or just plain selections),
543        * a simple redraw should work
544        * (see T39851 for an example of how this can go wrong)
545        */
546       else {
547         ED_area_tag_redraw(sa);
548       }
549       break;
550     case NC_SCENE:
551       switch (wmn->data) {
552         case ND_OB_ACTIVE:
553         case ND_OB_SELECT:
554           /* Selection changed, so force refresh to flush
555            * (needs flag set to do syncing). */
556           saction->runtime.flag |= SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC;
557           ED_area_tag_refresh(sa);
558           break;
559         case ND_RENDER_RESULT:
560           ED_area_tag_redraw(sa);
561           break;
562         case ND_FRAME_RANGE:
563           for (ARegion *ar = sa->regionbase.first; ar; ar = ar->next) {
564             if (ar->regiontype == RGN_TYPE_WINDOW) {
565               Scene *scene = wmn->reference;
566               ar->v2d.tot.xmin = (float)(SFRA - 4);
567               ar->v2d.tot.xmax = (float)(EFRA + 4);
568               break;
569             }
570           }
571           break;
572         default:
573           if (saction->mode != SACTCONT_TIMELINE) {
574             /* Just redrawing the view will do. */
575             ED_area_tag_redraw(sa);
576           }
577           break;
578       }
579       break;
580     case NC_OBJECT:
581       switch (wmn->data) {
582         case ND_BONE_SELECT: /* Selection changed, so force refresh to flush
583                               * (needs flag set to do syncing). */
584         case ND_BONE_ACTIVE:
585           saction->runtime.flag |= SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC;
586           ED_area_tag_refresh(sa);
587           break;
588         case ND_TRANSFORM:
589           /* moving object shouldn't need to redraw action */
590           break;
591         case ND_POINTCACHE:
592         case ND_MODIFIER:
593         case ND_PARTICLE:
594           /* only needed in timeline mode */
595           if (saction->mode == SACTCONT_TIMELINE) {
596             ED_area_tag_refresh(sa);
597             ED_area_tag_redraw(sa);
598           }
599           break;
600         default: /* just redrawing the view will do */
601           ED_area_tag_redraw(sa);
602           break;
603       }
604       break;
605     case NC_MASK:
606       if (saction->mode == SACTCONT_MASK) {
607         switch (wmn->data) {
608           case ND_DATA:
609             ED_area_tag_refresh(sa);
610             ED_area_tag_redraw(sa);
611             break;
612           default: /* just redrawing the view will do */
613             ED_area_tag_redraw(sa);
614             break;
615         }
616       }
617       break;
618     case NC_NODE:
619       if (wmn->action == NA_SELECTED) {
620         /* selection changed, so force refresh to flush (needs flag set to do syncing) */
621         saction->runtime.flag |= SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC;
622         ED_area_tag_refresh(sa);
623       }
624       break;
625     case NC_SPACE:
626       switch (wmn->data) {
627         case ND_SPACE_DOPESHEET:
628           ED_area_tag_redraw(sa);
629           break;
630         case ND_SPACE_TIME:
631           ED_area_tag_redraw(sa);
632           break;
633         case ND_SPACE_CHANGED:
634           saction->runtime.flag |= SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC;
635           ED_area_tag_refresh(sa);
636           break;
637       }
638       break;
639     case NC_WINDOW:
640       if (saction->runtime.flag & SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC) {
641         /* force redraw/refresh after undo/redo - [#28962] */
642         ED_area_tag_refresh(sa);
643       }
644       break;
645     case NC_WM:
646       switch (wmn->data) {
647         case ND_FILEREAD:
648           ED_area_tag_refresh(sa);
649           break;
650       }
651       break;
652   }
653 }
654
655 static void action_header_region_listener(
656     wmWindow *UNUSED(win), ScrArea *sa, ARegion *ar, wmNotifier *wmn, const Scene *UNUSED(scene))
657 {
658   SpaceAction *saction = (SpaceAction *)sa->spacedata.first;
659
660   /* context changes */
661   switch (wmn->category) {
662     case NC_SCREEN:
663       if (saction->mode == SACTCONT_TIMELINE) {
664         if (wmn->data == ND_ANIMPLAY)
665           ED_region_tag_redraw(ar);
666       }
667       break;
668     case NC_SCENE:
669       if (saction->mode == SACTCONT_TIMELINE) {
670         switch (wmn->data) {
671           case ND_RENDER_RESULT:
672           case ND_OB_SELECT:
673           case ND_FRAME:
674           case ND_FRAME_RANGE:
675           case ND_KEYINGSET:
676           case ND_RENDER_OPTIONS:
677             ED_region_tag_redraw(ar);
678             break;
679         }
680       }
681       else {
682         switch (wmn->data) {
683           case ND_OB_ACTIVE:
684             ED_region_tag_redraw(ar);
685             break;
686         }
687       }
688       break;
689     case NC_ID:
690       if (wmn->action == NA_RENAME)
691         ED_region_tag_redraw(ar);
692       break;
693     case NC_ANIMATION:
694       switch (wmn->data) {
695         case ND_ANIMCHAN: /* set of visible animchannels changed */
696           /* NOTE: for now, this should usually just mean that the filters changed
697            *       It may be better if we had a dedicated flag for that though
698            */
699           ED_region_tag_redraw(ar);
700           break;
701
702         case ND_KEYFRAME: /* new keyframed added -> active action may have changed */
703           //saction->flag |= SACTION_TEMP_NEEDCHANSYNC;
704           ED_region_tag_redraw(ar);
705           break;
706       }
707       break;
708   }
709 }
710
711 /* add handlers, stuff you only do once or on area/region changes */
712 static void action_buttons_area_init(wmWindowManager *wm, ARegion *ar)
713 {
714   wmKeyMap *keymap;
715
716   ED_region_panels_init(wm, ar);
717
718   keymap = WM_keymap_ensure(wm->defaultconf, "Dopesheet Generic", SPACE_ACTION, 0);
719   WM_event_add_keymap_handler(&ar->handlers, keymap);
720 }
721
722 static void action_buttons_area_draw(const bContext *C, ARegion *ar)
723 {
724   ED_region_panels(C, ar);
725 }
726
727 static void action_region_listener(wmWindow *UNUSED(win),
728                                    ScrArea *UNUSED(sa),
729                                    ARegion *ar,
730                                    wmNotifier *wmn,
731                                    const Scene *UNUSED(scene))
732 {
733   /* context changes */
734   switch (wmn->category) {
735     case NC_ANIMATION:
736       ED_region_tag_redraw(ar);
737       break;
738     case NC_SCENE:
739       switch (wmn->data) {
740         case ND_OB_ACTIVE:
741         case ND_FRAME:
742         case ND_MARKERS:
743           ED_region_tag_redraw(ar);
744           break;
745       }
746       break;
747     case NC_OBJECT:
748       switch (wmn->data) {
749         case ND_BONE_ACTIVE:
750         case ND_BONE_SELECT:
751         case ND_KEYS:
752           ED_region_tag_redraw(ar);
753           break;
754       }
755       break;
756     default:
757       if (wmn->data == ND_KEYS)
758         ED_region_tag_redraw(ar);
759       break;
760   }
761 }
762
763 static void action_refresh(const bContext *C, ScrArea *sa)
764 {
765   SpaceAction *saction = (SpaceAction *)sa->spacedata.first;
766
767   /* update the state of the animchannels in response to changes from the data they represent
768    * NOTE: the temp flag is used to indicate when this needs to be done, and will be cleared once handled
769    */
770   if (saction->runtime.flag & SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC) {
771     ARegion *ar;
772
773     /* Perform syncing of channel state incl. selection
774      * Active action setting also occurs here (as part of anim channel filtering in anim_filter.c)
775      */
776     ANIM_sync_animchannels_to_data(C);
777     saction->runtime.flag &= ~SACTION_RUNTIME_FLAG_NEED_CHAN_SYNC;
778
779     /* Tag everything for redraw
780      * - Regions (such as header) need to be manually tagged for redraw too
781      *   or else they don't update [#28962]
782      */
783     ED_area_tag_redraw(sa);
784     for (ar = sa->regionbase.first; ar; ar = ar->next)
785       ED_region_tag_redraw(ar);
786   }
787
788   /* region updates? */
789   // XXX re-sizing y-extents of tot should go here?
790 }
791
792 static void action_id_remap(ScrArea *UNUSED(sa), SpaceLink *slink, ID *old_id, ID *new_id)
793 {
794   SpaceAction *sact = (SpaceAction *)slink;
795
796   if ((ID *)sact->action == old_id) {
797     sact->action = (bAction *)new_id;
798   }
799
800   if ((ID *)sact->ads.filter_grp == old_id) {
801     sact->ads.filter_grp = (Collection *)new_id;
802   }
803   if ((ID *)sact->ads.source == old_id) {
804     sact->ads.source = new_id;
805   }
806 }
807
808 /**
809  * \note Used for splitting out a subset of modes is more involved,
810  * The previous non-timeline mode is stored so switching back to the
811  * dope-sheet doesn't always reset the sub-mode.
812  */
813 static int action_space_subtype_get(ScrArea *sa)
814 {
815   SpaceAction *sact = sa->spacedata.first;
816   return sact->mode == SACTCONT_TIMELINE ? SACTCONT_TIMELINE : SACTCONT_DOPESHEET;
817 }
818
819 static void action_space_subtype_set(ScrArea *sa, int value)
820 {
821   SpaceAction *sact = sa->spacedata.first;
822   if (value == SACTCONT_TIMELINE) {
823     if (sact->mode != SACTCONT_TIMELINE) {
824       sact->mode_prev = sact->mode;
825     }
826     sact->mode = value;
827   }
828   else {
829     sact->mode = sact->mode_prev;
830   }
831 }
832
833 static void action_space_subtype_item_extend(bContext *UNUSED(C),
834                                              EnumPropertyItem **item,
835                                              int *totitem)
836 {
837   RNA_enum_items_add(item, totitem, rna_enum_space_action_mode_items);
838 }
839
840 /* only called once, from space/spacetypes.c */
841 void ED_spacetype_action(void)
842 {
843   SpaceType *st = MEM_callocN(sizeof(SpaceType), "spacetype action");
844   ARegionType *art;
845
846   st->spaceid = SPACE_ACTION;
847   strncpy(st->name, "Action", BKE_ST_MAXNAME);
848
849   st->new = action_new;
850   st->free = action_free;
851   st->init = action_init;
852   st->duplicate = action_duplicate;
853   st->operatortypes = action_operatortypes;
854   st->keymap = action_keymap;
855   st->listener = action_listener;
856   st->refresh = action_refresh;
857   st->id_remap = action_id_remap;
858   st->space_subtype_item_extend = action_space_subtype_item_extend;
859   st->space_subtype_get = action_space_subtype_get;
860   st->space_subtype_set = action_space_subtype_set;
861
862   /* regions: main window */
863   art = MEM_callocN(sizeof(ARegionType), "spacetype action region");
864   art->regionid = RGN_TYPE_WINDOW;
865   art->init = action_main_region_init;
866   art->draw = action_main_region_draw;
867   art->listener = action_main_region_listener;
868   art->message_subscribe = saction_main_region_message_subscribe;
869   art->keymapflag = ED_KEYMAP_VIEW2D | ED_KEYMAP_MARKERS | ED_KEYMAP_ANIMATION | ED_KEYMAP_FRAMES;
870
871   BLI_addhead(&st->regiontypes, art);
872
873   /* regions: header */
874   art = MEM_callocN(sizeof(ARegionType), "spacetype action region");
875   art->regionid = RGN_TYPE_HEADER;
876   art->prefsizey = HEADERY;
877   art->keymapflag = ED_KEYMAP_UI | ED_KEYMAP_VIEW2D | ED_KEYMAP_FRAMES | ED_KEYMAP_HEADER;
878
879   art->init = action_header_region_init;
880   art->draw = action_header_region_draw;
881   art->listener = action_header_region_listener;
882
883   BLI_addhead(&st->regiontypes, art);
884
885   /* regions: channels */
886   art = MEM_callocN(sizeof(ARegionType), "spacetype action region");
887   art->regionid = RGN_TYPE_CHANNELS;
888   art->prefsizex = 200;
889   art->keymapflag = ED_KEYMAP_UI | ED_KEYMAP_VIEW2D | ED_KEYMAP_FRAMES;
890
891   art->init = action_channel_region_init;
892   art->draw = action_channel_region_draw;
893   art->listener = action_channel_region_listener;
894   art->message_subscribe = saction_channel_region_message_subscribe;
895
896   BLI_addhead(&st->regiontypes, art);
897
898   /* regions: UI buttons */
899   art = MEM_callocN(sizeof(ARegionType), "spacetype action region");
900   art->regionid = RGN_TYPE_UI;
901   art->prefsizex = 200;
902   art->keymapflag = ED_KEYMAP_UI;
903   art->listener = action_region_listener;
904   art->init = action_buttons_area_init;
905   art->draw = action_buttons_area_draw;
906
907   BLI_addhead(&st->regiontypes, art);
908
909   action_buttons_register(art);
910
911   BKE_spacetype_register(st);
912 }