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