RNA: correct tips
[blender.git] / source / blender / makesrna / intern / rna_object.c
index 531794f7ab4dda78aeb7620d3dbc60c90c6c304d..8255c220592a7184bc8e7f16481125d8ae6bd947 100644 (file)
 #include "DNA_meta_types.h"
 
 #include "BLI_utildefines.h"
+#include "BLI_listbase.h"
 
+#include "BKE_camera.h"
 #include "BKE_paint.h"
-#include "BKE_tessmesh.h"
-#include "BKE_group.h" /* needed for object_in_group() */
+#include "BKE_editmesh.h"
+#include "BKE_group.h" /* needed for BKE_group_object_exists() */
+#include "BKE_object_deform.h"
 
 #include "RNA_access.h"
 #include "RNA_define.h"
 
 #include "rna_internal.h"
 
-#include "BLO_sys_types.h" /* needed for intptr_t used in ED_mesh.h */
+#include "BLI_sys_types.h" /* needed for intptr_t used in ED_mesh.h */
 #include "ED_mesh.h"
 
 #include "WM_api.h"
 #include "WM_types.h"
 
 EnumPropertyItem object_mode_items[] = {
-       {OB_MODE_OBJECT, "OBJECT", ICON_OBJECT_DATAMODE, "Object", ""},
-       {OB_MODE_EDIT, "EDIT", ICON_EDITMODE_HLT, "Edit", ""},
-       {OB_MODE_SCULPT, "SCULPT", ICON_SCULPTMODE_HLT, "Sculpt", ""},
+       {OB_MODE_OBJECT, "OBJECT", ICON_OBJECT_DATAMODE, "Object Mode", ""},
+       {OB_MODE_EDIT, "EDIT", ICON_EDITMODE_HLT, "Edit Mode", ""},
+       {OB_MODE_POSE, "POSE", ICON_POSE_HLT, "Pose Mode", ""},
+       {OB_MODE_SCULPT, "SCULPT", ICON_SCULPTMODE_HLT, "Sculpt Mode", ""},
        {OB_MODE_VERTEX_PAINT, "VERTEX_PAINT", ICON_VPAINT_HLT, "Vertex Paint", ""},
        {OB_MODE_WEIGHT_PAINT, "WEIGHT_PAINT", ICON_WPAINT_HLT, "Weight Paint", ""},
        {OB_MODE_TEXTURE_PAINT, "TEXTURE_PAINT", ICON_TPAINT_HLT, "Texture Paint", ""},
        {OB_MODE_PARTICLE_EDIT, "PARTICLE_EDIT", ICON_PARTICLEMODE, "Particle Edit", ""},
-       {OB_MODE_POSE, "POSE", ICON_POSE_HLT, "Pose", ""},
        {0, NULL, 0, NULL, NULL}
 };
 
@@ -84,8 +87,6 @@ EnumPropertyItem object_empty_drawtype_items[] = {
 
 static EnumPropertyItem parent_type_items[] = {
        {PAROBJECT, "OBJECT", 0, "Object", "The object is parented to an object"},
-       {PARCURVE, "CURVE", 0, "Curve", "The object is parented to a curve"},
-       {PARKEY, "KEY", 0, "Key", ""},
        {PARSKEL, "ARMATURE", 0, "Armature", ""},
        {PARSKEL, "LATTICE", 0, "Lattice", "The object is parented to a lattice"}, /* PARSKEL reuse will give issues */
        {PARVERT1, "VERTEX", 0, "Vertex", "The object is parented to a vertex"},
@@ -106,13 +107,13 @@ static EnumPropertyItem dupli_items[] = {
 #endif
 
 static EnumPropertyItem collision_bounds_items[] = {
-       {OB_BOUND_BOX, "BOX", 0, "Box", ""},
-       {OB_BOUND_SPHERE, "SPHERE", 0, "Sphere", ""},
-       {OB_BOUND_CYLINDER, "CYLINDER", 0, "Cylinder", ""},
-       {OB_BOUND_CONE, "CONE", 0, "Cone", ""},
-       {OB_BOUND_CONVEX_HULL, "CONVEX_HULL", 0, "Convex Hull", ""},
-       {OB_BOUND_TRIANGLE_MESH, "TRIANGLE_MESH", 0, "Triangle Mesh", ""},
-       {OB_BOUND_CAPSULE, "CAPSULE", 0, "Capsule", ""},
+       {OB_BOUND_BOX, "BOX", ICON_MESH_CUBE, "Box", ""},
+       {OB_BOUND_SPHERE, "SPHERE", ICON_MESH_UVSPHERE, "Sphere", ""},
+       {OB_BOUND_CYLINDER, "CYLINDER", ICON_MESH_CYLINDER, "Cylinder", ""},
+       {OB_BOUND_CONE, "CONE", ICON_MESH_CONE, "Cone", ""},
+       {OB_BOUND_CONVEX_HULL, "CONVEX_HULL", ICON_MESH_ICOSPHERE, "Convex Hull", ""},
+       {OB_BOUND_TRIANGLE_MESH, "TRIANGLE_MESH", ICON_MESH_MONKEY, "Triangle Mesh", ""},
+       {OB_BOUND_CAPSULE, "CAPSULE", ICON_MESH_CAPSULE, "Capsule", ""},
        /*{OB_DYN_MESH, "DYNAMIC_MESH", 0, "Dynamic Mesh", ""}, */
        {0, NULL, 0, NULL, NULL}
 };
@@ -165,6 +166,14 @@ EnumPropertyItem object_axis_items[] = {
        {0, NULL, 0, NULL, NULL}
 };
 
+/* for general use (not just object) */
+EnumPropertyItem object_axis_unsigned_items[] = {
+       {0, "X", 0, "X", ""},
+       {1, "Y", 0, "Y", ""},
+       {2, "Z", 0, "Z", ""},
+       {0, NULL, 0, NULL, NULL}
+};
+
 #ifdef RNA_RUNTIME
 
 #include "BLI_math.h"
@@ -181,6 +190,7 @@ EnumPropertyItem object_axis_items[] = {
 #include "BKE_curve.h"
 #include "BKE_depsgraph.h"
 #include "BKE_effect.h"
+#include "BKE_global.h"
 #include "BKE_key.h"
 #include "BKE_object.h"
 #include "BKE_material.h"
@@ -189,7 +199,6 @@ EnumPropertyItem object_axis_items[] = {
 #include "BKE_scene.h"
 #include "BKE_deform.h"
 
-#include "ED_mesh.h"
 #include "ED_object.h"
 #include "ED_particle.h"
 #include "ED_curve.h"
@@ -203,7 +212,7 @@ static void rna_Object_internal_update(Main *UNUSED(bmain), Scene *UNUSED(scene)
 static void rna_Object_matrix_world_update(Main *bmain, Scene *scene, PointerRNA *ptr)
 {
        /* don't use compat so we get predictable rotation */
-       BKE_object_apply_mat4(ptr->id.data, ((Object *)ptr->id.data)->obmat, FALSE, TRUE);
+       BKE_object_apply_mat4(ptr->id.data, ((Object *)ptr->id.data)->obmat, false, true);
        rna_Object_internal_update(bmain, scene, ptr);
 }
 
@@ -215,35 +224,27 @@ static void rna_Object_hide_update(Main *bmain, Scene *UNUSED(scene), PointerRNA
 static void rna_Object_matrix_local_get(PointerRNA *ptr, float values[16])
 {
        Object *ob = ptr->id.data;
-
-       if (ob->parent) {
-               float invmat[4][4]; /* for inverse of parent's matrix */
-               invert_m4_m4(invmat, ob->parent->obmat);
-               mult_m4_m4m4((float(*)[4])values, invmat, ob->obmat);
-       }
-       else {
-               copy_m4_m4((float(*)[4])values, ob->obmat);
-       }
+       BKE_object_matrix_local_get(ob, (float(*)[4])values);
 }
 
 static void rna_Object_matrix_local_set(PointerRNA *ptr, const float values[16])
 {
        Object *ob = ptr->id.data;
+       float local_mat[4][4];
 
-       /* localspace matrix is truly relative to the parent, but parameters
-        * stored in object are relative to parentinv matrix.  Undo the parent
-        * inverse part before updating obmat and calling apply_obmat() */
+       /* localspace matrix is truly relative to the parent, but parameters stored in object are
+        * relative to parentinv matrix. Undo the parent inverse part before applying it as local matrix. */
        if (ob->parent) {
                float invmat[4][4];
                invert_m4_m4(invmat, ob->parentinv);
-               mult_m4_m4m4(ob->obmat, invmat, (float(*)[4])values);
+               mul_m4_m4m4(local_mat, invmat, (float(*)[4])values);
        }
        else {
-               copy_m4_m4(ob->obmat, (float(*)[4])values);
+               copy_m4_m4(local_mat, (float(*)[4])values);
        }
 
-       /* don't use compat so we get predictable rotation */
-       BKE_object_apply_mat4(ob, ob->obmat, FALSE, FALSE);
+       /* don't use compat so we get predictable rotation, and do not use parenting either, because it's a local matrix! */
+       BKE_object_apply_mat4(ob, local_mat, false, false);
 }
 
 static void rna_Object_matrix_basis_get(PointerRNA *ptr, float values[16])
@@ -255,7 +256,7 @@ static void rna_Object_matrix_basis_get(PointerRNA *ptr, float values[16])
 static void rna_Object_matrix_basis_set(PointerRNA *ptr, const float values[16])
 {
        Object *ob = ptr->id.data;
-       BKE_object_apply_mat4(ob, (float(*)[4])values, FALSE, FALSE);
+       BKE_object_apply_mat4(ob, (float(*)[4])values, false, false);
 }
 
 void rna_Object_internal_update_data(Main *UNUSED(bmain), Scene *UNUSED(scene), PointerRNA *ptr)
@@ -273,9 +274,9 @@ static void rna_Object_active_shape_update(Main *bmain, Scene *scene, PointerRNA
                switch (ob->type) {
                        case OB_MESH:
                                EDBM_mesh_load(ob);
-                               EDBM_mesh_make(scene->toolsettings, scene, ob);
+                               EDBM_mesh_make(scene->toolsettings, ob);
                                EDBM_mesh_normals_update(((Mesh *)ob->data)->edit_btmesh);
-                               BMEdit_RecalcTessellation(((Mesh *)ob->data)->edit_btmesh);
+                               BKE_editmesh_tessface_calc(((Mesh *)ob->data)->edit_btmesh);
                                break;
                        case OB_CURVE:
                        case OB_SURF:
@@ -368,8 +369,14 @@ static void rna_Object_data_set(PointerRNA *ptr, PointerRNA value)
        Object *ob = (Object *)ptr->data;
        ID *id = value.data;
 
-       if (id == NULL || ob->mode & OB_MODE_EDIT)
+       if (ob->mode & OB_MODE_EDIT) {
                return;
+       }
+
+       /* assigning NULL only for empties */
+       if ((id == NULL) && (ob->type != OB_EMPTY)) {
+               return;
+       }
 
        if (ob->type == OB_EMPTY) {
                if (ob->data) {
@@ -377,26 +384,25 @@ static void rna_Object_data_set(PointerRNA *ptr, PointerRNA value)
                        ob->data = NULL;
                }
 
-               if (id && GS(id->name) == ID_IM) {
+               if (!id || GS(id->name) == ID_IM) {
                        id_us_plus(id);
                        ob->data = id;
                }
        }
        else if (ob->type == OB_MESH) {
-               set_mesh(ob, (Mesh *)id);
+               BKE_mesh_assign_object(ob, (Mesh *)id);
        }
        else {
                if (ob->data) {
                        id_us_min((ID *)ob->data);
                }
-               if (id) {
-                       /* no need to type-check here ID. this is done in the _typef() function */
-                       BLI_assert(OB_DATA_SUPPORT_ID(GS(id->name)));
-                       id_us_plus(id);
-               }
+
+               /* no need to type-check here ID. this is done in the _typef() function */
+               BLI_assert(OB_DATA_SUPPORT_ID(GS(id->name)));
+               id_us_plus(id);
 
                ob->data = id;
-               test_object_materials(id);
+               test_object_materials(G.main, id);
 
                if (GS(id->name) == ID_CU)
                        BKE_curve_type_test(ob);
@@ -448,7 +454,7 @@ static void rna_Object_parent_type_set(PointerRNA *ptr, int value)
 }
 
 static EnumPropertyItem *rna_Object_parent_type_itemf(bContext *UNUSED(C), PointerRNA *ptr,
-                                                      PropertyRNA *UNUSED(prop), int *free)
+                                                      PropertyRNA *UNUSED(prop), bool *r_free)
 {
        Object *ob = (Object *)ptr->data;
        EnumPropertyItem *item = NULL;
@@ -459,36 +465,45 @@ static EnumPropertyItem *rna_Object_parent_type_itemf(bContext *UNUSED(C), Point
        if (ob->parent) {
                Object *par = ob->parent;
                
-               if (par->type == OB_CURVE)
-                       RNA_enum_items_add_value(&item, &totitem, parent_type_items, PARCURVE);
-               else if (par->type == OB_LATTICE)
+               if (par->type == OB_LATTICE) {
                        /* special hack: prevents this overriding others */
-                       RNA_enum_items_add_value(&item, &totitem, &parent_type_items[4], PARSKEL);
+                       RNA_enum_items_add_value(&item, &totitem, &parent_type_items[2], PARSKEL);
+               }
                else if (par->type == OB_ARMATURE) {
                        /* special hack: prevents this being overrided */
-                       RNA_enum_items_add_value(&item, &totitem, &parent_type_items[3], PARSKEL);
+                       RNA_enum_items_add_value(&item, &totitem, &parent_type_items[1], PARSKEL);
                        RNA_enum_items_add_value(&item, &totitem, parent_type_items, PARBONE);
                }
-               else if (par->type == OB_MESH) {
+
+               if (OB_TYPE_SUPPORT_PARVERT(par->type)) {
                        RNA_enum_items_add_value(&item, &totitem, parent_type_items, PARVERT1);
                        RNA_enum_items_add_value(&item, &totitem, parent_type_items, PARVERT3);
                }
        }
 
        RNA_enum_item_end(&item, &totitem);
-       *free = 1;
+       *r_free = true;
 
        return item;
 }
 
+static void rna_Object_empty_draw_type_set(PointerRNA *ptr, int value)
+{
+       Object *ob = (Object *)ptr->data;
+
+       BKE_object_empty_draw_type_set(ob, value);
+}
+
 static EnumPropertyItem *rna_Object_collision_bounds_itemf(bContext *UNUSED(C), PointerRNA *ptr,
-                                                           PropertyRNA *UNUSED(prop), int *free)
+                                                           PropertyRNA *UNUSED(prop), bool *r_free)
 {
        Object *ob = (Object *)ptr->data;
        EnumPropertyItem *item = NULL;
        int totitem = 0;
 
-       RNA_enum_items_add_value(&item, &totitem, collision_bounds_items, OB_BOUND_TRIANGLE_MESH);
+       if (ob->body_type != OB_BODY_TYPE_CHARACTER) {
+               RNA_enum_items_add_value(&item, &totitem, collision_bounds_items, OB_BOUND_TRIANGLE_MESH);
+       }
        RNA_enum_items_add_value(&item, &totitem, collision_bounds_items, OB_BOUND_CONVEX_HULL);
 
        if (ob->body_type != OB_BODY_TYPE_SOFT) {
@@ -500,7 +515,7 @@ static EnumPropertyItem *rna_Object_collision_bounds_itemf(bContext *UNUSED(C),
        }
 
        RNA_enum_item_end(&item, &totitem);
-       *free = 1;
+       *r_free = true;
 
        return item;
 }
@@ -520,7 +535,7 @@ static void rna_Object_dup_group_set(PointerRNA *ptr, PointerRNA value)
        /* must not let this be set if the object belongs in this group already,
         * thus causing a cycle/infinite-recursion leading to crashes on load [#25298]
         */
-       if (object_in_group(ob, grp) == 0)
+       if (BKE_group_object_exists(grp, ob) == 0)
                ob->dup_group = grp;
        else
                BKE_report(NULL, RPT_ERROR,
@@ -560,12 +575,13 @@ static void rna_Object_active_vertex_group_index_set(PointerRNA *ptr, int value)
        ob->actdef = value + 1;
 }
 
-static void rna_Object_active_vertex_group_index_range(PointerRNA *ptr, int *min, int *max, int *softmin, int *softmax)
+static void rna_Object_active_vertex_group_index_range(PointerRNA *ptr, int *min, int *max,
+                                                       int *UNUSED(softmin), int *UNUSED(softmax))
 {
        Object *ob = (Object *)ptr->id.data;
 
        *min = 0;
-       *max = max_ii(0, BLI_countlist(&ob->defbase) - 1);
+       *max = max_ii(0, BLI_listbase_count(&ob->defbase) - 1);
 }
 
 void rna_object_vgroup_name_index_get(PointerRNA *ptr, char *value, int index)
@@ -619,7 +635,7 @@ void rna_object_uvlayer_name_set(PointerRNA *ptr, const char *value, char *resul
                for (a = 0; a < me->pdata.totlayer; a++) {
                        layer = &me->pdata.layers[a];
 
-                       if (layer->type == CD_MTEXPOLY && strcmp(layer->name, value) == 0) {
+                       if (layer->type == CD_MTEXPOLY && STREQ(layer->name, value)) {
                                BLI_strncpy(result, value, maxlen);
                                return;
                        }
@@ -642,7 +658,7 @@ void rna_object_vcollayer_name_set(PointerRNA *ptr, const char *value, char *res
                for (a = 0; a < me->fdata.totlayer; a++) {
                        layer = &me->fdata.layers[a];
 
-                       if (layer->type == CD_MCOL && strcmp(layer->name, value) == 0) {
+                       if (layer->type == CD_MCOL && STREQ(layer->name, value)) {
                                BLI_strncpy(result, value, maxlen);
                                return;
                        }
@@ -671,7 +687,8 @@ static void rna_Object_active_material_index_set(PointerRNA *ptr, int value)
        }
 }
 
-static void rna_Object_active_material_index_range(PointerRNA *ptr, int *min, int *max, int *softmin, int *softmax)
+static void rna_Object_active_material_index_range(PointerRNA *ptr, int *min, int *max,
+                                                   int *UNUSED(softmin), int *UNUSED(softmax))
 {
        Object *ob = (Object *)ptr->id.data;
        *min = 0;
@@ -693,15 +710,31 @@ static void rna_Object_active_material_set(PointerRNA *ptr, PointerRNA value)
        Object *ob = (Object *)ptr->id.data;
 
        DAG_id_tag_update(value.data, 0);
-       assign_material(ob, value.data, ob->actcol, BKE_MAT_ASSIGN_USERPREF);
+       assign_material(ob, value.data, ob->actcol, BKE_MAT_ASSIGN_EXISTING);
+}
+
+static int rna_Object_active_material_editable(PointerRNA *ptr)
+{
+       Object *ob = (Object *)ptr->id.data;
+       bool is_editable;
+
+       if ((ob->matbits == NULL) || (ob->actcol == 0) || ob->matbits[ob->actcol - 1]) {
+               is_editable = (ob->id.lib == NULL);
+       }
+       else {
+               is_editable = ob->data ? (((ID *)ob->data)->lib == NULL) : false;
+       }
+
+       return is_editable ? PROP_EDITABLE : 0;
 }
 
+
 static void rna_Object_active_particle_system_index_range(PointerRNA *ptr, int *min, int *max,
-                                                          int *softmin, int *softmax)
+                                                          int *UNUSED(softmin), int *UNUSED(softmax))
 {
        Object *ob = (Object *)ptr->id.data;
        *min = 0;
-       *max = max_ii(0, BLI_countlist(&ob->particlesystem) - 1);
+       *max = max_ii(0, BLI_listbase_count(&ob->particlesystem) - 1);
 }
 
 static int rna_Object_active_particle_system_index_get(PointerRNA *ptr)
@@ -740,7 +773,7 @@ static void rna_Object_rotation_axis_angle_set(PointerRNA *ptr, const float *val
        
        /* for now, assume that rotation mode is axis-angle */
        ob->rotAngle = value[0];
-       copy_v3_v3(ob->rotAxis, (float *)&value[1]);
+       copy_v3_v3(ob->rotAxis, &value[1]);
        
        /* TODO: validate axis? */
 }
@@ -849,7 +882,7 @@ static void rna_MaterialSlot_material_set(PointerRNA *ptr, PointerRNA value)
        Object *ob = (Object *)ptr->id.data;
        int index = (Material **)ptr->data - ob->mat;
 
-       assign_material(ob, value.data, index + 1, BKE_MAT_ASSIGN_USERPREF);
+       assign_material(ob, value.data, index + 1, BKE_MAT_ASSIGN_EXISTING);
 }
 
 static int rna_MaterialSlot_link_get(PointerRNA *ptr)
@@ -906,7 +939,19 @@ static void rna_MaterialSlot_name_get(PointerRNA *ptr, char *str)
 static void rna_MaterialSlot_update(Main *bmain, Scene *scene, PointerRNA *ptr)
 {
        rna_Object_internal_update(bmain, scene, ptr);
+
        WM_main_add_notifier(NC_OBJECT | ND_OB_SHADING, ptr->id.data);
+       WM_main_add_notifier(NC_MATERIAL | ND_SHADING_LINKS, NULL);
+       DAG_relations_tag_update(bmain);
+}
+
+static char *rna_MaterialSlot_path(PointerRNA *ptr)
+{
+       Object *ob = (Object *)ptr->id.data;
+       int index = (Material **)ptr->data - ob->mat;
+
+       /* from armature... */
+       return BLI_sprintfN("material_slots[%d]", index);
 }
 
 /* why does this have to be so complicated?, can't all this crap be
@@ -958,12 +1003,12 @@ static int rna_GameObjectSettings_physics_type_get(PointerRNA *ptr)
 static void rna_GameObjectSettings_physics_type_set(PointerRNA *ptr, int value)
 {
        Object *ob = (Object *)ptr->id.data;
-       const int was_navmesh = (ob->gameflag & OB_NAVMESH);
+       const int gameflag_prev = ob->gameflag;
        ob->body_type = value;
 
        switch (ob->body_type) {
                case OB_BODY_TYPE_SENSOR:
-                       ob->gameflag |= OB_SENSOR | OB_COLLISION | OB_GHOST;
+                       ob->gameflag |= OB_SENSOR | OB_COLLISION;
                        ob->gameflag &= ~(OB_OCCLUDER | OB_CHARACTER | OB_DYNAMIC | OB_RIGID_BODY | OB_SOFT_BODY | OB_ACTOR |
                                          OB_ANISOTROPIC_FRICTION | OB_DO_FH | OB_ROT_FH | OB_COLLISION_RESPONSE | OB_NAVMESH);
                        break;
@@ -985,10 +1030,10 @@ static void rna_GameObjectSettings_physics_type_set(PointerRNA *ptr, int value)
                        ob->gameflag &= ~(OB_SENSOR | OB_RIGID_BODY | OB_SOFT_BODY | OB_COLLISION | OB_CHARACTER | OB_OCCLUDER | OB_DYNAMIC | OB_NAVMESH);
                        break;
                case OB_BODY_TYPE_CHARACTER:
-                       ob->gameflag |= OB_COLLISION | OB_GHOST | OB_CHARACTER;
+                       ob->gameflag |= OB_COLLISION | OB_CHARACTER;
                        ob->gameflag &= ~(OB_SENSOR | OB_OCCLUDER | OB_DYNAMIC | OB_RIGID_BODY | OB_SOFT_BODY | OB_ACTOR |
-                                     OB_ANISOTROPIC_FRICTION | OB_DO_FH | OB_ROT_FH | OB_COLLISION_RESPONSE | OB_NAVMESH);
-               break;
+                                         OB_ANISOTROPIC_FRICTION | OB_DO_FH | OB_ROT_FH | OB_COLLISION_RESPONSE | OB_NAVMESH);
+                       break;
                case OB_BODY_TYPE_STATIC:
                        ob->gameflag |= OB_COLLISION;
                        ob->gameflag &= ~(OB_DYNAMIC | OB_RIGID_BODY | OB_SOFT_BODY | OB_OCCLUDER | OB_CHARACTER | OB_SENSOR | OB_NAVMESH);
@@ -1016,7 +1061,7 @@ static void rna_GameObjectSettings_physics_type_set(PointerRNA *ptr, int value)
                        break;
        }
 
-       if (was_navmesh != (ob->gameflag & OB_NAVMESH)) {
+       if ((gameflag_prev & OB_NAVMESH) != (ob->gameflag & OB_NAVMESH)) {
                if (ob->type == OB_MESH) {
                        /* this is needed to refresh the derived meshes draw func */
                        DAG_id_tag_update(ptr->id.data, OB_RECALC_DATA);
@@ -1089,7 +1134,7 @@ static void rna_GameObjectSettings_state_get(PointerRNA *ptr, int *values)
 
        memset(values, 0, sizeof(int) * OB_MAX_STATES);
        for (i = 0; i < OB_MAX_STATES; i++) {
-               values[i] = (ob->state & (1 << i)) | all_states;
+               values[i] = (ob->state & (1 << i)) ? 1 : 0 | all_states;
        }
 }
 
@@ -1134,7 +1179,7 @@ static void rna_GameObjectSettings_col_group_get(PointerRNA *ptr, int *values)
        int i;
 
        for (i = 0; i < OB_MAX_COL_MASKS; i++) {
-               values[i] = (ob->col_group & (1 << i));
+               values[i] = (ob->col_group & (1 << i)) != 0;
        }
 }
 
@@ -1163,7 +1208,7 @@ static void rna_GameObjectSettings_col_mask_get(PointerRNA *ptr, int *values)
        int i;
 
        for (i = 0; i < OB_MAX_COL_MASKS; i++) {
-               values[i] = (ob->col_mask & (1 << i));
+               values[i] = (ob->col_mask & (1 << i)) != 0;
        }
 }
 
@@ -1187,14 +1232,15 @@ static void rna_GameObjectSettings_col_mask_set(PointerRNA *ptr, const int *valu
 }
 
 
-static void rna_Object_active_shape_key_index_range(PointerRNA *ptr, int *min, int *max, int *softmin, int *softmax)
+static void rna_Object_active_shape_key_index_range(PointerRNA *ptr, int *min, int *max,
+                                                    int *UNUSED(softmin), int *UNUSED(softmax))
 {
        Object *ob = (Object *)ptr->id.data;
        Key *key = BKE_key_from_object(ob);
 
        *min = 0;
        if (key) {
-               *max = BLI_countlist(&key->block) - 1;
+               *max = BLI_listbase_count(&key->block) - 1;
                if (*max < 0) *max = 0;
        }
        else {
@@ -1259,20 +1305,20 @@ static PointerRNA rna_Object_collision_get(PointerRNA *ptr)
 static PointerRNA rna_Object_active_constraint_get(PointerRNA *ptr)
 {
        Object *ob = (Object *)ptr->id.data;
-       bConstraint *con = BKE_constraints_get_active(&ob->constraints);
+       bConstraint *con = BKE_constraints_active_get(&ob->constraints);
        return rna_pointer_inherit_refine(ptr, &RNA_Constraint, con);
 }
 
 static void rna_Object_active_constraint_set(PointerRNA *ptr, PointerRNA value)
 {
        Object *ob = (Object *)ptr->id.data;
-       BKE_constraints_set_active(&ob->constraints, (bConstraint *)value.data);
+       BKE_constraints_active_set(&ob->constraints, (bConstraint *)value.data);
 }
 
 static bConstraint *rna_Object_constraints_new(Object *object, int type)
 {
        WM_main_add_notifier(NC_OBJECT | ND_CONSTRAINT | NA_ADDED, object);
-       return BKE_add_ob_constraint(object, NULL, type);
+       return BKE_constraint_add_for_object(object, NULL, type);
 }
 
 static void rna_Object_constraints_remove(Object *object, ReportList *reports, PointerRNA *con_ptr)
@@ -1283,7 +1329,7 @@ static void rna_Object_constraints_remove(Object *object, ReportList *reports, P
                return;
        }
 
-       BKE_remove_constraint(&object->constraints, con);
+       BKE_constraint_remove(&object->constraints, con);
        RNA_POINTER_INVALIDATE(con_ptr);
 
        ED_object_constraint_update(object);
@@ -1293,7 +1339,7 @@ static void rna_Object_constraints_remove(Object *object, ReportList *reports, P
 
 static void rna_Object_constraints_clear(Object *object)
 {
-       BKE_free_constraints(&object->constraints);
+       BKE_constraints_free(&object->constraints);
 
        ED_object_constraint_update(object);
        ED_object_constraint_set_active(object, NULL);
@@ -1310,7 +1356,7 @@ static ModifierData *rna_Object_modifier_new(Object *object, bContext *C, Report
 static void rna_Object_modifier_remove(Object *object, bContext *C, ReportList *reports, PointerRNA *md_ptr)
 {
        ModifierData *md = md_ptr->data;
-       if (ED_object_modifier_remove(reports, CTX_data_main(C), object, md) == FALSE) {
+       if (ED_object_modifier_remove(reports, CTX_data_main(C), object, md) == false) {
                /* error is already set */
                return;
        }
@@ -1335,14 +1381,14 @@ static void rna_Object_boundbox_get(PointerRNA *ptr, float *values)
                memcpy(values, bb->vec, sizeof(bb->vec));
        }
        else {
-               fill_vn_fl(values, sizeof(bb->vec) / sizeof(float), 0.0f);
+               copy_vn_fl(values, sizeof(bb->vec) / sizeof(float), 0.0f);
        }
 
 }
 
 static bDeformGroup *rna_Object_vgroup_new(Object *ob, const char *name)
 {
-       bDeformGroup *defgroup = ED_vgroup_add_name(ob, name);
+       bDeformGroup *defgroup = BKE_object_defgroup_add_name(ob, name);
 
        WM_main_add_notifier(NC_OBJECT | ND_DRAW, ob);
 
@@ -1357,7 +1403,7 @@ static void rna_Object_vgroup_remove(Object *ob, ReportList *reports, PointerRNA
                return;
        }
 
-       ED_vgroup_delete(ob, defgroup);
+       BKE_object_defgroup_remove(ob, defgroup);
        RNA_POINTER_INVALIDATE(defgroup_ptr);
 
        WM_main_add_notifier(NC_OBJECT | ND_DRAW, ob);
@@ -1365,7 +1411,7 @@ static void rna_Object_vgroup_remove(Object *ob, ReportList *reports, PointerRNA
 
 static void rna_Object_vgroup_clear(Object *ob)
 {
-       ED_vgroup_clear(ob);
+       BKE_object_defgroup_remove_all(ob);
 
        WM_main_add_notifier(NC_OBJECT | ND_DRAW, ob);
 }
@@ -1375,7 +1421,7 @@ static void rna_VertexGroup_vertex_add(ID *id, bDeformGroup *def, ReportList *re
 {
        Object *ob = (Object *)id;
 
-       if (ED_vgroup_object_is_edit_mode(ob)) {
+       if (BKE_object_is_in_editmode_vgroup(ob)) {
                BKE_report(reports, RPT_ERROR, "VertexGroup.add(): cannot be called while object is in edit mode");
                return;
        }
@@ -1390,7 +1436,7 @@ static void rna_VertexGroup_vertex_remove(ID *id, bDeformGroup *dg, ReportList *
 {
        Object *ob = (Object *)id;
 
-       if (ED_vgroup_object_is_edit_mode(ob)) {
+       if (BKE_object_is_in_editmode_vgroup(ob)) {
                BKE_report(reports, RPT_ERROR, "VertexGroup.remove(): cannot be called while object is in edit mode");
                return;
        }
@@ -1437,6 +1483,11 @@ int rna_Camera_object_poll(PointerRNA *UNUSED(ptr), PointerRNA value)
        return ((Object *)value.id.data)->type == OB_CAMERA;
 }
 
+int rna_Lamp_object_poll(PointerRNA *UNUSED(ptr), PointerRNA value)
+{
+       return ((Object *)value.id.data)->type == OB_LAMP;
+}
+
 int rna_DupliObject_index_get(PointerRNA *ptr)
 {
        DupliObject *dob = (DupliObject *)ptr->data;
@@ -1449,9 +1500,17 @@ int rna_Object_use_dynamic_topology_sculpting_get(PointerRNA *ptr)
        return (ss && ss->bm);
 }
 
-#else
+static void rna_Object_lod_distance_update(Main *UNUSED(bmain), Scene *UNUSED(scene), PointerRNA *ptr)
+{
+       Object *ob = (Object *)ptr->id.data;
 
-static int rna_matrix_dimsize_4x4[] = {4, 4};
+#ifdef WITH_GAMEENGINE
+       BKE_object_lod_sort(ob);
+#else
+       (void)ob;
+#endif
+}
+#else
 
 static void rna_def_vertex_group(BlenderRNA *brna)
 {
@@ -1553,6 +1612,8 @@ static void rna_def_material_slot(BlenderRNA *brna)
        RNA_def_property_ui_text(prop, "Name", "Material slot name");
        RNA_def_property_clear_flag(prop, PROP_EDITABLE);
        RNA_def_struct_name_property(srna, prop);
+
+       RNA_def_struct_path_func(srna, "rna_MaterialSlot_path");
 }
 
 static void rna_def_object_game_settings(BlenderRNA *brna)
@@ -1566,7 +1627,7 @@ static void rna_def_object_game_settings(BlenderRNA *brna)
                {OB_BODY_TYPE_DYNAMIC, "DYNAMIC", 0, "Dynamic", "Linear physics"},
                {OB_BODY_TYPE_RIGID, "RIGID_BODY", 0, "Rigid Body", "Linear and angular physics"},
                {OB_BODY_TYPE_SOFT, "SOFT_BODY", 0, "Soft Body", "Soft body"},
-               {OB_BODY_TYPE_OCCLUDER, "OCCLUDE", 0, "Occlude", "Occluder for optimizing scene rendering"},
+               {OB_BODY_TYPE_OCCLUDER, "OCCLUDER", 0, "Occluder", "Occluder for optimizing scene rendering"},
                {OB_BODY_TYPE_SENSOR, "SENSOR", 0, "Sensor",
                                      "Collision Sensor, detects static and dynamic objects but not the other "
                                      "collision sensor objects"},
@@ -1619,11 +1680,16 @@ static void rna_def_object_game_settings(BlenderRNA *brna)
        prop = RNA_def_property(srna, "physics_type", PROP_ENUM, PROP_NONE);
        RNA_def_property_enum_sdna(prop, NULL, "body_type");
        RNA_def_property_enum_items(prop, body_type_items);
+       RNA_def_property_enum_default(prop, OB_BODY_TYPE_STATIC);
        RNA_def_property_enum_funcs(prop, "rna_GameObjectSettings_physics_type_get",
                                    "rna_GameObjectSettings_physics_type_set", NULL);
        RNA_def_property_ui_text(prop, "Physics Type", "Select the type of physical representation");
        RNA_def_property_update(prop, NC_LOGIC, NULL);
 
+       prop = RNA_def_property(srna, "use_record_animation", PROP_BOOLEAN, PROP_NONE);
+       RNA_def_property_boolean_sdna(prop, NULL, "gameflag", OB_RECORD_ANIMATION);
+       RNA_def_property_ui_text(prop, "Record Animation", "Record animation objects without physics");
+
        prop = RNA_def_property(srna, "use_actor", PROP_BOOLEAN, PROP_NONE);
        RNA_def_property_boolean_sdna(prop, NULL, "gameflag", OB_ACTOR);
        RNA_def_property_ui_text(prop, "Actor", "Object is detected by the Near and Radar sensor");
@@ -1634,12 +1700,14 @@ static void rna_def_object_game_settings(BlenderRNA *brna)
 
        prop = RNA_def_property(srna, "mass", PROP_FLOAT, PROP_NONE);
        RNA_def_property_range(prop, 0.01, 10000.0);
+       RNA_def_property_float_default(prop, 1.0f);
        RNA_def_property_ui_text(prop, "Mass", "Mass of the object");
 
        prop = RNA_def_property(srna, "radius", PROP_FLOAT, PROP_NONE | PROP_UNIT_LENGTH);
        RNA_def_property_float_sdna(prop, NULL, "inertia");
        RNA_def_property_range(prop, 0.01f, FLT_MAX);
        RNA_def_property_ui_range(prop, 0.01f, 10.0f, 1, 3);
+       RNA_def_property_float_default(prop, 1.0f);
        RNA_def_property_ui_text(prop, "Radius", "Radius of bounding sphere and material physics");
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
 
@@ -1650,39 +1718,67 @@ static void rna_def_object_game_settings(BlenderRNA *brna)
        prop = RNA_def_property(srna, "damping", PROP_FLOAT, PROP_NONE);
        RNA_def_property_float_sdna(prop, NULL, "damping");
        RNA_def_property_range(prop, 0.0, 1.0);
+       RNA_def_property_float_default(prop, 0.04f);
        RNA_def_property_ui_text(prop, "Damping", "General movement damping");
 
        prop = RNA_def_property(srna, "rotation_damping", PROP_FLOAT, PROP_NONE);
        RNA_def_property_float_sdna(prop, NULL, "rdamping");
        RNA_def_property_range(prop, 0.0, 1.0);
+       RNA_def_property_float_default(prop, 0.1f);
        RNA_def_property_ui_text(prop, "Rotation Damping", "General rotation damping");
 
-       prop = RNA_def_property(srna, "velocity_min", PROP_FLOAT, PROP_NONE);
+       prop = RNA_def_property(srna, "velocity_min", PROP_FLOAT, PROP_DISTANCE);
        RNA_def_property_float_sdna(prop, NULL, "min_vel");
        RNA_def_property_range(prop, 0.0, 1000.0);
-       RNA_def_property_ui_text(prop, "Velocity Min", "Clamp velocity to this minimum speed (except when totally still)");
+       RNA_def_property_ui_text(prop, "Velocity Min", "Clamp velocity to this minimum speed (except when totally still), "
+                                "in distance per second");
 
-       prop = RNA_def_property(srna, "velocity_max", PROP_FLOAT, PROP_NONE);
+       prop = RNA_def_property(srna, "velocity_max", PROP_FLOAT, PROP_DISTANCE);
        RNA_def_property_float_sdna(prop, NULL, "max_vel");
        RNA_def_property_range(prop, 0.0, 1000.0);
-       RNA_def_property_ui_text(prop, "Velocity Max", "Clamp velocity to this maximum speed");
+       RNA_def_property_ui_text(prop, "Velocity Max", "Clamp velocity to this maximum speed, "
+                                "in distance per second");
        
+       prop = RNA_def_property(srna, "angular_velocity_min", PROP_FLOAT, PROP_ANGLE);
+       RNA_def_property_float_sdna(prop, NULL, "min_angvel");
+       RNA_def_property_range(prop, 0.0, 1000.0);
+       RNA_def_property_ui_text(prop, "Angular Velocity Min",
+                                "Clamp angular velocity to this minimum speed (except when totally still), "
+                                "in angle per second");
+
+       prop = RNA_def_property(srna, "angular_velocity_max", PROP_FLOAT, PROP_ANGLE);
+       RNA_def_property_float_sdna(prop, NULL, "max_angvel");
+       RNA_def_property_range(prop, 0.0, 1000.0);
+       RNA_def_property_ui_text(prop, "Angular Velocity Max", "Clamp angular velocity to this maximum speed, "
+                                "in angle per second");
+
        /* Character physics */
        prop = RNA_def_property(srna, "step_height", PROP_FLOAT, PROP_NONE);
        RNA_def_property_float_sdna(prop, NULL, "step_height");
        RNA_def_property_range(prop, 0.01, 1.0);
+       RNA_def_property_float_default(prop, 0.15f);
        RNA_def_property_ui_text(prop, "Step Height", "Maximum height of steps the character can run over");
 
        prop = RNA_def_property(srna, "jump_speed", PROP_FLOAT, PROP_NONE);
        RNA_def_property_float_sdna(prop, NULL, "jump_speed");
        RNA_def_property_range(prop, 0.0, 1000.0);
+       RNA_def_property_float_default(prop, 10.0f);
        RNA_def_property_ui_text(prop, "Jump Force", "Upward velocity applied to the character when jumping");
 
        prop = RNA_def_property(srna, "fall_speed", PROP_FLOAT, PROP_NONE);
        RNA_def_property_float_sdna(prop, NULL, "fall_speed");
        RNA_def_property_range(prop, 0.0, 1000.0);
+       RNA_def_property_float_default(prop, 55.0f);
        RNA_def_property_ui_text(prop, "Fall Speed Max", "Maximum speed at which the character will fall");
 
+       prop = RNA_def_property(srna, "max_jumps", PROP_INT, PROP_NONE);
+       RNA_def_property_int_sdna(prop, NULL, "max_jumps");
+       RNA_def_property_range(prop, 1, CHAR_MAX);
+       RNA_def_property_ui_range(prop, 1, 10, 1, 1);
+       RNA_def_property_int_default(prop, 1);
+       RNA_def_property_ui_text(prop, "Max Jumps",
+                                "The maximum number of jumps the character can make before it hits the ground");
+
        /* Collision Masks */
        prop = RNA_def_property(srna, "collision_group", PROP_BOOLEAN, PROP_LAYER_MEMBER);
        RNA_def_property_boolean_sdna(prop, NULL, "col_group", 1);
@@ -1741,13 +1837,14 @@ static void rna_def_object_game_settings(BlenderRNA *brna)
        prop = RNA_def_property(srna, "form_factor", PROP_FLOAT, PROP_NONE);
        RNA_def_property_float_sdna(prop, NULL, "formfactor");
        RNA_def_property_range(prop, 0.0, 1.0);
+       RNA_def_property_float_default(prop, 0.4f);
        RNA_def_property_ui_text(prop, "Form Factor", "Form factor scales the inertia tensor");
 
        prop = RNA_def_property(srna, "use_anisotropic_friction", PROP_BOOLEAN, PROP_NONE);
        RNA_def_property_boolean_sdna(prop, NULL, "gameflag", OB_ANISOTROPIC_FRICTION);
        RNA_def_property_ui_text(prop, "Anisotropic Friction", "Enable anisotropic friction");
 
-       prop = RNA_def_property(srna, "friction_coefficients", PROP_FLOAT, PROP_NONE);
+       prop = RNA_def_property(srna, "friction_coefficients", PROP_FLOAT, PROP_XYZ);
        RNA_def_property_float_sdna(prop, NULL, "anisotropicFriction");
        RNA_def_property_range(prop, 0.0, 1.0);
        RNA_def_property_ui_text(prop, "Friction Coefficients",
@@ -1763,7 +1860,7 @@ static void rna_def_object_game_settings(BlenderRNA *brna)
        RNA_def_property_enum_sdna(prop, NULL, "collision_boundtype");
        RNA_def_property_enum_items(prop, collision_bounds_items);
        RNA_def_property_enum_funcs(prop, NULL, NULL, "rna_Object_collision_bounds_itemf");
-       RNA_def_property_ui_text(prop, "Collision Bounds",  "Select the collision type");
+       RNA_def_property_ui_text(prop, "Collision Shape",  "Select the collision shape that better fits the object");
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
 
        prop = RNA_def_property(srna, "use_collision_compound", PROP_BOOLEAN, PROP_NONE);
@@ -1773,6 +1870,7 @@ static void rna_def_object_game_settings(BlenderRNA *brna)
        prop = RNA_def_property(srna, "collision_margin", PROP_FLOAT, PROP_NONE | PROP_UNIT_LENGTH);
        RNA_def_property_float_sdna(prop, NULL, "margin");
        RNA_def_property_range(prop, 0.0, 1.0);
+       RNA_def_property_float_default(prop, 0.04f);
        RNA_def_property_ui_text(prop, "Collision Margin",
                                 "Extra margin around object for collision detection, small amount required "
                                 "for stability");
@@ -1788,6 +1886,7 @@ static void rna_def_object_game_settings(BlenderRNA *brna)
        prop = RNA_def_property(srna, "obstacle_radius", PROP_FLOAT, PROP_NONE | PROP_UNIT_LENGTH);
        RNA_def_property_float_sdna(prop, NULL, "obstacleRad");
        RNA_def_property_range(prop, 0.0, 1000.0);
+       RNA_def_property_float_default(prop, 1.0f);
        RNA_def_property_ui_text(prop, "Obstacle Radius", "Radius of object representation in obstacle simulation");
        
        /* state */
@@ -2004,6 +2103,54 @@ static void rna_def_object_vertex_groups(BlenderRNA *brna, PropertyRNA *cprop)
 }
 
 
+static void rna_def_object_lodlevel(BlenderRNA *brna)
+{
+       StructRNA *srna;
+       PropertyRNA *prop;
+
+       srna = RNA_def_struct(brna, "LodLevel", NULL);
+       RNA_def_struct_sdna(srna, "LodLevel");
+
+       prop = RNA_def_property(srna, "distance", PROP_FLOAT, PROP_DISTANCE);
+       RNA_def_property_float_sdna(prop, NULL, "distance");
+       RNA_def_property_range(prop, 0.0, FLT_MAX);
+       RNA_def_property_ui_text(prop, "Distance", "Distance to begin using this level of detail");
+       RNA_def_property_update(prop, NC_OBJECT | ND_LOD, "rna_Object_lod_distance_update");
+
+       prop = RNA_def_property(srna, "object_hysteresis_percentage", PROP_INT, PROP_PERCENTAGE);
+       RNA_def_property_int_sdna(prop, NULL, "obhysteresis");
+       RNA_def_property_range(prop, 0, 100);
+       RNA_def_property_ui_range(prop, 0, 100, 10, 1);
+       RNA_def_property_ui_text(prop, "Hysteresis %",
+                                "Minimum distance change required to transition to the previous level of detail");
+       RNA_def_property_update(prop, NC_OBJECT | ND_LOD, NULL);
+
+       prop = RNA_def_property(srna, "object", PROP_POINTER, PROP_NONE);
+       RNA_def_property_pointer_sdna(prop, NULL, "source");
+       RNA_def_property_struct_type(prop, "Object");
+       RNA_def_property_flag(prop, PROP_EDITABLE);
+       RNA_def_property_ui_text(prop, "Object", "Object to use for this level of detail");
+       RNA_def_property_update(prop, NC_OBJECT | ND_LOD, NULL);
+
+       prop = RNA_def_property(srna, "use_mesh", PROP_BOOLEAN, PROP_NONE);
+       RNA_def_property_boolean_sdna(prop, NULL, "flags", OB_LOD_USE_MESH);
+       RNA_def_property_ui_text(prop, "Use Mesh", "Use the mesh from this object at this level of detail");
+       RNA_def_property_ui_icon(prop, ICON_MESH_DATA, 0);
+       RNA_def_property_update(prop, NC_OBJECT | ND_LOD, NULL);
+
+       prop = RNA_def_property(srna, "use_material", PROP_BOOLEAN, PROP_NONE);
+       RNA_def_property_boolean_sdna(prop, NULL, "flags", OB_LOD_USE_MAT);
+       RNA_def_property_ui_text(prop, "Use Material", "Use the material from this object at this level of detail");
+       RNA_def_property_ui_icon(prop, ICON_MATERIAL, 0);
+       RNA_def_property_update(prop, NC_OBJECT | ND_LOD, NULL);
+
+       prop = RNA_def_property(srna, "use_object_hysteresis", PROP_BOOLEAN, PROP_NONE);
+       RNA_def_property_boolean_sdna(prop, NULL, "flags", OB_LOD_USE_HYST);
+       RNA_def_property_ui_text(prop, "Hysteresis Override", "Override LoD Hysteresis scene setting for this LoD level");
+       RNA_def_property_update(prop, NC_OBJECT | ND_LOD, NULL);
+}
+
+
 static void rna_def_object(BlenderRNA *brna)
 {
        StructRNA *srna;
@@ -2030,6 +2177,7 @@ static void rna_def_object(BlenderRNA *brna)
                {OB_BOUND_SPHERE, "SPHERE", 0, "Sphere", "Draw bounds as sphere"},
                {OB_BOUND_CYLINDER, "CYLINDER", 0, "Cylinder", "Draw bounds as cylinder"},
                {OB_BOUND_CONE, "CONE", 0, "Cone", "Draw bounds as cone"},
+               {OB_BOUND_CAPSULE, "CAPSULE", 0, "Capsule", "Draw bounds as capsule"},
                {0, NULL, 0, NULL, NULL}
        };
 
@@ -2172,6 +2320,7 @@ static void rna_def_object(BlenderRNA *brna)
        RNA_def_property_pointer_funcs(prop, "rna_Object_active_material_get",
                                       "rna_Object_active_material_set", NULL, NULL);
        RNA_def_property_flag(prop, PROP_EDITABLE);
+       RNA_def_property_editable_func(prop, "rna_Object_active_material_editable");
        RNA_def_property_ui_text(prop, "Active Material", "Active material being displayed");
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, "rna_MaterialSlot_update");
 
@@ -2225,6 +2374,7 @@ static void rna_def_object(BlenderRNA *brna)
        
        prop = RNA_def_property(srna, "scale", PROP_FLOAT, PROP_XYZ);
        RNA_def_property_float_sdna(prop, NULL, "size");
+       RNA_def_property_flag(prop, PROP_PROPORTIONAL);
        RNA_def_property_editable_array_func(prop, "rna_Object_scale_editable");
        RNA_def_property_ui_range(prop, -FLT_MAX, FLT_MAX, 1, 3);
        RNA_def_property_float_array_default(prop, default_scale);
@@ -2234,7 +2384,7 @@ static void rna_def_object(BlenderRNA *brna)
        prop = RNA_def_property(srna, "dimensions", PROP_FLOAT, PROP_XYZ_LENGTH);
        RNA_def_property_array(prop, 3);
        RNA_def_property_float_funcs(prop, "rna_Object_dimensions_get", "rna_Object_dimensions_set", NULL);
-       RNA_def_property_ui_range(prop, -FLT_MAX, FLT_MAX, 1, 3);
+       RNA_def_property_ui_range(prop, 0.0f, FLT_MAX, 1, 3);
        RNA_def_property_ui_text(prop, "Dimensions", "Absolute bounding box dimensions of the object");
        RNA_def_property_update(prop, NC_OBJECT | ND_TRANSFORM, "rna_Object_internal_update");
        
@@ -2271,6 +2421,7 @@ static void rna_def_object(BlenderRNA *brna)
 
        prop = RNA_def_property(srna, "delta_scale", PROP_FLOAT, PROP_XYZ);
        RNA_def_property_float_sdna(prop, NULL, "dscale");
+       RNA_def_property_flag(prop, PROP_PROPORTIONAL);
        RNA_def_property_ui_range(prop, -FLT_MAX, FLT_MAX, 1, 3);
        RNA_def_property_float_array_default(prop, default_scale);
        RNA_def_property_ui_text(prop, "Delta Scale", "Extra scaling added to the scale of the object");
@@ -2322,7 +2473,9 @@ static void rna_def_object(BlenderRNA *brna)
        prop = RNA_def_property(srna, "matrix_local", PROP_FLOAT, PROP_MATRIX);
        RNA_def_property_multi_array(prop, 2, rna_matrix_dimsize_4x4);
        RNA_def_property_clear_flag(prop, PROP_ANIMATABLE);
-       RNA_def_property_ui_text(prop, "Local Matrix", "Parent relative transformation matrix");
+       RNA_def_property_ui_text(prop, "Local Matrix", "Parent relative transformation matrix - "
+                                "WARNING: Only takes into account 'Object' parenting, so e.g. in case of bone parenting "
+                                "you get a matrix relative to the Armature object, not to the actual parent bone");
        RNA_def_property_float_funcs(prop, "rna_Object_matrix_local_get", "rna_Object_matrix_local_set", NULL);
        RNA_def_property_update(prop, NC_OBJECT | ND_TRANSFORM, NULL);
 
@@ -2352,7 +2505,7 @@ static void rna_def_object(BlenderRNA *brna)
        prop = RNA_def_property(srna, "constraints", PROP_COLLECTION, PROP_NONE);
        RNA_def_property_struct_type(prop, "Constraint");
        RNA_def_property_ui_text(prop, "Constraints", "Constraints affecting the transformation of the object");
-/*     RNA_def_property_collection_funcs(prop, 0, 0, 0, 0, 0, 0, 0, "constraints__add", "constraints__remove"); */
+/*     RNA_def_property_collection_funcs(prop, NULL, NULL, NULL, NULL, NULL, NULL, NULL, "constraints__add", "constraints__remove"); */
        rna_def_object_constraints(brna, prop);
 
        /* game engine */
@@ -2373,6 +2526,7 @@ static void rna_def_object(BlenderRNA *brna)
        prop = RNA_def_property(srna, "empty_draw_type", PROP_ENUM, PROP_NONE);
        RNA_def_property_enum_sdna(prop, NULL, "empty_drawtype");
        RNA_def_property_enum_items(prop, object_empty_drawtype_items);
+       RNA_def_property_enum_funcs(prop, NULL, "rna_Object_empty_draw_type_set", NULL);
        RNA_def_property_ui_text(prop, "Empty Display Type", "Viewport display style for empties");
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
 
@@ -2383,17 +2537,24 @@ static void rna_def_object(BlenderRNA *brna)
        RNA_def_property_ui_text(prop, "Empty Display Size", "Size of display for empties in the viewport");
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
 
-       prop = RNA_def_property(srna, "empty_image_offset", PROP_FLOAT, PROP_DISTANCE);
+       prop = RNA_def_property(srna, "empty_image_offset", PROP_FLOAT, PROP_NONE);
        RNA_def_property_float_sdna(prop, NULL, "ima_ofs");
        RNA_def_property_ui_text(prop, "Origin Offset", "Origin offset distance");
        RNA_def_property_ui_range(prop, -FLT_MAX, FLT_MAX, 0.1f, 2);
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
 
+       prop = RNA_def_property(srna, "image_user", PROP_POINTER, PROP_NONE);
+       RNA_def_property_flag(prop, PROP_NEVER_NULL);
+       RNA_def_property_pointer_sdna(prop, NULL, "iuser");
+       RNA_def_property_ui_text(prop, "Image User",
+                                "Parameters defining which layer, pass and frame of the image is displayed");
+       RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
+
        /* render */
        prop = RNA_def_property(srna, "pass_index", PROP_INT, PROP_UNSIGNED);
        RNA_def_property_int_sdna(prop, NULL, "index");
        RNA_def_property_ui_text(prop, "Pass Index", "Index number for the IndexOB render pass");
-       RNA_def_property_update(prop, NC_OBJECT, NULL);
+       RNA_def_property_update(prop, NC_OBJECT, "rna_Object_internal_update");
        
        prop = RNA_def_property(srna, "color", PROP_FLOAT, PROP_COLOR);
        RNA_def_property_float_sdna(prop, NULL, "col");
@@ -2477,11 +2638,11 @@ static void rna_def_object(BlenderRNA *brna)
        RNA_def_property_update(prop, NC_OBJECT | ND_TRANSFORM, "rna_Object_internal_update");
        
        /* depsgraph hack */
-       prop = RNA_def_property(srna, "extra_recalc_object", PROP_BOOLEAN, PROP_NONE);
+       prop = RNA_def_property(srna, "use_extra_recalc_object", PROP_BOOLEAN, PROP_NONE);
        RNA_def_property_boolean_sdna(prop, NULL, "depsflag", OB_DEPS_EXTRA_OB_RECALC);
        RNA_def_property_ui_text(prop, "Extra Object Update", "Refresh this object again on frame changes, dependency graph hack");
        
-       prop = RNA_def_property(srna, "extra_recalc_data", PROP_BOOLEAN, PROP_NONE);
+       prop = RNA_def_property(srna, "use_extra_recalc_data", PROP_BOOLEAN, PROP_NONE);
        RNA_def_property_boolean_sdna(prop, NULL, "depsflag", OB_DEPS_EXTRA_DATA_RECALC);
        RNA_def_property_ui_text(prop, "Extra Data Update", "Refresh this object's data again on frame changes, dependency graph hack");
        
@@ -2512,7 +2673,7 @@ static void rna_def_object(BlenderRNA *brna)
        RNA_def_property_float_sdna(prop, NULL, "dupfacesca");
        RNA_def_property_range(prop, 0.001f, 10000.0f);
        RNA_def_property_ui_text(prop, "Dupli Faces Scale", "Scale the DupliFace objects");
-       RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
+       RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, "rna_Object_internal_update");
 
        prop = RNA_def_property(srna, "dupli_group", PROP_POINTER, PROP_NONE);
        RNA_def_property_pointer_sdna(prop, NULL, "dup_group");
@@ -2536,14 +2697,14 @@ static void rna_def_object(BlenderRNA *brna)
        prop = RNA_def_property(srna, "dupli_frames_on", PROP_INT, PROP_NONE | PROP_UNIT_TIME);
        RNA_def_property_int_sdna(prop, NULL, "dupon");
        RNA_def_property_range(prop, MINFRAME, MAXFRAME);
-       RNA_def_property_ui_range(prop, 1, 1500, 1, 0);
+       RNA_def_property_ui_range(prop, 1, 1500, 1, -1);
        RNA_def_property_ui_text(prop, "Dupli Frames On", "Number of frames to use between DupOff frames");
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, "rna_Object_internal_update");
 
        prop = RNA_def_property(srna, "dupli_frames_off", PROP_INT, PROP_NONE | PROP_UNIT_TIME);
        RNA_def_property_int_sdna(prop, NULL, "dupoff");
        RNA_def_property_range(prop, 0, MAXFRAME);
-       RNA_def_property_ui_range(prop, 0, 1500, 1, 0);
+       RNA_def_property_ui_range(prop, 0, 1500, 1, -1);
        RNA_def_property_ui_text(prop, "Dupli Frames Off", "Recurring frames to exclude from the Dupliframes");
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, "rna_Object_internal_update");
 
@@ -2561,7 +2722,7 @@ static void rna_def_object(BlenderRNA *brna)
        RNA_def_property_enum_sdna(prop, NULL, "dt");
        RNA_def_property_enum_items(prop, drawtype_items);
        RNA_def_property_ui_text(prop, "Maximum Draw Type",  "Maximum draw type to display object with in viewport");
-       RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
+       RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, "rna_Object_internal_update");
 
        prop = RNA_def_property(srna, "show_bounds", PROP_BOOLEAN, PROP_NONE);
        RNA_def_property_boolean_sdna(prop, NULL, "dtx", OB_DRAWBOUNDOX);
@@ -2614,16 +2775,16 @@ static void rna_def_object(BlenderRNA *brna)
        /* Grease Pencil */
        prop = RNA_def_property(srna, "grease_pencil", PROP_POINTER, PROP_NONE);
        RNA_def_property_pointer_sdna(prop, NULL, "gpd");
-       RNA_def_property_flag(prop, PROP_EDITABLE);
        RNA_def_property_struct_type(prop, "GreasePencil");
+       RNA_def_property_flag(prop, PROP_EDITABLE | PROP_ID_REFCOUNT);
        RNA_def_property_ui_text(prop, "Grease Pencil Data", "Grease Pencil datablock");
        RNA_def_property_update(prop, NC_OBJECT | ND_DRAW, NULL);
        
        /* pose */
        prop = RNA_def_property(srna, "pose_library", PROP_POINTER, PROP_NONE);
        RNA_def_property_pointer_sdna(prop, NULL, "poselib");
-       RNA_def_property_flag(prop, PROP_EDITABLE);
        RNA_def_property_struct_type(prop, "Action");
+       RNA_def_property_flag(prop, PROP_EDITABLE | PROP_ID_REFCOUNT);
        RNA_def_property_ui_text(prop, "Pose Library", "Action used as a pose library for armatures");
 
        prop = RNA_def_property(srna, "pose", PROP_POINTER, PROP_NONE);
@@ -2663,6 +2824,13 @@ static void rna_def_object(BlenderRNA *brna)
        RNA_def_property_clear_flag(prop, PROP_EDITABLE);
        RNA_def_property_ui_text(prop, "Dynamic Topology Sculpting", NULL);
 
+       /* Levels of Detail */
+       prop = RNA_def_property(srna, "lod_levels", PROP_COLLECTION, PROP_NONE);
+       RNA_def_property_collection_sdna(prop, NULL, "lodlevels", NULL);
+       RNA_def_property_struct_type(prop, "LodLevel");
+       RNA_def_property_ui_text(prop, "Level of Detail Levels", "A collection of detail levels to automatically switch between");
+       RNA_def_property_update(prop, NC_OBJECT | ND_LOD, NULL);
+
        RNA_api_object(srna);
 }
 
@@ -2681,12 +2849,6 @@ static void rna_def_dupli_object(BlenderRNA *brna)
        /* RNA_def_property_pointer_funcs(prop, "rna_DupliObject_object_get", NULL, NULL, NULL); */
        RNA_def_property_ui_text(prop, "Object", "Object being duplicated");
 
-       prop = RNA_def_property(srna, "matrix_original", PROP_FLOAT, PROP_MATRIX);
-       RNA_def_property_float_sdna(prop, NULL, "omat");
-       RNA_def_property_multi_array(prop, 2, rna_matrix_dimsize_4x4);
-       RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE);
-       RNA_def_property_ui_text(prop, "Object Matrix", "The original matrix of this object before it was duplicated");
-
        prop = RNA_def_property(srna, "matrix", PROP_FLOAT, PROP_MATRIX);
        RNA_def_property_float_sdna(prop, NULL, "mat");
        RNA_def_property_multi_array(prop, 2, rna_matrix_dimsize_4x4);
@@ -2765,11 +2927,15 @@ static void rna_def_object_base(BlenderRNA *brna)
 void RNA_def_object(BlenderRNA *brna)
 {
        rna_def_object(brna);
+
+       RNA_define_animate_sdna(false);
        rna_def_object_game_settings(brna);
        rna_def_object_base(brna);
        rna_def_vertex_group(brna);
        rna_def_material_slot(brna);
        rna_def_dupli_object(brna);
+       RNA_define_animate_sdna(true);
+       rna_def_object_lodlevel(brna);
 }
 
 #endif