Merge branch 'master' into blender2.8
authorCampbell Barton <ideasman42@gmail.com>
Mon, 20 Nov 2017 09:45:03 +0000 (20:45 +1100)
committerCampbell Barton <ideasman42@gmail.com>
Mon, 20 Nov 2017 09:45:03 +0000 (20:45 +1100)
26 files changed:
1  2 
release/scripts/startup/bl_ui/space_view3d_toolbar.py
source/blender/blenkernel/CMakeLists.txt
source/blender/editors/animation/anim_markers.c
source/blender/editors/armature/pose_edit.c
source/blender/editors/armature/pose_transform.c
source/blender/editors/curve/editcurve.c
source/blender/editors/include/ED_mesh.h
source/blender/editors/mask/mask_ops.c
source/blender/editors/mesh/editface.c
source/blender/editors/mesh/editmesh_tools.c
source/blender/editors/mesh/editmesh_utils.c
source/blender/editors/metaball/mball_edit.c
source/blender/editors/object/object_lattice.c
source/blender/editors/physics/particle_edit.c
source/blender/editors/sculpt_paint/paint_utils.c
source/blender/editors/space_graph/graph_buttons.c
source/blender/editors/space_outliner/outliner_intern.h
source/blender/editors/space_outliner/outliner_select.c
source/blender/editors/space_outliner/outliner_tools.c
source/blender/editors/space_view3d/view3d_select.c
source/blender/editors/transform/transform_generics.c
source/blender/editors/uvedit/uvedit_ops.c
source/blender/editors/uvedit/uvedit_unwrap_ops.c
source/blender/makesrna/intern/rna_wm_manipulator.c
source/blender/render/extern/include/RE_pipeline.h
source/blender/windowmanager/intern/wm_files.c

index 6063c0689092b2ba878aa249cce6c3ed3a3b03c0,96c653dee23e0bd28cc05abed20eb9bb4b8f84cf..24b0adb1adb418a807dd87c3e0953b7bad124d29
@@@ -304,8 -292,6 +304,7 @@@ set(SR
        BKE_texture.h
        BKE_tracking.h
        BKE_unit.h
-       BKE_utildefines.h
 +      BKE_workspace.h
        BKE_world.h
        BKE_writeavi.h
        BKE_writeframeserver.h
index 948c780342071df1f0f8fd40fee95f2f90fdd223,3707b914ecbfc8c3fb5abafea3d01363713f6a26..18d6408f0266af7ced35ac52eb0199b9bfa85418
@@@ -542,8 -541,15 +548,15 @@@ static int pose_paste_exec(bContext *C
                }
        }
        BKE_main_free(tmp_bmain);
+       
        /* Update event for pose and deformation children. */
 -      DAG_id_tag_update(&ob->id, OB_RECALC_DATA);
 +      DEG_id_tag_update(&ob->id, OB_RECALC_DATA);
+       
+       /* Recalculate paths if any of the bones have paths... */
+       if ((ob->pose->avs.path_bakeflag & MOTIONPATH_BAKE_HAS_PATHS)) {
 -              ED_pose_recalculate_paths(scene, ob);
++              ED_pose_recalculate_paths(C, scene, ob);
+       }
+       
        /* Notifiers for updates, */
        WM_event_add_notifier(C, NC_OBJECT | ND_POSE, ob);
  
index 175b40d72ff1d97b94855df93de813eb31af9d4c,5b4078b98b79e2ae96941212c5c1d404877e5fa8..935f1a5ea4aeed730c3b8c1134e9800abc00407d
@@@ -547,17 -546,19 +547,19 @@@ static int reveal_metaelems_exec(bConte
  {
        Object *obedit = CTX_data_edit_object(C);
        MetaBall *mb = (MetaBall *)obedit->data;
-       MetaElem *ml;
-       ml = mb->editelems->first;
+       const bool select = RNA_boolean_get(op->ptr, "select");
+       bool changed = false;
  
-       if (ml) {
-               while (ml) {
+       for (MetaElem *ml = mb->editelems->first; ml; ml = ml->next) {
+               if (ml->flag & MB_HIDE) {
+                       SET_FLAG_FROM_TEST(ml->flag, select, SELECT);
                        ml->flag &= ~MB_HIDE;
-                       ml = ml->next;
+                       changed = true;
                }
+       }
+       if (changed) {
                WM_event_add_notifier(C, NC_GEOM | ND_DATA, mb);
 -              DAG_id_tag_update(obedit->data, 0);
 +              DEG_id_tag_update(obedit->data, 0);
        }
        
        return OPERATOR_FINISHED;
index 557e0a0eccf983cb2ed5d2de83951f1199aaafc1,57053ddc020adc8edcd73fd133a9b1be5ee278b4..b2f9bee27ff8af2c7d9be274c286a71102929e28
  #include "BKE_lattice.h"
  #include "BKE_deform.h"
  #include "BKE_report.h"
- #include "BKE_utildefines.h"
  
 +#include "DEG_depsgraph.h"
 +
  #include "ED_lattice.h"
  #include "ED_object.h"
  #include "ED_screen.h"
index a46425a937e3b3c29855e0bb342c43223df26d57,da66ec44235fef8ce7f6203f9fc975f75f82b39e..8986ebd26e944d87e67af4aa89153cb9a4af6aef
@@@ -1977,8 -1954,8 +1977,9 @@@ static int reveal_exec(bContext *C, wmO
  {
        Object *ob= CTX_data_active_object(C);
        Scene *scene= CTX_data_scene(C);
 -      PTCacheEdit *edit = PE_get_current(scene, ob);
 +      SceneLayer *sl = CTX_data_scene_layer(C);
 +      PTCacheEdit *edit= PE_get_current(scene, sl, ob);
+       const bool select = RNA_boolean_get(op->ptr, "select");
        POINT_P; KEY_K;
  
        LOOP_POINTS {
index 8d93f6bca55a634f011feefed90bc8bde7fd6fe6,ca7dbe4f73c364277f141a3293809dc37832631d..1edb3c883966af48f534f4c4366f6f0047b19902
@@@ -192,11 -146,17 +192,17 @@@ void restrictbutton_gr_restrict_flag(vo
  
  /* outliner_select.c -------------------------------------------- */
  eOLDrawState tree_element_type_active(
 -        struct bContext *C, struct Scene *scene, struct SpaceOops *soops,
 +        struct bContext *C, struct Scene *scene, struct SceneLayer *sl, struct SpaceOops *soops,
          TreeElement *te, TreeStoreElem *tselem, const eOLSetState set, bool recursive);
 -eOLDrawState tree_element_active(struct bContext *C, struct Scene *scene, SpaceOops *soops,
 +eOLDrawState tree_element_active(struct bContext *C, struct Scene *scene, struct SceneLayer *sl, SpaceOops *soops,
                                   TreeElement *te, const eOLSetState set, const bool handle_all_types);
- int outliner_item_activate_or_toggle_closed(struct bContext *C, int x, int y, bool extend, bool recursive);
+ void outliner_item_do_activate_from_tree_element(
+         struct bContext *C, TreeElement *te, TreeStoreElem *tselem,
+         bool extend, bool recursive);
+ int outliner_item_do_activate_from_cursor(
+         struct bContext *C, const int mval[2],
+         bool extend, bool recursive);
  
  /* outliner_edit.c ---------------------------------------------- */
  typedef void (*outliner_operation_cb)(
index 606cc2fd14a9c711daa06228441742a0306d63f5,9f79b575966fd3ce88370441f4f657ec07dbdc73..670be7611e4f277834be278b9e4de5746e089f18
@@@ -876,21 -892,25 +876,25 @@@ eOLDrawState tree_element_type_active
  
  /* ================================================ */
  
- static void outliner_item_activate(
-         bContext *C, SpaceOops *soops, TreeElement *te,
+ /**
+  * Action when clicking to activate an item (typically under the mouse cursor),
+  * but don't do any cursor intersection checks.
+  *
+  * Needed to run from operators accessed from a menu.
+  */
+ static void do_outliner_item_activate_tree_element(
 -        bContext *C, Scene *scene, SpaceOops *soops,
++        bContext *C, Scene *scene, SceneLayer *sl, SpaceOops *soops,
+         TreeElement *te, TreeStoreElem *tselem,
 -        bool extend, bool recursive)
 +        const bool extend, const bool recursive)
  {
-       Scene *scene = CTX_data_scene(C);
-       SceneLayer *sl = CTX_data_scene_layer(C);
-       TreeStoreElem *tselem = TREESTORE(te);
        /* always makes active object, except for some specific types.
         * Note about TSE_EBONE: In case of a same ID_AR datablock shared among several objects, we do not want
         * to switch out of edit mode (see T48328 for details). */
 -      if (!ELEM(tselem->type, TSE_SEQUENCE, TSE_SEQ_STRIP, TSE_SEQUENCE_DUP, TSE_EBONE)) {
 +      if (!ELEM(tselem->type, TSE_SEQUENCE, TSE_SEQ_STRIP, TSE_SEQUENCE_DUP, TSE_EBONE, TSE_LAYER_COLLECTION)) {
-               tree_element_set_active_object(C, scene, sl, soops, te,
-                                              (extend && tselem->type == 0) ? OL_SETSEL_EXTEND : OL_SETSEL_NORMAL,
-                                              recursive && tselem->type == 0);
+               tree_element_set_active_object(
 -                      C, scene, soops, te,
++                      C, scene, sl, soops, te,
+                       (extend && tselem->type == 0) ? OL_SETSEL_EXTEND : OL_SETSEL_NORMAL,
+                       recursive && tselem->type == 0);
        }
  
        if (tselem->type == 0) { // the lib blocks
  }
  
  /**
 - * Activates tree items, also handles clicking on arrows.
 + * \param extend: Don't deselect other items, only modify \a te.
 + * \param toggle: Select \a te when not selected, deselect when selected.
   */
 -static bool do_outliner_item_activate_from_cursor(
 -        bContext *C, Scene *scene, ARegion *ar, SpaceOops *soops,
 -        TreeElement *te, bool extend, bool recursive, const float mval[2])
 +static void outliner_item_select(SpaceOops *soops, const TreeElement *te, const bool extend, const bool toggle)
  {
 -      if (mval[1] > te->ys && mval[1] < te->ys + UI_UNIT_Y) {
 -              TreeStoreElem *tselem = TREESTORE(te);
 -              bool openclose = false;
 -              
 -              /* open close icon */
 -              if ((te->flag & TE_ICONROW) == 0) {               // hidden icon, no open/close
 -                      if (mval[0] > te->xs && mval[0] < te->xs + UI_UNIT_X)
 -                              openclose = true;
 -              }
 -              
 -              if (openclose) {
 -                      /* all below close/open? */
 -                      if (extend) {
 -                              tselem->flag &= ~TSE_CLOSED;
 -                              outliner_set_flag(&te->subtree, TSE_CLOSED, !outliner_has_one_flag(&te->subtree, TSE_CLOSED, 1));
 -                      }
 -                      else {
 -                              if (tselem->flag & TSE_CLOSED) tselem->flag &= ~TSE_CLOSED;
 -                              else tselem->flag |= TSE_CLOSED;
 -                              
 -                      }
 -                      
 -                      return true;
 -              }
 -              /* name and first icon */
 -              else if (mval[0] > te->xs + UI_UNIT_X && mval[0] < te->xend) {
 -                      do_outliner_item_activate_tree_element(
 -                              C, scene, soops,
 -                              te, tselem,
 -                              extend, recursive);
 -                      return true;
 -              }
 +      TreeStoreElem *tselem = TREESTORE(te);
 +      const short new_flag = toggle ? (tselem->flag ^ TSE_SELECTED) : (tselem->flag | TSE_SELECTED);
 +
 +      if (extend == false) {
 +              outliner_set_flag(&soops->tree, TSE_SELECTED, false);
        }
 -      
 -      for (te = te->subtree.first; te; te = te->next) {
 -              if (do_outliner_item_activate_from_cursor(C, scene, ar, soops, te, extend, recursive, mval)) {
 -                      return true;
 -              }
 +      tselem->flag = new_flag;
 +}
 +
 +static void outliner_item_toggle_closed(TreeElement *te, const bool toggle_children)
 +{
 +      TreeStoreElem *tselem = TREESTORE(te);
 +      if (toggle_children) {
 +              tselem->flag &= ~TSE_CLOSED;
 +
 +              const bool all_opened = !outliner_has_one_flag(&te->subtree, TSE_CLOSED, 1);
 +              outliner_set_flag(&te->subtree, TSE_CLOSED, all_opened);
        }
 -      return false;
 +      else {
 +              tselem->flag ^= TSE_CLOSED;
 +      }
 +}
 +
 +static bool outliner_item_is_co_within_close_toggle(TreeElement *te, float view_co_x)
 +{
 +      return ((te->flag & TE_ICONROW) == 0) && (view_co_x > te->xs) && (view_co_x < te->xs + UI_UNIT_X);
 +}
 +
 +static bool outliner_is_co_within_restrict_columns(const SpaceOops *soops, const ARegion *ar, float view_co_x)
 +{
 +      return (!ELEM(soops->outlinevis, SO_DATABLOCKS, SO_USERDEF) &&
 +              !(soops->flag & SO_HIDE_RESTRICTCOLS) &&
 +              (view_co_x > ar->v2d.cur.xmax - OL_TOG_RESTRICT_VIEWX));
  }
  
- int outliner_item_activate_or_toggle_closed(bContext *C, int x, int y, bool extend, bool recursive)
+ /**
+  * A version of #outliner_item_do_acticate_from_cursor that takes the tree element directly.
+  * and doesn't depend on the pointer position.
+  *
+  * This allows us to simulate clicking on an item without dealing with the mouse cursor.
+  */
+ void outliner_item_do_activate_from_tree_element(
+         bContext *C, TreeElement *te, TreeStoreElem *tselem,
+         bool extend, bool recursive)
+ {
+       Scene *scene = CTX_data_scene(C);
++      SceneLayer *sl = CTX_data_scene_layer(C);
+       SpaceOops *soops = CTX_wm_space_outliner(C);
+       do_outliner_item_activate_tree_element(
 -              C, scene, soops,
++              C, scene, sl, soops,
+               te, tselem,
+               extend, recursive);
+ }
+ /**
+  * Action to run when clicking in the outliner,
+  *
+  * May expend/collapse branches or activate items.
+  * */
+ int outliner_item_do_activate_from_cursor(
+         bContext *C, const int mval[2],
+         bool extend, bool recursive)
  {
 -      Scene *scene = CTX_data_scene(C);
        ARegion *ar = CTX_wm_region(C);
        SpaceOops *soops = CTX_wm_space_outliner(C);
        TreeElement *te;
 -      float fmval[2];
 +      float view_mval[2];
 +      bool changed = false, rebuild_tree = false;
  
-       UI_view2d_region_to_view(&ar->v2d, x, y, &view_mval[0], &view_mval[1]);
 -      UI_view2d_region_to_view(&ar->v2d, mval[0], mval[1], &fmval[0], &fmval[1]);
++      UI_view2d_region_to_view(&ar->v2d, mval[0], mval[1], &view_mval[0], &view_mval[1]);
  
 -      if (!ELEM(soops->outlinevis, SO_DATABLOCKS, SO_USERDEF) &&
 -          !(soops->flag & SO_HIDE_RESTRICTCOLS) &&
 -          (fmval[0] > ar->v2d.cur.xmax - OL_TOG_RESTRICT_VIEWX))
 -      {
 +      if (outliner_is_co_within_restrict_columns(soops, ar, view_mval[0])) {
                return OPERATOR_CANCELLED;
        }
  
 -      for (te = soops->tree.first; te; te = te->next) {
 -              if (do_outliner_item_activate_from_cursor(C, scene, ar, soops, te, extend, recursive, fmval)) {
 -                      break;
 -              }
 +      if (!(te = outliner_find_item_at_y(soops, &soops->tree, view_mval[1]))) {
 +              /* skip */
        }
 -      
 -      if (te) {
 -              ED_undo_push(C, "Outliner click event");
 +      else if (outliner_item_is_co_within_close_toggle(te, view_mval[0])) {
 +              outliner_item_toggle_closed(te, extend);
 +              changed = true;
 +              rebuild_tree = true;
        }
        else {
 -              short selecting = -1;
 -              int row;
 -              
 -              /* get row number - 100 here is just a dummy value since we don't need the column */
 -              UI_view2d_listview_view_to_cell(&ar->v2d, 1000, UI_UNIT_Y, 0.0f, OL_Y_OFFSET, 
 -                                              fmval[0], fmval[1], NULL, &row);
 -              
 -              /* select relevant row */
 -              if (outliner_select(soops, &soops->tree, &row, &selecting)) {
 -              
++              Scene *scene = CTX_data_scene(C);
++              SceneLayer *sl = CTX_data_scene_layer(C);
 +              /* the row may also contain children, if one is hovered we want this instead of current te */
 +              TreeElement *activate_te = outliner_find_item_at_x_in_row(soops, te, view_mval[0]);
++              TreeStoreElem *activate_tselem = TREESTORE(activate_te);
 +
 +              outliner_item_select(soops, activate_te, extend, extend);
-               outliner_item_activate(C, soops, activate_te, extend, recursive);
++              do_outliner_item_activate_tree_element(C, scene, sl, soops, activate_te, activate_tselem, extend, recursive);
 +              changed = true;
 +      }
 +
 +      if (changed) {
 +              if (!rebuild_tree) {
 +                      /* only needs to redraw, no rebuild */
                        soops->storeflag |= SO_TREESTORE_REDRAW;
 -              
 -                      /* no need for undo push here, only changing outliner data which is
 -                       * scene level - campbell */
 -                      /* ED_undo_push(C, "Outliner selection event"); */
                }
 +              ED_undo_push(C, "Outliner selection change");
 +              ED_region_tag_redraw(ar);
        }
 -      
 -      ED_region_tag_redraw(ar);
  
        return OPERATOR_FINISHED;
  }
index ca168094ae2c57fe7c8d8c0d0389eaa4a4ff3ed4,cf9deeaedf8e2f57ce004624aa56f4c4a2021414..79122de70dd41eae62a61ae4c82deebd05f380dc
@@@ -376,27 -372,24 +376,24 @@@ static void object_select_cb
  }
  
  static void object_select_hierarchy_cb(
-         bContext *C, ReportList *UNUSED(reports), Scene *UNUSED(scene), TreeElement *UNUSED(te),
-         TreeStoreElem *UNUSED(tsep), TreeStoreElem *UNUSED(tselem), void *UNUSED(user_data))
+         bContext *C, ReportList *UNUSED(reports), Scene *UNUSED(scene), TreeElement *te,
+         TreeStoreElem *UNUSED(tsep), TreeStoreElem *tselem, void *UNUSED(user_data))
  {
-       /* From where do i get the x,y coordinate of the mouse event ? */
-       wmWindow *win = CTX_wm_window(C);
-       int x = win->eventstate->mval[0];
-       int y = win->eventstate->mval[1];
-       outliner_item_activate_or_toggle_closed(C, x, y, true, true);
+       /* Don't extend because this toggles, which is nice for Ctrl-Click but not for a menu item.
+        * it's especially confusing when multiple items are selected since some toggle on/off. */
+       outliner_item_do_activate_from_tree_element(C, te, tselem, false, true);
  }
  
  static void object_deselect_cb(
 -        bContext *UNUSED(C), ReportList *UNUSED(reports), Scene *scene, TreeElement *te,
 +        bContext *C, ReportList *UNUSED(reports), Scene *UNUSED(scene), TreeElement *UNUSED(te),
          TreeStoreElem *UNUSED(tsep), TreeStoreElem *tselem, void *UNUSED(user_data))
  {
 -      Base *base = (Base *)te->directdata;
 -      
 -      if (base == NULL) base = BKE_scene_base_find(scene, (Object *)tselem->id);
 +      SceneLayer *sl = CTX_data_scene_layer(C);
 +      Object *ob = (Object *)tselem->id;
 +      Base *base = BKE_scene_layer_base_find(sl, ob);
 +
        if (base) {
 -              base->flag &= ~SELECT;
 -              base->object->flag &= ~SELECT;
 +              base->flag &= ~BASE_SELECTED;
        }
  }
  
index 9a058598580e4fb8d2331ce5e8ce7e03e983e34a,05cf552e5ccacc6114a338ace97c5cbcf1a8a226..297bd5c4aa113a2c4eee60ac670b433e70bc9eb8
  #include "BKE_object.h"
  #include "BKE_paint.h"
  #include "BKE_editmesh.h"
 +#include "BKE_scene.h"
  #include "BKE_tracking.h"
- #include "BKE_utildefines.h"
  
 -#include "BIF_gl.h"
 -#include "BIF_glutil.h"
 +#include "DEG_depsgraph.h"
  
  #include "WM_api.h"
  #include "WM_types.h"
@@@ -703,10 -724,10 +702,10 @@@ static void do_lasso_select_meshobject_
        if (BLI_rctf_isect_pt_v(data->rect_fl, screen_co) &&
            BLI_lasso_is_point_inside(data->mcords, data->moves, screen_co[0], screen_co[1], IS_CLIPPED))
        {
-               BKE_BIT_TEST_SET(mv->flag, data->select, SELECT);
+               SET_FLAG_FROM_TEST(mv->flag, data->select, SELECT);
        }
  }
 -static void do_lasso_select_paintvert(ViewContext *vc, const int mcords[][2], short moves, bool extend, bool select)
 +static void do_lasso_select_paintvert(const struct EvaluationContext *eval_ctx, ViewContext *vc, const int mcords[][2], short moves, bool extend, bool select)
  {
        const bool use_zbuf = (vc->v3d->flag & V3D_ZBUF_SELECT) != 0;
        Object *ob = vc->obact;
@@@ -1660,11 -1670,10 +1659,11 @@@ static void do_paintvert_box_select__do
        BoxSelectUserData *data = userData;
  
        if (BLI_rctf_isect_pt_v(data->rect_fl, screen_co)) {
-               BKE_BIT_TEST_SET(mv->flag, data->select, SELECT);
+               SET_FLAG_FROM_TEST(mv->flag, data->select, SELECT);
        }
  }
 -static int do_paintvert_box_select(ViewContext *vc, rcti *rect, bool select, bool extend)
 +static int do_paintvert_box_select(
 +        const EvaluationContext *eval_ctx, ViewContext *vc, rcti *rect, bool select, bool extend)
  {
        const bool use_zbuf = (vc->v3d->flag & V3D_ZBUF_SELECT) != 0;
        Mesh *me;
@@@ -2522,10 -2496,10 +2521,10 @@@ static void paint_vertsel_circle_select
        CircleSelectUserData *data = userData;
  
        if (len_squared_v2v2(data->mval_fl, screen_co) <= data->radius_squared) {
-               BKE_BIT_TEST_SET(mv->flag, data->select, SELECT);
+               SET_FLAG_FROM_TEST(mv->flag, data->select, SELECT);
        }
  }
 -static void paint_vertsel_circle_select(ViewContext *vc, const bool select, const int mval[2], float rad)
 +static void paint_vertsel_circle_select(const struct EvaluationContext *eval_ctx, ViewContext *vc, const bool select, const int mval[2], float rad)
  {
        const bool use_zbuf = (vc->v3d->flag & V3D_ZBUF_SELECT) != 0;
        Object *ob = vc->obact;
index 863e0c822dd88595542766fbf9ab322d5b15509d,277e01d1e2bd74e9a8919a93b3318cff8ede8006..8d2fb595e2438cde1afd7734e846e74dec440a10
  #include "BKE_editmesh.h"
  #include "BKE_tracking.h"
  #include "BKE_mask.h"
- #include "BKE_utildefines.h"
 +#include "BKE_workspace.h"
 +
 +#include "DEG_depsgraph.h"
  
  #include "ED_anim_api.h"
  #include "ED_armature.h"
index a8dcd563ed39c5e4b635aa512d6101e8c98812b9,0000000000000000000000000000000000000000..00705456fc7fec6201c82aff07cb039f32e22d64
mode 100644,000000..100644
--- /dev/null
@@@ -1,1307 -1,0 +1,1306 @@@
- #include "BKE_utildefines.h"
 +/*
 + * ***** BEGIN GPL LICENSE BLOCK *****
 + *
 + * This program is free software; you can redistribute it and/or
 + * modify it under the terms of the GNU General Public License
 + * as published by the Free Software Foundation; either version 2
 + * of the License, or (at your option) any later version.
 + *
 + * This program is distributed in the hope that it will be useful,
 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 + * GNU General Public License for more details.
 + *
 + * You should have received a copy of the GNU General Public License
 + * along with this program; if not, write to the Free Software Foundation,
 + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 + *
 + * ***** END GPL LICENSE BLOCK *****
 + */
 +
 +/** \file blender/makesrna/intern/rna_wm_manipulator.c
 + *  \ingroup RNA
 + */
 +
 +#include <stdlib.h>
 +
 +#include "DNA_screen_types.h"
 +#include "DNA_space_types.h"
 +#include "DNA_userdef_types.h"
 +#include "DNA_view3d_types.h"
 +#include "DNA_windowmanager_types.h"
 +
 +#include "BLI_utildefines.h"
 +#include "BLI_string_utils.h"
 +
 +#include "BLT_translation.h"
 +
 +#include "RNA_access.h"
 +#include "RNA_define.h"
 +#include "RNA_enum_types.h"
 +
 +#include "rna_internal.h"
 +
 +#include "WM_types.h"
 +
 +#ifdef RNA_RUNTIME
 +/* enum definitions */
 +#endif /* RNA_RUNTIME */
 +
 +#ifdef RNA_RUNTIME
 +
 +#include <assert.h>
 +
 +#include "WM_api.h"
 +
 +#include "DNA_workspace_types.h"
 +
 +#include "ED_screen.h"
 +
 +#include "UI_interface.h"
 +
 +#include "BKE_global.h"
 +#include "BKE_idprop.h"
 +#include "BKE_workspace.h"
-       BKE_BIT_TEST_SET(mpr->member_id, value, flag_value); \
 +
 +#include "MEM_guardedalloc.h"
 +
 +#ifdef WITH_PYTHON
 +#  include "BPY_extern.h"
 +#endif
 +
 +/* -------------------------------------------------------------------- */
 +
 +/** \name Manipulator API
 + * \{ */
 +
 +static void rna_manipulator_draw_cb(
 +        const struct bContext *C, struct wmManipulator *mpr)
 +{
 +      extern FunctionRNA rna_Manipulator_draw_func;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      PointerRNA mpr_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      RNA_pointer_create(NULL, mpr->type->ext.srna, mpr, &mpr_ptr);
 +      /* RNA_struct_find_function(&mpr_ptr, "draw"); */
 +      func = &rna_Manipulator_draw_func;
 +      RNA_parameter_list_create(&list, &mpr_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      mgroup->type->ext.call((bContext *)C, &mpr_ptr, func, &list);
 +      RNA_parameter_list_free(&list);
 +}
 +
 +static void rna_manipulator_draw_select_cb(
 +        const struct bContext *C, struct wmManipulator *mpr, int select_id)
 +{
 +      extern FunctionRNA rna_Manipulator_draw_select_func;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      PointerRNA mpr_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      RNA_pointer_create(NULL, mpr->type->ext.srna, mpr, &mpr_ptr);
 +      /* RNA_struct_find_function(&mpr_ptr, "draw_select"); */
 +      func = &rna_Manipulator_draw_select_func;
 +      RNA_parameter_list_create(&list, &mpr_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      RNA_parameter_set_lookup(&list, "select_id", &select_id);
 +      mgroup->type->ext.call((bContext *)C, &mpr_ptr, func, &list);
 +      RNA_parameter_list_free(&list);
 +}
 +
 +static int rna_manipulator_test_select_cb(
 +        struct bContext *C, struct wmManipulator *mpr, const struct wmEvent *event)
 +{
 +      extern FunctionRNA rna_Manipulator_test_select_func;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      PointerRNA mpr_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      RNA_pointer_create(NULL, mpr->type->ext.srna, mpr, &mpr_ptr);
 +      /* RNA_struct_find_function(&mpr_ptr, "test_select"); */
 +      func = &rna_Manipulator_test_select_func;
 +      RNA_parameter_list_create(&list, &mpr_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      RNA_parameter_set_lookup(&list, "event", &event);
 +      mgroup->type->ext.call((bContext *)C, &mpr_ptr, func, &list);
 +
 +      void *ret;
 +      RNA_parameter_get_lookup(&list, "intersect_id", &ret);
 +      int intersect_id = *(int *)ret;
 +
 +      RNA_parameter_list_free(&list);
 +      return intersect_id;
 +}
 +
 +static int rna_manipulator_modal_cb(
 +        struct bContext *C, struct wmManipulator *mpr, const struct wmEvent *event,
 +        eWM_ManipulatorTweak tweak_flag)
 +{
 +      extern FunctionRNA rna_Manipulator_modal_func;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      PointerRNA mpr_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      const int tweak_flag_int = tweak_flag;
 +      RNA_pointer_create(NULL, mpr->type->ext.srna, mpr, &mpr_ptr);
 +      /* RNA_struct_find_function(&mpr_ptr, "modal"); */
 +      func = &rna_Manipulator_modal_func;
 +      RNA_parameter_list_create(&list, &mpr_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      RNA_parameter_set_lookup(&list, "event", &event);
 +      RNA_parameter_set_lookup(&list, "tweak", &tweak_flag_int);
 +      mgroup->type->ext.call((bContext *)C, &mpr_ptr, func, &list);
 +
 +      void *ret;
 +      RNA_parameter_get_lookup(&list, "result", &ret);
 +      int ret_enum = *(int *)ret;
 +
 +      RNA_parameter_list_free(&list);
 +      return ret_enum;
 +}
 +
 +static void rna_manipulator_setup_cb(
 +        struct wmManipulator *mpr)
 +{
 +      extern FunctionRNA rna_Manipulator_setup_func;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      PointerRNA mpr_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      RNA_pointer_create(NULL, mpr->type->ext.srna, mpr, &mpr_ptr);
 +      /* RNA_struct_find_function(&mpr_ptr, "setup"); */
 +      func = &rna_Manipulator_setup_func;
 +      RNA_parameter_list_create(&list, &mpr_ptr, func);
 +      mgroup->type->ext.call((bContext *)NULL, &mpr_ptr, func, &list);
 +      RNA_parameter_list_free(&list);
 +}
 +
 +
 +static int rna_manipulator_invoke_cb(
 +        struct bContext *C, struct wmManipulator *mpr, const struct wmEvent *event)
 +{
 +      extern FunctionRNA rna_Manipulator_invoke_func;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      PointerRNA mpr_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      RNA_pointer_create(NULL, mpr->type->ext.srna, mpr, &mpr_ptr);
 +      /* RNA_struct_find_function(&mpr_ptr, "invoke"); */
 +      func = &rna_Manipulator_invoke_func;
 +      RNA_parameter_list_create(&list, &mpr_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      RNA_parameter_set_lookup(&list, "event", &event);
 +      mgroup->type->ext.call((bContext *)C, &mpr_ptr, func, &list);
 +
 +      void *ret;
 +      RNA_parameter_get_lookup(&list, "result", &ret);
 +      int ret_enum = *(int *)ret;
 +
 +      RNA_parameter_list_free(&list);
 +      return ret_enum;
 +}
 +
 +static void rna_manipulator_exit_cb(
 +        struct bContext *C, struct wmManipulator *mpr, bool cancel)
 +{
 +      extern FunctionRNA rna_Manipulator_exit_func;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      PointerRNA mpr_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      RNA_pointer_create(NULL, mpr->type->ext.srna, mpr, &mpr_ptr);
 +      /* RNA_struct_find_function(&mpr_ptr, "exit"); */
 +      func = &rna_Manipulator_exit_func;
 +      RNA_parameter_list_create(&list, &mpr_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      {
 +              int cancel_i = cancel;
 +              RNA_parameter_set_lookup(&list, "cancel", &cancel_i);
 +      }
 +      mgroup->type->ext.call((bContext *)C, &mpr_ptr, func, &list);
 +      RNA_parameter_list_free(&list);
 +}
 +
 +static void rna_manipulator_select_refresh_cb(
 +        struct wmManipulator *mpr)
 +{
 +      extern FunctionRNA rna_Manipulator_select_refresh_func;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      PointerRNA mpr_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      RNA_pointer_create(NULL, mpr->type->ext.srna, mpr, &mpr_ptr);
 +      /* RNA_struct_find_function(&mpr_ptr, "select_refresh"); */
 +      func = &rna_Manipulator_select_refresh_func;
 +      RNA_parameter_list_create(&list, &mpr_ptr, func);
 +      mgroup->type->ext.call((bContext *)NULL, &mpr_ptr, func, &list);
 +      RNA_parameter_list_free(&list);
 +}
 +
 +/* just to work around 'const char *' warning and to ensure this is a python op */
 +static void rna_Manipulator_bl_idname_set(PointerRNA *ptr, const char *value)
 +{
 +      wmManipulator *data = ptr->data;
 +      char *str = (char *)data->type->idname;
 +      if (!str[0]) {
 +              BLI_strncpy(str, value, MAX_NAME);    /* utf8 already ensured */
 +      }
 +      else {
 +              assert(!"setting the bl_idname on a non-builtin operator");
 +      }
 +}
 +
 +static wmManipulator *rna_ManipulatorProperties_find_operator(PointerRNA *ptr)
 +{
 +#if 0
 +      wmWindowManager *wm = ptr->id.data;
 +#endif
 +
 +      /* We could try workaruond this lookup, but not trivial. */
 +      for (bScreen *screen = G.main->screen.first; screen; screen = screen->id.next) {
 +              IDProperty *properties = ptr->data;
 +              for (ScrArea *sa = screen->areabase.first; sa; sa = sa->next) {
 +                      for (ARegion *ar = sa->regionbase.first; ar; ar = ar->next) {
 +                              if (ar->manipulator_map) {
 +                                      wmManipulatorMap *mmap = ar->manipulator_map;
 +                                      for (wmManipulatorGroup *mgroup = WM_manipulatormap_group_list(mmap)->first;
 +                                           mgroup;
 +                                           mgroup = mgroup->next)
 +                                      {
 +                                              for (wmManipulator *mpr = mgroup->manipulators.first; mpr; mpr = mpr->next) {
 +                                                      if (mpr->properties == properties) {
 +                                                              return mpr;
 +                                                      }
 +                                              }
 +                                      }
 +                              }
 +                      }
 +              }
 +      }
 +      return NULL;
 +}
 +
 +static StructRNA *rna_ManipulatorProperties_refine(PointerRNA *ptr)
 +{
 +      wmManipulator *mpr = rna_ManipulatorProperties_find_operator(ptr);
 +
 +      if (mpr)
 +              return mpr->type->srna;
 +      else
 +              return ptr->type;
 +}
 +
 +static IDProperty *rna_ManipulatorProperties_idprops(PointerRNA *ptr, bool create)
 +{
 +      if (create && !ptr->data) {
 +              IDPropertyTemplate val = {0};
 +              ptr->data = IDP_New(IDP_GROUP, &val, "RNA_ManipulatorProperties group");
 +      }
 +
 +      return ptr->data;
 +}
 +
 +static PointerRNA rna_Manipulator_properties_get(PointerRNA *ptr)
 +{
 +      wmManipulator *mpr = ptr->data;
 +      return rna_pointer_inherit_refine(ptr, mpr->type->srna, mpr->properties);
 +}
 +
 +/* wmManipulator.float */
 +#define RNA_MANIPULATOR_GENERIC_FLOAT_RW_DEF(func_id, member_id) \
 +static float rna_Manipulator_##func_id##_get(PointerRNA *ptr) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      return mpr->member_id; \
 +} \
 +static void rna_Manipulator_##func_id##_set(PointerRNA *ptr, float value) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      mpr->member_id = value; \
 +}
 +#define RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_INDEX_RW_DEF(func_id, member_id, index) \
 +static float rna_Manipulator_##func_id##_get(PointerRNA *ptr) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      return mpr->member_id[index]; \
 +} \
 +static void rna_Manipulator_##func_id##_set(PointerRNA *ptr, float value) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      mpr->member_id[index] = value; \
 +}
 +/* wmManipulator.float[len] */
 +#define RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_RW_DEF(func_id, member_id, len) \
 +static void rna_Manipulator_##func_id##_get(PointerRNA *ptr, float value[len]) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      memcpy(value, mpr->member_id, sizeof(float[len])); \
 +} \
 +static void rna_Manipulator_##func_id##_set(PointerRNA *ptr, const float value[len]) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      memcpy(mpr->member_id, value, sizeof(float[len])); \
 +}
 +
 +/* wmManipulator.flag */
 +#define RNA_MANIPULATOR_GENERIC_FLAG_RW_DEF(func_id, member_id, flag_value) \
 +static int rna_Manipulator_##func_id##_get(PointerRNA *ptr) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      return (mpr->member_id & flag_value) != 0; \
 +} \
 +static void rna_Manipulator_##func_id##_set(PointerRNA *ptr, int value) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
-       BKE_BIT_TEST_SET(mpr->member_id, !value, flag_value); \
++      SET_FLAG_FROM_TEST(mpr->member_id, value, flag_value); \
 +}
 +
 +/* wmManipulator.flag (negative) */
 +#define RNA_MANIPULATOR_GENERIC_FLAG_NEG_RW_DEF(func_id, member_id, flag_value) \
 +static int rna_Manipulator_##func_id##_get(PointerRNA *ptr) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      return (mpr->member_id & flag_value) == 0; \
 +} \
 +static void rna_Manipulator_##func_id##_set(PointerRNA *ptr, int value) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
++      SET_FLAG_FROM_TEST(mpr->member_id, !value, flag_value); \
 +}
 +
 +#define RNA_MANIPULATOR_FLAG_RO_DEF(func_id, member_id, flag_value) \
 +static int rna_Manipulator_##func_id##_get(PointerRNA *ptr) \
 +{ \
 +      wmManipulator *mpr = ptr->data; \
 +      return (mpr->member_id & flag_value) != 0; \
 +}
 +
 +RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_RW_DEF(color, color, 3);
 +RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_RW_DEF(color_hi, color_hi, 3);
 +
 +RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_INDEX_RW_DEF(alpha, color, 3);
 +RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_INDEX_RW_DEF(alpha_hi, color_hi, 3);
 +
 +RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_RW_DEF(matrix_space, matrix_space, 16);
 +RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_RW_DEF(matrix_basis, matrix_basis, 16);
 +RNA_MANIPULATOR_GENERIC_FLOAT_ARRAY_RW_DEF(matrix_offset, matrix_offset, 16);
 +
 +static void rna_Manipulator_matrix_world_get(PointerRNA *ptr, float value[16])
 +{
 +      wmManipulator *mpr = ptr->data;
 +      WM_manipulator_calc_matrix_final(mpr, (float (*)[4])value);
 +}
 +
 +RNA_MANIPULATOR_GENERIC_FLOAT_RW_DEF(scale_basis, scale_basis);
 +RNA_MANIPULATOR_GENERIC_FLOAT_RW_DEF(line_width, line_width);
 +
 +RNA_MANIPULATOR_GENERIC_FLAG_RW_DEF(flag_use_draw_hover, flag, WM_MANIPULATOR_DRAW_HOVER);
 +RNA_MANIPULATOR_GENERIC_FLAG_RW_DEF(flag_use_draw_modal, flag, WM_MANIPULATOR_DRAW_MODAL);
 +RNA_MANIPULATOR_GENERIC_FLAG_RW_DEF(flag_use_draw_value, flag, WM_MANIPULATOR_DRAW_VALUE);
 +RNA_MANIPULATOR_GENERIC_FLAG_RW_DEF(flag_use_draw_offset_scale, flag, WM_MANIPULATOR_DRAW_OFFSET_SCALE);
 +RNA_MANIPULATOR_GENERIC_FLAG_NEG_RW_DEF(flag_use_draw_scale, flag, WM_MANIPULATOR_DRAW_OFFSET_SCALE);
 +RNA_MANIPULATOR_GENERIC_FLAG_RW_DEF(flag_hide, flag, WM_MANIPULATOR_HIDDEN);
 +
 +/* wmManipulator.state */
 +RNA_MANIPULATOR_FLAG_RO_DEF(state_is_highlight, state, WM_MANIPULATOR_STATE_HIGHLIGHT);
 +RNA_MANIPULATOR_FLAG_RO_DEF(state_is_modal, state, WM_MANIPULATOR_STATE_MODAL);
 +RNA_MANIPULATOR_FLAG_RO_DEF(state_select, state, WM_MANIPULATOR_STATE_SELECT);
 +
 +static void rna_Manipulator_state_select_set(struct PointerRNA *ptr, int value)
 +{
 +      wmManipulator *mpr = ptr->data;
 +      wmManipulatorGroup *mgroup = mpr->parent_mgroup;
 +      WM_manipulator_select_set(mgroup->parent_mmap, mpr, value);
 +}
 +
 +static PointerRNA rna_Manipulator_group_get(PointerRNA *ptr)
 +{
 +      wmManipulator *mpr = ptr->data;
 +      return rna_pointer_inherit_refine(ptr, &RNA_ManipulatorGroup, mpr->parent_mgroup);
 +}
 +
 +#ifdef WITH_PYTHON
 +
 +static void rna_Manipulator_unregister(struct Main *bmain, StructRNA *type);
 +void BPY_RNA_manipulator_wrapper(wmManipulatorType *wgt, void *userdata);
 +
 +static StructRNA *rna_Manipulator_register(
 +        Main *bmain, ReportList *reports, void *data, const char *identifier,
 +        StructValidateFunc validate, StructCallbackFunc call, StructFreeFunc free)
 +{
 +      struct {
 +              char idname[MAX_NAME];
 +      } temp_buffers;
 +
 +      wmManipulatorType dummywt = {NULL};
 +      wmManipulator dummymnp = {NULL};
 +      PointerRNA mnp_ptr;
 +
 +      /* Two sets of functions. */
 +      int have_function[8];
 +
 +      /* setup dummy manipulator & manipulator type to store static properties in */
 +      dummymnp.type = &dummywt;
 +      dummywt.idname = temp_buffers.idname;
 +      RNA_pointer_create(NULL, &RNA_Manipulator, &dummymnp, &mnp_ptr);
 +
 +      /* Clear so we can detect if it's left unset. */
 +      temp_buffers.idname[0] = '\0';
 +
 +      /* validate the python class */
 +      if (validate(&mnp_ptr, data, have_function) != 0)
 +              return NULL;
 +
 +      if (strlen(identifier) >= sizeof(temp_buffers.idname)) {
 +              BKE_reportf(reports, RPT_ERROR, "Registering manipulator class: '%s' is too long, maximum length is %d",
 +                          identifier, (int)sizeof(temp_buffers.idname));
 +              return NULL;
 +      }
 +
 +      /* check if we have registered this manipulator type before, and remove it */
 +      {
 +              const wmManipulatorType *wt = WM_manipulatortype_find(dummywt.idname, true);
 +              if (wt && wt->ext.srna) {
 +                      rna_Manipulator_unregister(bmain, wt->ext.srna);
 +              }
 +      }
 +      if (!RNA_struct_available_or_report(reports, dummywt.idname)) {
 +              return NULL;
 +      }
 +
 +      {   /* allocate the idname */
 +              /* For multiple strings see ManipulatorGroup. */
 +              dummywt.idname = BLI_strdup(temp_buffers.idname);
 +      }
 +
 +      /* create a new manipulator type */
 +      dummywt.ext.srna = RNA_def_struct_ptr(&BLENDER_RNA, dummywt.idname, &RNA_Manipulator);
 +      /* manipulator properties are registered separately */
 +      RNA_def_struct_flag(dummywt.ext.srna, STRUCT_NO_IDPROPERTIES);
 +      dummywt.ext.data = data;
 +      dummywt.ext.call = call;
 +      dummywt.ext.free = free;
 +
 +      {
 +              int i = 0;
 +              dummywt.draw = (have_function[i++]) ? rna_manipulator_draw_cb : NULL;
 +              dummywt.draw_select = (have_function[i++]) ? rna_manipulator_draw_select_cb : NULL;
 +              dummywt.test_select = (have_function[i++]) ? rna_manipulator_test_select_cb : NULL;
 +              dummywt.modal = (have_function[i++]) ? rna_manipulator_modal_cb : NULL;
 +//            dummywt.property_update = (have_function[i++]) ? rna_manipulator_property_update : NULL;
 +//            dummywt.position_get = (have_function[i++]) ? rna_manipulator_position_get : NULL;
 +              dummywt.setup = (have_function[i++]) ? rna_manipulator_setup_cb : NULL;
 +              dummywt.invoke = (have_function[i++]) ? rna_manipulator_invoke_cb : NULL;
 +              dummywt.exit = (have_function[i++]) ? rna_manipulator_exit_cb : NULL;
 +              dummywt.select_refresh = (have_function[i++]) ? rna_manipulator_select_refresh_cb : NULL;
 +
 +              BLI_assert(i == ARRAY_SIZE(have_function));
 +      }
 +
 +      WM_manipulatortype_append_ptr(BPY_RNA_manipulator_wrapper, (void *)&dummywt);
 +
 +      /* update while blender is running */
 +      WM_main_add_notifier(NC_SCREEN | NA_EDITED, NULL);
 +
 +      return dummywt.ext.srna;
 +}
 +
 +static void rna_Manipulator_unregister(struct Main *bmain, StructRNA *type)
 +{
 +      wmManipulatorType *wt = RNA_struct_blender_type_get(type);
 +
 +      if (!wt)
 +              return;
 +
 +      RNA_struct_free_extension(type, &wt->ext);
 +      RNA_struct_free(&BLENDER_RNA, type);
 +
 +      WM_main_add_notifier(NC_SCREEN | NA_EDITED, NULL);
 +
 +      WM_manipulatortype_remove_ptr(NULL, bmain, wt);
 +}
 +
 +static void **rna_Manipulator_instance(PointerRNA *ptr)
 +{
 +      wmManipulator *mpr = ptr->data;
 +      return &mpr->py_instance;
 +}
 +
 +#endif  /* WITH_PYTHON */
 +
 +
 +static StructRNA *rna_Manipulator_refine(PointerRNA *mnp_ptr)
 +{
 +      wmManipulator *mpr = mnp_ptr->data;
 +      return (mpr->type && mpr->type->ext.srna) ? mpr->type->ext.srna : &RNA_Manipulator;
 +}
 +
 +/** \} */
 +
 +/** \name Manipulator Group API
 + * \{ */
 +
 +static wmManipulator *rna_ManipulatorGroup_manipulator_new(
 +        wmManipulatorGroup *mgroup, ReportList *reports, const char *idname)
 +{
 +      const wmManipulatorType *wt = WM_manipulatortype_find(idname, true);
 +      if (wt == NULL) {
 +              BKE_reportf(reports, RPT_ERROR, "ManipulatorType '%s' not known", idname);
 +              return NULL;
 +      }
 +      wmManipulator *mpr = WM_manipulator_new_ptr(wt, mgroup, NULL);
 +      return mpr;
 +}
 +
 +static void rna_ManipulatorGroup_manipulator_remove(
 +        wmManipulatorGroup *mgroup, bContext *C, wmManipulator *mpr)
 +{
 +      WM_manipulator_unlink(&mgroup->manipulators, mgroup->parent_mmap, mpr, C);
 +}
 +
 +static void rna_ManipulatorGroup_manipulator_clear(
 +        wmManipulatorGroup *mgroup, bContext *C)
 +{
 +      while (mgroup->manipulators.first) {
 +              WM_manipulator_unlink(&mgroup->manipulators, mgroup->parent_mmap, mgroup->manipulators.first, C);
 +      }
 +}
 +
 +static void rna_ManipulatorGroup_name_get(PointerRNA *ptr, char *value)
 +{
 +      wmManipulatorGroup *mgroup = ptr->data;
 +      strcpy(value, mgroup->type->name);
 +}
 +
 +static int rna_ManipulatorGroup_name_length(PointerRNA *ptr)
 +{
 +      wmManipulatorGroup *mgroup = ptr->data;
 +      return strlen(mgroup->type->name);
 +}
 +
 +/* just to work around 'const char *' warning and to ensure this is a python op */
 +static void rna_ManipulatorGroup_bl_idname_set(PointerRNA *ptr, const char *value)
 +{
 +      wmManipulatorGroup *data = ptr->data;
 +      char *str = (char *)data->type->idname;
 +      if (!str[0])
 +              BLI_strncpy(str, value, MAX_NAME);    /* utf8 already ensured */
 +      else
 +              assert(!"setting the bl_idname on a non-builtin operator");
 +}
 +
 +static void rna_ManipulatorGroup_bl_label_set(PointerRNA *ptr, const char *value)
 +{
 +      wmManipulatorGroup *data = ptr->data;
 +      char *str = (char *)data->type->name;
 +      if (!str[0])
 +              BLI_strncpy(str, value, MAX_NAME);    /* utf8 already ensured */
 +      else
 +              assert(!"setting the bl_label on a non-builtin operator");
 +}
 +
 +static int rna_ManipulatorGroup_has_reports_get(PointerRNA *ptr)
 +{
 +      wmManipulatorGroup *mgroup = ptr->data;
 +      return (mgroup->reports && mgroup->reports->list.first);
 +}
 +
 +#ifdef WITH_PYTHON
 +
 +static bool rna_manipulatorgroup_poll_cb(const bContext *C, wmManipulatorGroupType *wgt)
 +{
 +
 +      extern FunctionRNA rna_ManipulatorGroup_poll_func;
 +
 +      PointerRNA ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +      void *ret;
 +      int visible;
 +
 +      RNA_pointer_create(NULL, wgt->ext.srna, NULL, &ptr); /* dummy */
 +      func = &rna_ManipulatorGroup_poll_func; /* RNA_struct_find_function(&ptr, "poll"); */
 +
 +      RNA_parameter_list_create(&list, &ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      wgt->ext.call((bContext *)C, &ptr, func, &list);
 +
 +      RNA_parameter_get_lookup(&list, "visible", &ret);
 +      visible = *(int *)ret;
 +
 +      RNA_parameter_list_free(&list);
 +
 +      return visible;
 +}
 +
 +static void rna_manipulatorgroup_setup_cb(const bContext *C, wmManipulatorGroup *mgroup)
 +{
 +      extern FunctionRNA rna_ManipulatorGroup_setup_func;
 +
 +      PointerRNA mgroup_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +
 +      RNA_pointer_create(NULL, mgroup->type->ext.srna, mgroup, &mgroup_ptr);
 +      func = &rna_ManipulatorGroup_setup_func; /* RNA_struct_find_function(&wgroupr, "setup"); */
 +
 +      RNA_parameter_list_create(&list, &mgroup_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      mgroup->type->ext.call((bContext *)C, &mgroup_ptr, func, &list);
 +
 +      RNA_parameter_list_free(&list);
 +}
 +
 +static wmKeyMap *rna_manipulatorgroup_setup_keymap_cb(const wmManipulatorGroupType *wgt, wmKeyConfig *config)
 +{
 +      extern FunctionRNA rna_ManipulatorGroup_setup_keymap_func;
 +      void *ret;
 +
 +      PointerRNA ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +
 +      RNA_pointer_create(NULL, wgt->ext.srna, NULL, &ptr); /* dummy */
 +      func = &rna_ManipulatorGroup_setup_keymap_func; /* RNA_struct_find_function(&wgroupr, "setup_keymap"); */
 +
 +      RNA_parameter_list_create(&list, &ptr, func);
 +      RNA_parameter_set_lookup(&list, "keyconfig", &config);
 +      wgt->ext.call(NULL, &ptr, func, &list);
 +
 +      RNA_parameter_get_lookup(&list, "keymap", &ret);
 +      wmKeyMap *keymap = *(wmKeyMap **)ret;
 +
 +      RNA_parameter_list_free(&list);
 +
 +      return keymap;
 +}
 +
 +static void rna_manipulatorgroup_refresh_cb(const bContext *C, wmManipulatorGroup *mgroup)
 +{
 +      extern FunctionRNA rna_ManipulatorGroup_refresh_func;
 +
 +      PointerRNA mgroup_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +
 +      RNA_pointer_create(NULL, mgroup->type->ext.srna, mgroup, &mgroup_ptr);
 +      func = &rna_ManipulatorGroup_refresh_func; /* RNA_struct_find_function(&wgroupr, "refresh"); */
 +
 +      RNA_parameter_list_create(&list, &mgroup_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      mgroup->type->ext.call((bContext *)C, &mgroup_ptr, func, &list);
 +
 +      RNA_parameter_list_free(&list);
 +}
 +
 +static void rna_manipulatorgroup_draw_prepare_cb(const bContext *C, wmManipulatorGroup *mgroup)
 +{
 +      extern FunctionRNA rna_ManipulatorGroup_draw_prepare_func;
 +
 +      PointerRNA mgroup_ptr;
 +      ParameterList list;
 +      FunctionRNA *func;
 +
 +      RNA_pointer_create(NULL, mgroup->type->ext.srna, mgroup, &mgroup_ptr);
 +      func = &rna_ManipulatorGroup_draw_prepare_func; /* RNA_struct_find_function(&wgroupr, "draw_prepare"); */
 +
 +      RNA_parameter_list_create(&list, &mgroup_ptr, func);
 +      RNA_parameter_set_lookup(&list, "context", &C);
 +      mgroup->type->ext.call((bContext *)C, &mgroup_ptr, func, &list);
 +
 +      RNA_parameter_list_free(&list);
 +}
 +
 +void BPY_RNA_manipulatorgroup_wrapper(wmManipulatorGroupType *wgt, void *userdata);
 +static void rna_ManipulatorGroup_unregister(struct Main *bmain, StructRNA *type);
 +
 +static StructRNA *rna_ManipulatorGroup_register(
 +        Main *bmain, ReportList *reports, void *data, const char *identifier,
 +        StructValidateFunc validate, StructCallbackFunc call, StructFreeFunc free)
 +{
 +      struct {
 +              char name[MAX_NAME];
 +              char idname[MAX_NAME];
 +      } temp_buffers;
 +
 +      wmManipulatorGroupType dummywgt = {NULL};
 +      wmManipulatorGroup dummywg = {NULL};
 +      PointerRNA wgptr;
 +
 +      /* Two sets of functions. */
 +      int have_function[5];
 +
 +      /* setup dummy manipulatorgroup & manipulatorgroup type to store static properties in */
 +      dummywg.type = &dummywgt;
 +      dummywgt.name = temp_buffers.name;
 +      dummywgt.idname = temp_buffers.idname;
 +
 +      RNA_pointer_create(NULL, &RNA_ManipulatorGroup, &dummywg, &wgptr);
 +
 +      /* Clear so we can detect if it's left unset. */
 +      temp_buffers.idname[0] = temp_buffers.name[0] = '\0';
 +
 +      /* validate the python class */
 +      if (validate(&wgptr, data, have_function) != 0)
 +              return NULL;
 +
 +      if (strlen(identifier) >= sizeof(temp_buffers.idname)) {
 +              BKE_reportf(reports, RPT_ERROR, "Registering manipulatorgroup class: '%s' is too long, maximum length is %d",
 +                          identifier, (int)sizeof(temp_buffers.idname));
 +              return NULL;
 +      }
 +
 +      /* check if the area supports widgets */
 +      const struct wmManipulatorMapType_Params wmap_params = {
 +              .spaceid = dummywgt.mmap_params.spaceid,
 +              .regionid = dummywgt.mmap_params.regionid,
 +      };
 +
 +      wmManipulatorMapType *mmap_type = WM_manipulatormaptype_ensure(&wmap_params);
 +      if (mmap_type == NULL) {
 +              BKE_reportf(reports, RPT_ERROR, "Area type does not support manipulators");
 +              return NULL;
 +      }
 +
 +      /* check if we have registered this manipulatorgroup type before, and remove it */
 +      {
 +              wmManipulatorGroupType *wgt = WM_manipulatorgrouptype_find(dummywgt.idname, true);
 +              if (wgt && wgt->ext.srna) {
 +                      rna_ManipulatorGroup_unregister(bmain, wgt->ext.srna);
 +              }
 +      }
 +      if (!RNA_struct_available_or_report(reports, dummywgt.idname)) {
 +              return NULL;
 +      }
 +
 +      {   /* allocate the idname */
 +              const char *strings[] = {
 +                      temp_buffers.idname,
 +                      temp_buffers.name,
 +              };
 +              char *strings_table[ARRAY_SIZE(strings)];
 +              BLI_string_join_array_by_sep_char_with_tableN('\0', strings_table, strings, ARRAY_SIZE(strings));
 +
 +              dummywgt.idname = strings_table[0];  /* allocated string stored here */
 +              dummywgt.name = strings_table[1];
 +              BLI_assert(ARRAY_SIZE(strings) == 2);
 +      }
 +
 +      /* create a new manipulatorgroup type */
 +      dummywgt.ext.srna = RNA_def_struct_ptr(&BLENDER_RNA, dummywgt.idname, &RNA_ManipulatorGroup);
 +      RNA_def_struct_flag(dummywgt.ext.srna, STRUCT_NO_IDPROPERTIES); /* manipulatorgroup properties are registered separately */
 +      dummywgt.ext.data = data;
 +      dummywgt.ext.call = call;
 +      dummywgt.ext.free = free;
 +
 +      /* We used to register widget group types like this, now we do it similar to
 +       * operator types. Thus we should be able to do the same as operator types now. */
 +      dummywgt.poll = (have_function[0]) ? rna_manipulatorgroup_poll_cb : NULL;
 +      dummywgt.setup_keymap =     (have_function[1]) ? rna_manipulatorgroup_setup_keymap_cb : NULL;
 +      dummywgt.setup =            (have_function[2]) ? rna_manipulatorgroup_setup_cb : NULL;
 +      dummywgt.refresh =          (have_function[3]) ? rna_manipulatorgroup_refresh_cb : NULL;
 +      dummywgt.draw_prepare =     (have_function[4]) ? rna_manipulatorgroup_draw_prepare_cb : NULL;
 +
 +      wmManipulatorGroupType *wgt = WM_manipulatorgrouptype_append_ptr(
 +              BPY_RNA_manipulatorgroup_wrapper, (void *)&dummywgt);
 +
 +      if (wgt->flag & WM_MANIPULATORGROUPTYPE_PERSISTENT) {
 +              WM_manipulator_group_type_add_ptr_ex(wgt, mmap_type);
 +
 +              /* update while blender is running */
 +              WM_main_add_notifier(NC_SCREEN | NA_EDITED, NULL);
 +      }
 +
 +      return dummywgt.ext.srna;
 +}
 +
 +static void rna_ManipulatorGroup_unregister(struct Main *bmain, StructRNA *type)
 +{
 +      wmManipulatorGroupType *wgt = RNA_struct_blender_type_get(type);
 +
 +      if (!wgt)
 +              return;
 +
 +      RNA_struct_free_extension(type, &wgt->ext);
 +      RNA_struct_free(&BLENDER_RNA, type);
 +
 +      WM_main_add_notifier(NC_SCREEN | NA_EDITED, NULL);
 +
 +      WM_manipulator_group_type_remove_ptr(bmain, wgt);
 +}
 +
 +static void **rna_ManipulatorGroup_instance(PointerRNA *ptr)
 +{
 +      wmManipulatorGroup *mgroup = ptr->data;
 +      return &mgroup->py_instance;
 +}
 +
 +#endif  /* WITH_PYTHON */
 +
 +static StructRNA *rna_ManipulatorGroup_refine(PointerRNA *mgroup_ptr)
 +{
 +      wmManipulatorGroup *mgroup = mgroup_ptr->data;
 +      return (mgroup->type && mgroup->type->ext.srna) ? mgroup->type->ext.srna : &RNA_ManipulatorGroup;
 +}
 +
 +static void rna_ManipulatorGroup_manipulators_begin(CollectionPropertyIterator *iter, PointerRNA *mgroup_ptr)
 +{
 +      wmManipulatorGroup *mgroup = mgroup_ptr->data;
 +      rna_iterator_listbase_begin(iter, &mgroup->manipulators, NULL);
 +}
 +
 +/** \} */
 +
 +
 +#else /* RNA_RUNTIME */
 +
 +
 +/* ManipulatorGroup.manipulators */
 +static void rna_def_manipulators(BlenderRNA *brna, PropertyRNA *cprop)
 +{
 +      StructRNA *srna;
 +
 +      FunctionRNA *func;
 +      PropertyRNA *parm;
 +
 +      RNA_def_property_srna(cprop, "Manipulators");
 +      srna = RNA_def_struct(brna, "Manipulators", NULL);
 +      RNA_def_struct_sdna(srna, "wmManipulatorGroup");
 +      RNA_def_struct_ui_text(srna, "Manipulators", "Collection of manipulators");
 +
 +      func = RNA_def_function(srna, "new", "rna_ManipulatorGroup_manipulator_new");
 +      RNA_def_function_ui_description(func, "Add manipulator");
 +      RNA_def_function_flag(func, FUNC_USE_REPORTS);
 +      RNA_def_string(func, "type", "Type", 0, "", "Manipulator identifier"); /* optional */
 +      parm = RNA_def_pointer(func, "manipulator", "Manipulator", "", "New manipulator");
 +      RNA_def_function_return(func, parm);
 +
 +      func = RNA_def_function(srna, "remove", "rna_ManipulatorGroup_manipulator_remove");
 +      RNA_def_function_flag(func, FUNC_USE_CONTEXT);
 +      RNA_def_function_ui_description(func, "Delete manipulator");
 +      parm = RNA_def_pointer(func, "manipulator", "Manipulator", "", "New manipulator");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      RNA_def_parameter_clear_flags(parm, PROP_THICK_WRAP, 0);
 +
 +      func = RNA_def_function(srna, "clear", "rna_ManipulatorGroup_manipulator_clear");
 +      RNA_def_function_flag(func, FUNC_USE_CONTEXT);
 +      RNA_def_function_ui_description(func, "Delete all manipulators");
 +}
 +
 +
 +static void rna_def_manipulator(BlenderRNA *brna, PropertyRNA *cprop)
 +{
 +      StructRNA *srna;
 +      PropertyRNA *prop;
 +
 +      FunctionRNA *func;
 +      PropertyRNA *parm;
 +
 +      RNA_def_property_srna(cprop, "Manipulator");
 +      srna = RNA_def_struct(brna, "Manipulator", NULL);
 +      RNA_def_struct_sdna(srna, "wmManipulator");
 +      RNA_def_struct_ui_text(srna, "Manipulator", "Collection of manipulators");
 +      RNA_def_struct_refine_func(srna, "rna_Manipulator_refine");
 +
 +#ifdef WITH_PYTHON
 +      RNA_def_struct_register_funcs(
 +              srna,
 +              "rna_Manipulator_register",
 +              "rna_Manipulator_unregister",
 +              "rna_Manipulator_instance");
 +#endif
 +      RNA_def_struct_translation_context(srna, BLT_I18NCONTEXT_OPERATOR_DEFAULT);
 +
 +      prop = RNA_def_property(srna, "properties", PROP_POINTER, PROP_NONE);
 +      RNA_def_property_flag(prop, PROP_NEVER_NULL);
 +      RNA_def_property_struct_type(prop, "ManipulatorProperties");
 +      RNA_def_property_ui_text(prop, "Properties", "");
 +      RNA_def_property_pointer_funcs(prop, "rna_Manipulator_properties_get", NULL, NULL, NULL);
 +
 +      /* -------------------------------------------------------------------- */
 +      /* Registerable Variables */
 +
 +      RNA_define_verify_sdna(0); /* not in sdna */
 +
 +      prop = RNA_def_property(srna, "bl_idname", PROP_STRING, PROP_NONE);
 +      RNA_def_property_string_sdna(prop, NULL, "type->idname");
 +      RNA_def_property_string_maxlength(prop, MAX_NAME);
 +      RNA_def_property_string_funcs(prop, NULL, NULL, "rna_Manipulator_bl_idname_set");
 +      /* RNA_def_property_clear_flag(prop, PROP_EDITABLE); */
 +      RNA_def_property_flag(prop, PROP_REGISTER);
 +
 +      RNA_define_verify_sdna(1); /* not in sdna */
 +
 +      /* wmManipulator.draw */
 +      func = RNA_def_function(srna, "draw", NULL);
 +      RNA_def_function_ui_description(func, "");
 +      RNA_def_function_flag(func, FUNC_REGISTER);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +
 +      /* wmManipulator.draw_select */
 +      func = RNA_def_function(srna, "draw_select", NULL);
 +      RNA_def_function_ui_description(func, "");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      parm = RNA_def_int(func, "select_id", 0, 0, INT_MAX, "", "", 0, INT_MAX);
 +
 +      /* wmManipulator.test_select */
 +      func = RNA_def_function(srna, "test_select", NULL);
 +      RNA_def_function_ui_description(func, "");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      parm = RNA_def_pointer(func, "event", "Event", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      parm = RNA_def_int(func, "intersect_id", 0, 0, INT_MAX, "", "", 0, INT_MAX);
 +      RNA_def_function_return(func, parm);
 +
 +      /* wmManipulator.handler */
 +      static EnumPropertyItem tweak_actions[] = {
 +              {WM_MANIPULATOR_TWEAK_PRECISE, "PRECISE", 0, "Precise", ""},
 +              {WM_MANIPULATOR_TWEAK_SNAP, "SNAP", 0, "Snap", ""},
 +              {0, NULL, 0, NULL, NULL}
 +      };
 +      func = RNA_def_function(srna, "modal", NULL);
 +      RNA_def_function_ui_description(func, "");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL | FUNC_ALLOW_WRITE);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      parm = RNA_def_pointer(func, "event", "Event", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      /* TODO, shuold be a enum-flag */
 +      parm = RNA_def_enum_flag(func, "tweak", tweak_actions, 0, "Tweak", "");
 +      RNA_def_parameter_flags(parm, 0, PARM_REQUIRED);
 +      parm = RNA_def_enum_flag(func, "result", rna_enum_operator_return_items, OPERATOR_CANCELLED, "result", "");
 +      RNA_def_function_return(func, parm);
 +      /* wmManipulator.property_update */
 +      /* TODO */
 +
 +      /* wmManipulator.setup */
 +      func = RNA_def_function(srna, "setup", NULL);
 +      RNA_def_function_ui_description(func, "");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL | FUNC_ALLOW_WRITE);
 +
 +      /* wmManipulator.invoke */
 +      func = RNA_def_function(srna, "invoke", NULL);
 +      RNA_def_function_ui_description(func, "");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL | FUNC_ALLOW_WRITE);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      parm = RNA_def_pointer(func, "event", "Event", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      parm = RNA_def_enum_flag(func, "result", rna_enum_operator_return_items, OPERATOR_CANCELLED, "result", "");
 +      RNA_def_function_return(func, parm);
 +
 +      /* wmManipulator.exit */
 +      func = RNA_def_function(srna, "exit", NULL);
 +      RNA_def_function_ui_description(func, "");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL | FUNC_ALLOW_WRITE);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      parm = RNA_def_boolean(func, "cancel", 0, "Cancel, otherwise confirm", "");
 +      RNA_def_parameter_flags(parm, 0, PARM_REQUIRED);
 +
 +      /* wmManipulator.cursor_get */
 +      /* TODO */
 +
 +      /* wmManipulator.select_refresh */
 +      func = RNA_def_function(srna, "select_refresh", NULL);
 +      RNA_def_function_ui_description(func, "");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL | FUNC_ALLOW_WRITE);
 +
 +
 +      /* -------------------------------------------------------------------- */
 +      /* Instance Variables */
 +
 +      prop = RNA_def_property(srna, "group", PROP_POINTER, PROP_NONE);
 +      RNA_def_property_clear_flag(prop, PROP_EDITABLE);
 +      RNA_def_property_struct_type(prop, "ManipulatorGroup");
 +      RNA_def_property_pointer_funcs(prop, "rna_Manipulator_group_get", NULL, NULL, NULL);
 +      RNA_def_property_ui_text(prop, "", "Manipulator group this manipulator is a member of");
 +
 +      /* Color & Alpha */
 +      prop = RNA_def_property(srna, "color", PROP_FLOAT, PROP_COLOR);
 +      RNA_def_property_array(prop, 3);
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_color_get", "rna_Manipulator_color_set", NULL);
 +
 +      prop = RNA_def_property(srna, "alpha", PROP_FLOAT, PROP_NONE);
 +      RNA_def_property_ui_text(prop, "Alpha", "");
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_alpha_get", "rna_Manipulator_alpha_set", NULL);
 +      RNA_def_property_range(prop, 0.0f, 1.0f);
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +
 +      /* Color & Alpha (highlight) */
 +      prop = RNA_def_property(srna, "color_highlight", PROP_FLOAT, PROP_COLOR);
 +      RNA_def_property_array(prop, 3);
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_color_hi_get", "rna_Manipulator_color_hi_set", NULL);
 +
 +      prop = RNA_def_property(srna, "alpha_highlight", PROP_FLOAT, PROP_NONE);
 +      RNA_def_property_ui_text(prop, "Alpha", "");
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_alpha_hi_get", "rna_Manipulator_alpha_hi_set", NULL);
 +      RNA_def_property_range(prop, 0.0f, 1.0f);
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +
 +      prop = RNA_def_property(srna, "matrix_space", PROP_FLOAT, PROP_MATRIX);
 +      RNA_def_property_multi_array(prop, 2, rna_matrix_dimsize_4x4);
 +      RNA_def_property_ui_text(prop, "Space Matrix", "");
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_matrix_space_get", "rna_Manipulator_matrix_space_set", NULL);
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +
 +      prop = RNA_def_property(srna, "matrix_basis", PROP_FLOAT, PROP_MATRIX);
 +      RNA_def_property_multi_array(prop, 2, rna_matrix_dimsize_4x4);
 +      RNA_def_property_ui_text(prop, "Basis Matrix", "");
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_matrix_basis_get", "rna_Manipulator_matrix_basis_set", NULL);
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +
 +      prop = RNA_def_property(srna, "matrix_offset", PROP_FLOAT, PROP_MATRIX);
 +      RNA_def_property_multi_array(prop, 2, rna_matrix_dimsize_4x4);
 +      RNA_def_property_ui_text(prop, "Offset Matrix", "");
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_matrix_offset_get", "rna_Manipulator_matrix_offset_set", NULL);
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +
 +      prop = RNA_def_property(srna, "matrix_world", PROP_FLOAT, PROP_MATRIX);
 +      RNA_def_property_clear_flag(prop, PROP_EDITABLE);
 +      RNA_def_property_multi_array(prop, 2, rna_matrix_dimsize_4x4);
 +      RNA_def_property_ui_text(prop, "Final World Matrix", "");
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_matrix_world_get", NULL, NULL);
 +
 +      prop = RNA_def_property(srna, "scale_basis", PROP_FLOAT, PROP_NONE);
 +      RNA_def_property_ui_text(prop, "Scale Basis", "");
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_scale_basis_get", "rna_Manipulator_scale_basis_set", NULL);
 +      RNA_def_property_range(prop, 0.0f, FLT_MAX);
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +
 +      prop = RNA_def_property(srna, "line_width", PROP_FLOAT, PROP_PIXEL);
 +      RNA_def_property_ui_text(prop, "Line Width", "");
 +      RNA_def_property_float_funcs(prop, "rna_Manipulator_line_width_get", "rna_Manipulator_line_width_set", NULL);
 +      RNA_def_property_range(prop, 0.0f, FLT_MAX);
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +
 +      /* wmManipulator.flag */
 +      /* WM_MANIPULATOR_HIDDEN */
 +      prop = RNA_def_property(srna, "hide", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(
 +              prop, "rna_Manipulator_flag_hide_get", "rna_Manipulator_flag_hide_set");
 +      RNA_def_property_ui_text(prop, "Hide", "");
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +      /* WM_MANIPULATOR_DRAW_HOVER */
 +      prop = RNA_def_property(srna, "use_draw_hover", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(
 +              prop, "rna_Manipulator_flag_use_draw_hover_get", "rna_Manipulator_flag_use_draw_hover_set");
 +      RNA_def_property_ui_text(prop, "Draw Hover", "");
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +      /* WM_MANIPULATOR_DRAW_MODAL */
 +      prop = RNA_def_property(srna, "use_draw_modal", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(
 +              prop, "rna_Manipulator_flag_use_draw_modal_get", "rna_Manipulator_flag_use_draw_modal_set");
 +      RNA_def_property_ui_text(prop, "Draw Active", "Draw while dragging");
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +      /* WM_MANIPULATOR_DRAW_VALUE */
 +      prop = RNA_def_property(srna, "use_draw_value", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(
 +              prop, "rna_Manipulator_flag_use_draw_value_get", "rna_Manipulator_flag_use_draw_value_set");
 +      RNA_def_property_ui_text(prop, "Draw Value", "Show an indicator for the current value while dragging");
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +      /* WM_MANIPULATOR_DRAW_OFFSET_SCALE */
 +      prop = RNA_def_property(srna, "use_draw_offset_scale", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(
 +              prop, "rna_Manipulator_flag_use_draw_offset_scale_get", "rna_Manipulator_flag_use_draw_offset_scale_set");
 +      RNA_def_property_ui_text(prop, "Scale Offset", "Scale the offset matrix (use to apply screen-space offset)");
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +      /* WM_MANIPULATOR_DRAW_NO_SCALE (negated) */
 +      prop = RNA_def_property(srna, "use_draw_scale", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(
 +              prop, "rna_Manipulator_flag_use_draw_scale_get", "rna_Manipulator_flag_use_draw_scale_set");
 +      RNA_def_property_ui_text(prop, "Scale", "Use scale when calculating the matrix");
 +      RNA_def_property_update(prop, NC_SCREEN | NA_EDITED, NULL);
 +
 +      /* wmManipulator.state (readonly) */
 +      /* WM_MANIPULATOR_STATE_HIGHLIGHT */
 +      prop = RNA_def_property(srna, "is_highlight", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(prop, "rna_Manipulator_state_is_highlight_get", NULL);
 +      RNA_def_property_ui_text(prop, "Highlight", "");
 +      RNA_def_property_clear_flag(prop, PROP_EDITABLE);
 +      /* WM_MANIPULATOR_STATE_MODAL */
 +      prop = RNA_def_property(srna, "is_modal", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(prop, "rna_Manipulator_state_is_modal_get", NULL);
 +      RNA_def_property_ui_text(prop, "Highlight", "");
 +      RNA_def_property_clear_flag(prop, PROP_EDITABLE);
 +      /* WM_MANIPULATOR_STATE_SELECT */
 +      /* (note that setting is involved, needs to handle array) */
 +      prop = RNA_def_property(srna, "select", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_boolean_funcs(prop, "rna_Manipulator_state_select_get", "rna_Manipulator_state_select_set");
 +      RNA_def_property_ui_text(prop, "Select", "");
 +
 +      RNA_api_manipulator(srna);
 +
 +      srna = RNA_def_struct(brna, "ManipulatorProperties", NULL);
 +      RNA_def_struct_ui_text(srna, "Manipulator Properties", "Input properties of an Manipulator");
 +      RNA_def_struct_refine_func(srna, "rna_ManipulatorProperties_refine");
 +      RNA_def_struct_idprops_func(srna, "rna_ManipulatorProperties_idprops");
 +      RNA_def_struct_flag(srna, STRUCT_NO_DATABLOCK_IDPROPERTIES);
 +}
 +
 +static void rna_def_manipulatorgroup(BlenderRNA *brna)
 +{
 +      StructRNA *srna;
 +      PropertyRNA *prop;
 +
 +      FunctionRNA *func;
 +      PropertyRNA *parm;
 +
 +      srna = RNA_def_struct(brna, "ManipulatorGroup", NULL);
 +      RNA_def_struct_ui_text(srna, "ManipulatorGroup", "Storage of an operator being executed, or registered after execution");
 +      RNA_def_struct_sdna(srna, "wmManipulatorGroup");
 +      RNA_def_struct_refine_func(srna, "rna_ManipulatorGroup_refine");
 +#ifdef WITH_PYTHON
 +      RNA_def_struct_register_funcs(
 +              srna,
 +              "rna_ManipulatorGroup_register",
 +              "rna_ManipulatorGroup_unregister",
 +              "rna_ManipulatorGroup_instance");
 +#endif
 +      RNA_def_struct_translation_context(srna, BLT_I18NCONTEXT_OPERATOR_DEFAULT);
 +
 +      /* -------------------------------------------------------------------- */
 +      /* Registration */
 +
 +      RNA_define_verify_sdna(0); /* not in sdna */
 +
 +      prop = RNA_def_property(srna, "bl_idname", PROP_STRING, PROP_NONE);
 +      RNA_def_property_string_sdna(prop, NULL, "type->idname");
 +      RNA_def_property_string_maxlength(prop, MAX_NAME);
 +      RNA_def_property_string_funcs(prop, NULL, NULL, "rna_ManipulatorGroup_bl_idname_set");
 +      RNA_def_property_flag(prop, PROP_REGISTER);
 +      RNA_def_struct_name_property(srna, prop);
 +
 +      prop = RNA_def_property(srna, "bl_label", PROP_STRING, PROP_NONE);
 +      RNA_def_property_string_sdna(prop, NULL, "type->name");
 +      RNA_def_property_string_maxlength(prop, MAX_NAME); /* else it uses the pointer size! */
 +      RNA_def_property_string_funcs(prop, NULL, NULL, "rna_ManipulatorGroup_bl_label_set");
 +      /* RNA_def_property_clear_flag(prop, PROP_EDITABLE); */
 +      RNA_def_property_flag(prop, PROP_REGISTER);
 +
 +      prop = RNA_def_property(srna, "bl_space_type", PROP_ENUM, PROP_NONE);
 +      RNA_def_property_enum_sdna(prop, NULL, "type->mmap_params.spaceid");
 +      RNA_def_property_enum_items(prop, rna_enum_space_type_items);
 +      RNA_def_property_flag(prop, PROP_REGISTER);
 +      RNA_def_property_ui_text(prop, "Space type", "The space where the panel is going to be used in");
 +
 +      prop = RNA_def_property(srna, "bl_region_type", PROP_ENUM, PROP_NONE);
 +      RNA_def_property_enum_sdna(prop, NULL, "type->mmap_params.regionid");
 +      RNA_def_property_enum_items(prop, rna_enum_region_type_items);
 +      RNA_def_property_flag(prop, PROP_REGISTER);
 +      RNA_def_property_ui_text(prop, "Region Type", "The region where the panel is going to be used in");
 +
 +      /* bl_options */
 +      static EnumPropertyItem manipulatorgroup_flag_items[] = {
 +              {WM_MANIPULATORGROUPTYPE_3D, "3D", 0, "3D",
 +               "Use in 3D viewport"},
 +              {WM_MANIPULATORGROUPTYPE_SCALE, "SCALE", 0, "Scale",
 +               "Scale to respect zoom (otherwise zoom independent draw size)"},
 +              {WM_MANIPULATORGROUPTYPE_DEPTH_3D, "DEPTH_3D", 0, "Depth 3D",
 +               "Supports culled depth by other objects in the view"},
 +              {WM_MANIPULATORGROUPTYPE_SELECT, "SELECT", 0, "Select",
 +               "Supports selection"},
 +              {WM_MANIPULATORGROUPTYPE_PERSISTENT, "PERSISTENT", 0, "Persistent",
 +               ""},
 +              {WM_MANIPULATORGROUPTYPE_DRAW_MODAL_ALL, "SHOW_MODAL_ALL", 0, "Show Modal All",
 +               "Show all while interacting"},
 +              {0, NULL, 0, NULL, NULL}
 +      };
 +      prop = RNA_def_property(srna, "bl_options", PROP_ENUM, PROP_NONE);
 +      RNA_def_property_enum_sdna(prop, NULL, "type->flag");
 +      RNA_def_property_enum_items(prop, manipulatorgroup_flag_items);
 +      RNA_def_property_flag(prop, PROP_REGISTER_OPTIONAL | PROP_ENUM_FLAG);
 +      RNA_def_property_ui_text(prop, "Options",  "Options for this operator type");
 +
 +      RNA_define_verify_sdna(1); /* not in sdna */
 +
 +
 +      /* Functions */
 +
 +      /* poll */
 +      func = RNA_def_function(srna, "poll", NULL);
 +      RNA_def_function_ui_description(func, "Test if the manipulator group can be called or not");
 +      RNA_def_function_flag(func, FUNC_NO_SELF | FUNC_REGISTER_OPTIONAL);
 +      RNA_def_function_return(func, RNA_def_boolean(func, "visible", 1, "", ""));
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +
 +      /* setup_keymap */
 +      func = RNA_def_function(srna, "setup_keymap", NULL);
 +      RNA_def_function_ui_description(
 +              func,
 +              "Initialize keymaps for this manipulator group, use fallback keymap when not present");
 +      RNA_def_function_flag(func, FUNC_NO_SELF | FUNC_REGISTER_OPTIONAL);
 +      parm = RNA_def_pointer(func, "keyconfig", "KeyConfig", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +      /* return */
 +      parm = RNA_def_pointer(func, "keymap", "KeyMap", "", "");
 +      RNA_def_property_flag(parm, PROP_NEVER_NULL);
 +      RNA_def_function_return(func, parm);
 +
 +      /* setup */
 +      func = RNA_def_function(srna, "setup", NULL);
 +      RNA_def_function_ui_description(func, "Create manipulators function for the manipulator group");
 +      RNA_def_function_flag(func, FUNC_REGISTER);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +
 +      /* refresh */
 +      func = RNA_def_function(srna, "refresh", NULL);
 +      RNA_def_function_ui_description(func, "Refresh data (called on common state changes such as selection)");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +
 +      func = RNA_def_function(srna, "draw_prepare", NULL);
 +      RNA_def_function_ui_description(func, "Run before each redraw");
 +      RNA_def_function_flag(func, FUNC_REGISTER_OPTIONAL);
 +      parm = RNA_def_pointer(func, "context", "Context", "", "");
 +      RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED);
 +
 +      /* -------------------------------------------------------------------- */
 +      /* Instance Variables */
 +
 +      prop = RNA_def_property(srna, "name", PROP_STRING, PROP_NONE);
 +      RNA_def_property_clear_flag(prop, PROP_EDITABLE);
 +      RNA_def_property_string_funcs(prop, "rna_ManipulatorGroup_name_get", "rna_ManipulatorGroup_name_length", NULL);
 +      RNA_def_property_ui_text(prop, "Name", "");
 +
 +      prop = RNA_def_property(srna, "has_reports", PROP_BOOLEAN, PROP_NONE);
 +      RNA_def_property_clear_flag(prop, PROP_EDITABLE); /* this is 'virtual' property */
 +      RNA_def_property_boolean_funcs(prop, "rna_ManipulatorGroup_has_reports_get", NULL);
 +      RNA_def_property_ui_text(prop, "Has Reports",
 +                               "ManipulatorGroup has a set of reports (warnings and errors) from last execution");
 +
 +
 +      RNA_define_verify_sdna(0); /* not in sdna */
 +
 +      prop = RNA_def_property(srna, "manipulators", PROP_COLLECTION, PROP_NONE);
 +      RNA_def_property_collection_sdna(prop, NULL, "manipulators", NULL);
 +      RNA_def_property_struct_type(prop, "Manipulator");
 +      RNA_def_property_collection_funcs(
 +              prop, "rna_ManipulatorGroup_manipulators_begin", "rna_iterator_listbase_next",
 +              "rna_iterator_listbase_end", "rna_iterator_listbase_get",
 +              NULL, NULL, NULL, NULL);
 +
 +      RNA_def_property_ui_text(prop, "Manipulators", "List of manipulators in the Manipulator Map");
 +      rna_def_manipulator(brna, prop);
 +      rna_def_manipulators(brna, prop);
 +
 +      RNA_define_verify_sdna(1); /* not in sdna */
 +
 +      RNA_api_manipulatorgroup(srna);
 +}
 +
 +void RNA_def_wm_manipulator(BlenderRNA *brna)
 +{
 +      rna_def_manipulatorgroup(brna);
 +}
 +
 +#endif /* RNA_RUNTIME */
index 07104e034fbf526801dba87cc71a74369cded82f,e991e901bb1140da6db487058d060c89e41f3905..81c6443b2d9f3cfb2d4f9e8fa2b5e82357840f86
  #include "DNA_scene_types.h"
  #include "DNA_screen_types.h"
  #include "DNA_windowmanager_types.h"
 +#include "DNA_workspace_types.h"
  
  #include "BKE_appdir.h"
- #include "BKE_utildefines.h"
  #include "BKE_autoexec.h"
  #include "BKE_blender.h"
  #include "BKE_blendfile.h"